Class: Squared::Workspace::Project::Base

Inherits:
Object
  • Object
show all
Includes:
Rake::DSL, Shell, Common::Format, System, Utils
Defined in:
lib/squared/workspace/project/base.rb

Direct Known Subclasses

Docker, Git

Constant Summary collapse

@@task_desc =
Rake::TaskManager.
0

Constants included from Common

Common::ARG, Common::PATH

Instance Attribute Summary collapse

Class Method Summary collapse

Instance Method Summary collapse

Methods included from Common::Format

#enable_aixterm

Constructor Details

#initialize(workspace, path, name, group: nil, graph: nil, pass: nil, exclude: nil, release: nil, archive: nil, first: {}, last: {}, error: {}, common: , **kwargs) ⇒ Base

Returns a new instance of Base.



65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
# File 'lib/squared/workspace/project/base.rb', line 65

def initialize(workspace, path, name, *, group: nil, graph: nil, pass: nil, exclude: nil, release: nil,
               archive: nil, first: {}, last: {}, error: {}, common: ARG[:COMMON], **kwargs)
  @path = path
  @workspace = workspace
  @name = name.to_s.freeze
  @project = @path.basename.to_s.freeze
  @group = group&.to_s.freeze
  @depend = kwargs[:depend]
  @doc = kwargs[:doc]
  @lint = kwargs[:lint]
  @test = kwargs[:test]
  @copy = kwargs[:copy]
  @clean = kwargs[:clean]
  @version = kwargs[:version]
  @archive = case archive
             when String, Array
               { uri: archive }
             when Hash
               archive
             end
  @release = release
  @envname = @name.gsub(/[^\w]+/, '_').upcase.freeze
  @exception = env_bool(kwargs[:exception], workspace.exception, strict: true)
  @pipe = env_pipe(kwargs[:pipe], workspace.pipe, strict: true)
  @verbose = case verbose
             when nil
               workspace.verbose
             when String
               env_pipe(verbose, workspace.verbose, strict: true, index: true)
             else
               verbose
             end
  @theme = if !@verbose
             {}
           elsif common
             workspace.theme
           else
             __get__(:theme)[:project][to_sym] ||= {}
           end
  @output = []
  @ref = []
  @children = []
  @graph = if graph
             as_a(graph, workspace.prefix ? ->(val) { workspace.task_name(val).to_sym } : :to_sym).freeze
           end
  @pass = (pass ? as_a(pass) : []).freeze
  @exclude = (exclude ? as_a(exclude, :to_sym) : []).freeze
  @events = {
    first: first,
    last: last,
    error: error
  }
  @desc = (@name.include?(':') ? @name.split(':').join(ARG[:SPACE]) : @name).freeze
  @parent = nil
  @global = false
  run_set(kwargs[:run], kwargs[:env], opts: kwargs.fetch(:opts, true))
  initialize_ref Base.ref
end

Instance Attribute Details

#dependfileObject (readonly)

Returns the value of attribute dependfile.



62
63
64
# File 'lib/squared/workspace/project/base.rb', line 62

def dependfile
  @dependfile
end

#exceptionObject (readonly)

Returns the value of attribute exception.



62
63
64
# File 'lib/squared/workspace/project/base.rb', line 62

def exception
  @exception
end

#groupObject (readonly)

Returns the value of attribute group.



62
63
64
# File 'lib/squared/workspace/project/base.rb', line 62

def group
  @group
end

#nameObject (readonly)

Returns the value of attribute name.



62
63
64
# File 'lib/squared/workspace/project/base.rb', line 62

def name
  @name
end

#parentObject (readonly)

Returns the value of attribute parent.



62
63
64
# File 'lib/squared/workspace/project/base.rb', line 62

def parent
  @parent
end

#pathObject (readonly)

Returns the value of attribute path.



62
63
64
# File 'lib/squared/workspace/project/base.rb', line 62

def path
  @path
end

#pipeObject (readonly)

Returns the value of attribute pipe.



62
63
64
# File 'lib/squared/workspace/project/base.rb', line 62

def pipe
  @pipe
end

#projectObject (readonly)

Returns the value of attribute project.



62
63
64
# File 'lib/squared/workspace/project/base.rb', line 62

def project
  @project
end

#themeObject (readonly)

Returns the value of attribute theme.



62
63
64
# File 'lib/squared/workspace/project/base.rb', line 62

def theme
  @theme
end

#verboseObject (readonly)

Returns the value of attribute verbose.



62
63
64
# File 'lib/squared/workspace/project/base.rb', line 62

def verbose
  @verbose
end

#workspaceObject (readonly)

Returns the value of attribute workspace.



62
63
64
# File 'lib/squared/workspace/project/base.rb', line 62

def workspace
  @workspace
end

Class Method Details

.aliasargsObject



26
# File 'lib/squared/workspace/project/base.rb', line 26

def aliasargs(*); end

.as_path(val) ⇒ Object



33
34
35
36
37
38
39
40
# File 'lib/squared/workspace/project/base.rb', line 33

def as_path(val)
  case val
  when Pathname
    val
  when String
    Pathname.new(val)
  end
end

.bannerargsObject



27
# File 'lib/squared/workspace/project/base.rb', line 27

def bannerargs(*); end

.batchargsObject



25
# File 'lib/squared/workspace/project/base.rb', line 25

def batchargs(*); end

.config?Boolean

Returns:

  • (Boolean)


46
47
48
# File 'lib/squared/workspace/project/base.rb', line 46

def config?(*)
  false
end

.populateObject



24
# File 'lib/squared/workspace/project/base.rb', line 24

def populate(*); end

.refObject



42
43
44
# File 'lib/squared/workspace/project/base.rb', line 42

def ref
  @ref ||= to_s.downcase.to_sym
end

.tasksObject



29
30
31
# File 'lib/squared/workspace/project/base.rb', line 29

def tasks
  (%i[build archive graph] + BLK_SET).freeze
end

.to_sObject



50
51
52
# File 'lib/squared/workspace/project/base.rb', line 50

def to_s
  super[/[^:]+\z/, 0]
end

Instance Method Details

#add(path, name = nil, **kwargs, &blk) ⇒ Object



327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
# File 'lib/squared/workspace/project/base.rb', line 327

def add(path, name = nil, **kwargs, &blk)
  checkdir = lambda do |val|
    if val.directory? && !val.empty?
      true
    else
      log&.warn "workspace \"#{val}\" (#{val.empty? ? 'empty' : 'not found'})"
      false
    end
  end
  if path.is_a?(String) && (seg = path[%r{^(.+)[\\/]\*+$}, 1])
    return self unless checkdir.(path = basepath(seg))

    path = path.children.select { |val| checkdir.(val) }
  end
  if path.is_a?(Array)
    name = @name if name == true
    path.each { |val| add(val, name && task_join(name, File.basename(val)), **kwargs, &blk) }
    return self
  elsif !projectpath?(path = basepath(path)) || !checkdir.(path)
    return self
  elsif name.is_a?(Symbol)
    name = name.to_s
  elsif !name.is_a?(String)
    name = nil
  end
  if @withargs
    data = @withargs.dup
    data.merge!(kwargs)
    kwargs = data
  end
  kwargs[:group] = group unless kwargs.key?(:group)
  kwargs[:ref] = ref unless kwargs.key?(:ref)
  parent = self
  proj = nil
  workspace.add(path, name || path.basename, **kwargs) do
    variable_set :parent, parent
    proj = self
  end
  @children << proj
  proj.instance_eval(&blk) if block_given?
  self
end

#allrefObject



816
817
818
# File 'lib/squared/workspace/project/base.rb', line 816

def allref
  @ref.reverse_each
end

#archive(sync: invoked_sync?('archive')) ⇒ Object



443
444
445
446
447
# File 'lib/squared/workspace/project/base.rb', line 443

def archive(*, sync: invoked_sync?('archive'), **)
  return unless @archive.is_a?(Array)

  unpack(path, **@archive, sync: sync, from: :archive)
end

#archive?Boolean

Returns:

  • (Boolean)


756
757
758
# File 'lib/squared/workspace/project/base.rb', line 756

def archive?
  @archive.is_a?(Hash) && (!path.exist? || path.empty?)
end

#as(cmd, script, to = nil) ⇒ Object



675
676
677
678
679
680
# File 'lib/squared/workspace/project/base.rb', line 675

def as(cmd, script, to = nil)
  script = { "#{script}": to } if to
  data = (@as ||= {})[cmd.to_sym] ||= {}
  script.each { |key, val| data[key.to_s] = val }
  self
end

#basepath(*args) ⇒ Object



820
821
822
# File 'lib/squared/workspace/project/base.rb', line 820

def basepath(*args)
  path.join(*args)
end

#build(*args, sync: invoked_sync?('build'), from: @buildtype || :build) ⇒ Object



382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
# File 'lib/squared/workspace/project/base.rb', line 382

def build(*args, sync: invoked_sync?('build'), from: @buildtype || :build, **)
  banner = verbose
  if args.empty?
    return unless from == (@buildtype || :build)

    run_b(@run, sync: sync, from: from) if series?(@run)
    args = @output
    banner = verbose == 1 if task_invoked?('build', 'build:sync')
  end
  if args.all? { |val| val.is_a?(Array) }
    cmd = []
    var = {}
    args.each do |val|
      a, b, c, d, e = val
      case b
      when Hash
        b = append_hash(b, build: true).join(' ')
      when Enumerable
        b = b.to_a.join(' ')
      end
      d = append_hash(d).join(' ') if d.is_a?(Hash)
      if a
        cmd << [a, d, b].compact.join(' ')
      else
        next unless respond_to?(:compose)

        cmd << a if (a = compose(as_get(b), d, script: true, args: e, from: from))
      end
      var.merge!(c) if c.is_a?(Hash)
    end
    cmd = cmd.join(' && ')
  else
    cmd, opts, var, flags, extra = args
  end
  if cmd
    cmd = as_get(cmd)
    opts = compose(opts, script: false) if opts && respond_to?(:compose)
    flags = append_hash(flags).join(' ') if flags.is_a?(Hash)
    case opts
    when Hash
      opts = append_hash(opts, build: true)
      cmd = as_a(cmd).push(flags).concat(opts).compact.join(' ')
    when Enumerable
      cmd = as_a(cmd).concat(opts.to_a)
      cmd.map! { |val| "#{val} #{flags}" } if flags
      cmd = cmd.join(' && ')
    else
      cmd = [cmd, flags, opts].compact.join(' ') if opts || flags
    end
  else
    return unless (opts || extra) && respond_to?(:compose)

    cmd = compose(as_get(opts), flags, script: true, args: extra, from: from)
  end
  run(cmd, var, from: from, banner: banner, sync: sync)
end

#build?Boolean

Returns:

  • (Boolean)


744
745
746
# File 'lib/squared/workspace/project/base.rb', line 744

def build?
  !!@output[0] || script? || series?(@run)
end

#clean(sync: invoked_sync?('clean')) ⇒ Object



465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
# File 'lib/squared/workspace/project/base.rb', line 465

def clean(*, sync: invoked_sync?('clean'), **)
  return unless @clean

  on :first, :clean
  case @clean
  when String
    run_s(@clean, sync: sync)
  when Hash
    begin
      @clean.each { |cmd, opts| build(cmd.to_s, opts, sync: sync) }
    rescue StandardError => e
      log&.error e
      ret = on(:error, from, e)
      raise if exception && ret != true
    end
  else
    if @clean.is_a?(Enumerable) && !series?(@clean)
      as_a(@clean).each do |val|
        val = val.to_s
        path = basepath(val)
        if path.directory? && val.match?(%r{[\\/]$})
          log&.warn "rm -rf #{path}"
          path.rmtree(verbose: verbose)
        else
          log&.warn "rm #{path}"
          (val.include?('*') ? Dir[path] : [path]).each do |file|
            next unless File.file?(file)

            begin
              File.delete(file)
            rescue StandardError => e
              log&.error e
            end
          end
        end
      end
    else
      run_b(@clean, sync: sync)
    end
  end
  on :last, :clean
end

#clean?Boolean

Returns:

  • (Boolean)


780
781
782
# File 'lib/squared/workspace/project/base.rb', line 780

def clean?
  runnable?(@clean) || workspace.task_defined?(name, 'clean')
end

#copy(sync: invoked_sync?('copy')) ⇒ Object



461
462
463
# File 'lib/squared/workspace/project/base.rb', line 461

def copy(*, sync: invoked_sync?('copy'), **)
  run_b(@copy, sync: sync, from: :copy)
end

#copy?Boolean

Returns:

  • (Boolean)


764
765
766
# File 'lib/squared/workspace/project/base.rb', line 764

def copy?
  runnable?(@copy) || workspace.task_defined?(name, 'copy')
end

#depend(sync: invoked_sync?('depend')) ⇒ Object



439
440
441
# File 'lib/squared/workspace/project/base.rb', line 439

def depend(*, sync: invoked_sync?('depend'), **)
  run_b(@depend, sync: sync, from: :depend)
end

#depend?Boolean

Returns:

  • (Boolean)


752
753
754
# File 'lib/squared/workspace/project/base.rb', line 752

def depend?
  !!@depend
end

#dependtypeObject



806
807
808
# File 'lib/squared/workspace/project/base.rb', line 806

def dependtype(*)
  @dependindex ? @dependindex.succ : 0
end

#dev?Boolean

Returns:

  • (Boolean)


784
785
786
# File 'lib/squared/workspace/project/base.rb', line 784

def dev?
  @dev != false && workspace.dev?(pat: @dev, **scriptargs)
end

#doc(sync: invoked_sync?('doc')) ⇒ Object



449
450
451
# File 'lib/squared/workspace/project/base.rb', line 449

def doc(*, sync: invoked_sync?('doc'), **)
  run_b(@doc, sync: sync, from: :doc)
end

#doc?Boolean

Returns:

  • (Boolean)


768
769
770
# File 'lib/squared/workspace/project/base.rb', line 768

def doc?
  !!@doc
end

#enabled?(ref = nil) ⇒ Boolean

Returns:

  • (Boolean)


728
729
730
731
732
# File 'lib/squared/workspace/project/base.rb', line 728

def enabled?(ref = nil, **)
  return false if ref && !ref?(ref)

  (path.directory? && !path.empty?) || archive?
end

#error(key, *args, **kwargs, &blk) ⇒ Object



660
661
662
# File 'lib/squared/workspace/project/base.rb', line 660

def error(key, *args, **kwargs, &blk)
  event(:error, key, *args, **kwargs, &blk)
end

#event(name, key, *args, override: false, **kwargs, &blk) ⇒ Object



664
665
666
667
668
669
670
671
672
673
# File 'lib/squared/workspace/project/base.rb', line 664

def event(name, key, *args, override: false, **kwargs, &blk)
  data = @events[name.to_sym] ||= {}
  items = if override
            data[key.to_sym] = []
          else
            data[key.to_sym] ||= []
          end
  items << [block_given? ? [blk] + args : args, kwargs]
  self
end

#exclude?(*refs) ⇒ Boolean

Returns:

  • (Boolean)


792
793
794
795
796
# File 'lib/squared/workspace/project/base.rb', line 792

def exclude?(*refs)
  return false if @exclude.empty?

  refs.flatten.any? { |ref| @exclude.include?(ref) }
end

#first(key, *args, **kwargs, &blk) ⇒ Object



652
653
654
# File 'lib/squared/workspace/project/base.rb', line 652

def first(key, *args, **kwargs, &blk)
  event(:first, key, *args, **kwargs, &blk)
end

#generate(keys) ⇒ Object



314
315
316
# File 'lib/squared/workspace/project/base.rb', line 314

def generate(keys, **)
  task_build keys
end

#graph(start = [], tasks = nil, sync: invoked_sync?('graph'), pass: [], out: nil) ⇒ Object



508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
# File 'lib/squared/workspace/project/base.rb', line 508

def graph(start = [], tasks = nil, sync: invoked_sync?('graph'), pass: [], out: nil)
  if (val = env('GRAPH', strict: true))
    tasks ||= []
    split_escape(val).each do |task|
      if ref?(task.to_sym) && (script = workspace.script_get(:graph, ref: task.to_sym))
        tasks += script[:graph]
      else
        tasks << task
      end
    end
  end
  pass += split_escape(val) if (val = env('GRAPH', suffix: 'PASS'))
  start, neg = start.partition { |name| !name.start_with?('-') }
  data = graph_collect(self, start, pass: neg.map { |name| name[1..-1] })
  unless out
    data[name] << self
    on :first, :graph
  end
  begin
    done = graph_branch(self, data, tasks, out, sync: sync, pass: pass)
  rescue StandardError => e
    ret = on(:error, :graph, e)
    raise unless ret == true
  end
  if out
    [out, done]
  else
    on :last, :graph
  end
end

#graph?Boolean

Returns:

  • (Boolean)


760
761
762
# File 'lib/squared/workspace/project/base.rb', line 760

def graph?
  @graph.is_a?(Array) && !@graph.empty?
end

#has?(meth, ref = nil) ⇒ Boolean

Returns:

  • (Boolean)


734
735
736
737
738
# File 'lib/squared/workspace/project/base.rb', line 734

def has?(meth, ref = nil)
  return false if ref && !ref?(ref)

  respond_to?(meth = :"#{meth}?") && __send__(meth)
end

#initialize_build(ref, **kwargs) ⇒ Object



128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
# File 'lib/squared/workspace/project/base.rb', line 128

def initialize_build(ref, **kwargs)
  initialize_ref ref
  if (@script = @workspace.script_get(group: @group, ref: ref))
    if @script[:log] && !kwargs.key?(:log)
      kwargs[:log] = @script[:log]
      @log = nil
    end
    @depend = @script[:depend] if @depend.nil?
    @doc = @script[:doc] if @doc.nil?
    @lint = @script[:lint] if @lint.nil?
    @test = @script[:test] if @test.nil?
    @clean = @script[:clean] if @clean.nil?
    @exclude = @script[:exclude] if @exclude.empty? && @script.key?(:exclude)
  end
  initialize_events(ref, **kwargs)
  initialize_logger(**kwargs)
  return if @output[0] == false

  data = @workspace.script_find(*@ref, @group)
  if @output[0].nil?
    if (scr = data[:script])
      @global = true
      script_set(scr, args: data.fetch(:args, kwargs[:args]), prod: kwargs[:prod])
    elsif (run = data[:run])
      @global = true
      run_set run
    end
    unless data[:env]
      if (scr = kwargs[:script])
        @global = false
        script_set(scr, args: kwargs[:args])
      elsif @script && !data[:global]
        if (scr = @script[:script])
          @global = false
          script_set(scr, args: @script.fetch(:args, kwargs[:args]))
        elsif (run = @script[:run])
          @global = false
          run_set run
        end
      end
    end
  elsif data[:env] && data[:run]
    @global = true
    run_set data[:run]
  end
  @global = true
end

#initialize_env(dev: nil, prod: nil) ⇒ Object



218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
# File 'lib/squared/workspace/project/base.rb', line 218

def initialize_env(dev: nil, prod: nil, **)
  @dev = env_match('BUILD', dev, suffix: 'DEV', strict: true)
  @prod = env_match('BUILD', prod, suffix: 'PROD', strict: true)
  unless @output[2] == false || !(val = env('BUILD', suffix: 'ENV'))
    data = parse_json(val, hint: "BUILD_#{@envname}_ENV")
    @output[2] = data if data
  end
  unless @output[0] == false || @output[0].is_a?(Array)
    if (val = env('BUILD', suffix: 'OPTS'))
      n = @output[0] ? 1 : 3
      @output[n] = merge_opts(@output[n], shell_split(val, escape: false))
    end
    if (val = env(ref.to_s.upcase, suffix: 'OPTS'))
      @output[4] = merge_opts(@output[4], shell_split(val, escape: false))
    end
  end
  @version = val if (val = env('BUILD', suffix: 'VERSION'))
  return unless (val = env('BUILD', strict: true))

  @global = false
  if val == '0'
    @output = [false]
  elsif script?
    script_set val
  else
    run_set val
  end
end

#initialize_events(ref) ⇒ Object



176
177
178
179
180
181
182
# File 'lib/squared/workspace/project/base.rb', line 176

def initialize_events(ref, **)
  return unless (events = @workspace.events_get(group: @group, ref: ref))

  events.each do |task, data|
    data.each { |ev, blk| (@events[ev] ||= {})[task] ||= [blk] }
  end
end

#initialize_logger(log: nil) ⇒ Object



184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
# File 'lib/squared/workspace/project/base.rb', line 184

def initialize_logger(log: nil, **)
  return if @log

  log = log.is_a?(Hash) ? log.dup : { file: log }
  unless (file = env('LOG_FILE'))
    file = case env('LOG_AUTO')
           when 'y', 'year'
             "#{@name}-#{Date.today.year}.log"
           when 'm', 'month'
             "#{@name}-#{Date.today.strftime('%Y-%m')}.log"
           when 'd', 'day', '1'
             "#{@name}-#{Date.today}.log"
           end
  end
  if file ||= log[:file]
    file = Date.today.strftime(file)
    file = (dir = env('LOG_DIR')) ? @workspace.home.join(dir, file) : @workspace.home.join(file)
    begin
      file = file.realdirpath
    rescue StandardError => e
      raise if @exception

      file = nil
      warn log_message(Logger::WARN, e, pass: true) if warning?
    end
  end
  log[:progname] ||= @name
  if (val = env('LOG_LEVEL', ignore: false))
    log[:level] = val.match?(/^\d$/) ? log_sym(val.to_i) : val
  end
  log.delete(:file)
  @log = [file, log]
end

#initialize_ref(ref) ⇒ Object



124
125
126
# File 'lib/squared/workspace/project/base.rb', line 124

def initialize_ref(ref)
  @ref << ref unless @exclude.include?(ref)
end

#inject(obj, *args, **kwargs, &blk) ⇒ Object



370
371
372
373
374
375
376
377
378
379
380
# File 'lib/squared/workspace/project/base.rb', line 370

def inject(obj, *args, **kwargs, &blk)
  return self unless enabled?

  out = obj.link(self, *args, **kwargs, &blk) if obj.respond_to?(:link)
  if !out
    warn log_message(Logger::WARN, 'link not compatible', subject: obj.to_s, hint: name, pass: true)
  elsif out.respond_to?(:build)
    out.build
  end
  self
end

#inspectObject



840
841
842
# File 'lib/squared/workspace/project/base.rb', line 840

def inspect
  "#<#{self.class}: #{name} => #{self}>"
end

#last(key, *args, **kwargs, &blk) ⇒ Object



656
657
658
# File 'lib/squared/workspace/project/base.rb', line 656

def last(key, *args, **kwargs, &blk)
  event(:last, key, *args, **kwargs, &blk)
end

#lint(sync: invoked_sync?('lint')) ⇒ Object



453
454
455
# File 'lib/squared/workspace/project/base.rb', line 453

def lint(*, sync: invoked_sync?('lint'), **)
  run_b(@lint, sync: sync, from: :lint)
end

#lint?Boolean

Returns:

  • (Boolean)


772
773
774
# File 'lib/squared/workspace/project/base.rb', line 772

def lint?
  !!@lint
end

#localnameObject



836
837
838
# File 'lib/squared/workspace/project/base.rb', line 836

def localname
  workspace.task_localname(name)
end

#logObject



810
811
812
813
814
# File 'lib/squared/workspace/project/base.rb', line 810

def log
  return @log unless @log.is_a?(Array)

  @log = Logger.new(enabled? ? @log[0] : nil, **@log[1])
end

#populate(keys) ⇒ Object



251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
# File 'lib/squared/workspace/project/base.rb', line 251

def populate(keys, **)
  task_build keys
  return unless ref?(Base.ref)

  namespace name do
    @@tasks[Base.ref].each do |action, flags|
      next if @pass.include?(action)

      namespace action do
        flags.each do |flag|
          case action
          when 'graph'
            next unless graph?

            format_desc action, flag, '(-)project*'
            task flag do |_, args|
              args = args.to_a.reject { |val| name == val.to_s }
              if flag == :run
                graph args
              else
                out, done = graph(args, out: [])
                out.map! do |val|
                  done.each_with_index do |proj, i|
                    next unless val.match?(/ #{Regexp.escape(proj.name)}(?:@\d|\z)/)

                    val += " (#{i.succ})"
                    break
                  end
                  val
                end
                emphasize(out, title: path, right: true, border: borderstyle, sub: [
                  { pat: /\A(#{Regexp.escape(path.to_s)})(.*)\z/, styles: theme[:header] },
                  { pat: /\A(#{Regexp.escape(name)})(.*)\z/, styles: theme[:active] },
                  { pat: /\A(.+ )(\()(\d+)(\))(.*)\z/, styles: theme[:inline], index: 3 }
                ])
              end
            end
          when 'unpack'
            format_desc action, flag, 'tag/url,dir,digest?,f/force?'
            task flag, [:tag, :dir, :digest, :force] do |_, args|
              tag = param_guard(action, flag, args: args, key: :tag)
              dir = param_guard(action, flag, args: args, key: :dir)
              unless tag.match?(URI_SCHEME)
                raise_error 'no base uri' unless @release

                tag = "#{@release.include?('??') ? @release.sub('??', tag) : @release + tag}.#{flag}"
              end
              case (digest = args.digest)
              when 'f', 'force'
                digest = nil
                force = true
              else
                force = args.fetch(:force, false)
              end
              unpack(basepath(dir), uri: tag, digest: digest, ext: flag.to_s, force: force)
            end
          end
        end
      end
    end
  end
end

#prod?Boolean

Returns:

  • (Boolean)


788
789
790
# File 'lib/squared/workspace/project/base.rb', line 788

def prod?
  @prod != false && workspace.prod?(pat: @prod, **scriptargs)
end

#refObject



247
248
249
# File 'lib/squared/workspace/project/base.rb', line 247

def ref
  Base.ref
end

#ref?(val) ⇒ Boolean

Returns:

  • (Boolean)


740
741
742
# File 'lib/squared/workspace/project/base.rb', line 740

def ref?(val)
  @ref.include?(val)
end

#rootpath(*args, ascend: nil) ⇒ Object



824
825
826
827
828
829
830
831
832
833
834
# File 'lib/squared/workspace/project/base.rb', line 824

def rootpath(*args, ascend: nil)
  ret = basepath(*args)
  return ret unless ascend && !ret.exist?

  path.parent.ascend.each do |dir|
    target = dir.join(*args)
    return target if target.exist?
    break if (ascend.is_a?(String) && dir.join(ascend).exist?) || workspace.root == dir || parent&.path == dir
  end
  ret
end

#script?Boolean

Returns:

  • (Boolean)


748
749
750
# File 'lib/squared/workspace/project/base.rb', line 748

def script?
  @output[0].nil? && !!@output[1] && respond_to?(:compose)
end

#series(key, override: false, &blk) ⇒ Object



682
683
684
685
686
687
688
689
690
691
692
693
# File 'lib/squared/workspace/project/base.rb', line 682

def series(key, override: false, &blk)
  if blocks.include?(key.to_sym) && block_given?
    if !override && series?(target = instance_variable_get(:"@#{key}"))
      target << blk
    else
      instance_variable_set :"@#{key}", [blk]
    end
  else
    log&.warn "series: @#{key} (invalid)"
  end
  self
end

#task_include?(key, ref = nil) ⇒ Boolean

Returns:

  • (Boolean)


798
799
800
# File 'lib/squared/workspace/project/base.rb', line 798

def task_include?(key, ref = nil)
  workspace.task_include?(self, key, ref) && !@pass.include?(key.to_s)
end

#test(sync: invoked_sync?('test')) ⇒ Object



457
458
459
# File 'lib/squared/workspace/project/base.rb', line 457

def test(*, sync: invoked_sync?('test'), **)
  run_b(@test, sync: sync, from: :test)
end

#test?Boolean

Returns:

  • (Boolean)


776
777
778
# File 'lib/squared/workspace/project/base.rb', line 776

def test?
  !!@test
end

#to_sObject



844
845
846
# File 'lib/squared/workspace/project/base.rb', line 844

def to_s
  path.to_s
end

#to_symObject



848
849
850
# File 'lib/squared/workspace/project/base.rb', line 848

def to_sym
  name.to_sym
end

#unpack(target, sync: true, uri: nil, digest: nil, ext: nil, force: false, depth: 1, headers: {}, from: :unpack) ⇒ Object



539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
# File 'lib/squared/workspace/project/base.rb', line 539

def unpack(target, sync: true, uri: nil, digest: nil, ext: nil, force: false, depth: 1, headers: {},
           from: :unpack)
  if !target.exist?
    target.mkpath
  elsif !target.directory?
    raise_error('invalid location', hint: target)
  elsif !target.empty?
    raise_error('directory not empty', hint: target) unless force || env('UNPACK_FORCE')
    create = true
  end
  if digest
    if (n = digest.index(':').to_i) > 0
      size = digest[0, n].downcase
      digest = digest[n + 1..-1]
    else
      size = digest.size
    end
    algo = case size
           when 32, 'md5'
             Digest::MD5
           when 'rmd160'
             Digest::RMD160
           when 40, 'sha1'
             Digest::SHA1
           when 64, 'sha256'
             Digest::SHA256
           when 96, 'sha384'
             Digest::SHA384
           when 128, 'sha512'
             Digest::SHA512
           else
             raise_error("invalid checksum: #{digest}", hint: name)
           end
  end
  if (val = env('HEADERS')) && (out = parse_json(val, hint: "HEADERS_#{@envname}"))
    headers = out
  end
  require 'open-uri'
  data = nil
  (uri = as_a(uri)).each_with_index do |url, index|
    last = index == uri.size - 1
    URI.open(url, headers) do |f|
      data = f.read
      if algo && algo.hexdigest(data) != digest
        data = nil
        raise_error("checksum failed: #{digest}", hint: url) if last
      end
      next if ext && index == 0

      case f.content_type
      when 'application/zip'
        ext = 'zip'
      when 'application/x-gzip'
        ext = 'tgz'
      when 'application/x-xz'
        ext = 'txz'
      end
    end
    if data
      uri = url
      break
    elsif last
      raise_error('no content', hint: url)
    end
  end
  ext ||= URI.parse(uri).path[/^.+?\.((?:tar\.)?\w+)$/i, 1]
  if (n = env("#{ext == 'zip' ? 'ZIP' : 'TAR'}_DEPTH", ignore: false))
    depth = n.to_i
  end
  begin
    require 'tempfile'
    file = Tempfile.new("#{name}-")
    file.write(data)
    file.close
    if create
      warn log_message(Logger::WARN, name, 'force remove', hint: target, pass: true)
      target.rmtree(verbose: true)
      target.mkpath
    end
    case ext
    when 'zip', 'aar'
      session 'unzip', shell_quote(file.path), quote_option('d', target)
    when 'tar', 'tgz', 'tar.gz', 'tar.xz'
      flags = +(verbose ? 'v' : '')
      case ext
      when 'tgz', 'tar.gz'
        flags += 'z'
      when 'txz', 'tar.xz'
        flags += 'J'
      end
      session 'tar', "-x#{flags}", basic_option('strip-components', depth), quote_option('f', file.path),
              quote_option('C', target)
      depth = 0
    else
      raise_error("unsupported format: #{ext}", hint: uri)
    end
    run(sync: sync, from: from)
    while depth > 0 && target.children.size == 1
      entry = target.children.first
      break unless entry.directory?

      dest = target.join(File.basename(file.path))
      FileUtils.mv(entry, dest)
      dest.children.each { |child| FileUtils.mv(child, target) }
      dest.rmdir
      target = entry
      depth -= 1
    end
  ensure
    file&.unlink
  end
end

#variable_set(key, *val, **kwargs, &blk) ⇒ Object



695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
# File 'lib/squared/workspace/project/base.rb', line 695

def variable_set(key, *val, **kwargs, &blk)
  if variables.include?(key)
    case key
    when :build, :run
      run_set(*val, **kwargs)
    when :script
      script_set(*val, **kwargs)
    when :env
      run_set(output[0], *val, **kwargs)
    when :parent
      @parent = val if (val = val.first).is_a?(Project::Base)
    when :graph
      @graph = case val.first
               when nil, false
                 nil
               else
                 val.flatten.map!(&:to_s).freeze
               end
    when :dependfile
      @dependfile = basepath(*val)
    else
      if blocks.include?(key) && block_given?
        series key, &blk
      else
        instance_variable_set(:"@#{key}", block_given? && val.empty? ? blk : val.first)
      end
    end
  else
    log&.warn "variable_set: @#{key} (private)"
  end
  self
end

#versionObject



802
803
804
# File 'lib/squared/workspace/project/base.rb', line 802

def version(*)
  @version
end

#with(**kwargs, &blk) ⇒ Object



318
319
320
321
322
323
324
325
# File 'lib/squared/workspace/project/base.rb', line 318

def with(**kwargs, &blk)
  @withargs = kwargs.empty? ? nil : kwargs
  if block_given?
    instance_eval(&blk)
    @withargs = nil
  end
  self
end