Class: DitzStr::Operator

Inherits:
Object show all
Defined in:
lib/ditzstr/brick.rb,
lib/ditzstr/operator.rb

Defined Under Namespace

Classes: Error

Class Method Summary collapse

Instance Method Summary collapse

Class Method Details

.build_args(project, method, args) ⇒ Object

parse the specs, and the commandline arguments, and resolve them. does typechecking but currently doesn’t check for open_issues actually being open, unstarted_issues being unstarted, etc. probably will check for this in the future.

Raises:



32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
# File 'lib/ditzstr/operator.rb', line 32

def build_args project, method, args
  specs = @operations[method][:args_spec]
  command = "command '#{method_to_op method}'"

  if specs.empty? && args == ["<options>"]
    die_with_completions project, method, nil
  end

  built_args = specs.map do |spec|
    optional = spec.to_s =~ /^maybe_/
    spec = spec.to_s.gsub(/^maybe_/, "").intern # :(
    val = args.shift

    case val
    when nil
      next if optional
      specname = spec.to_s.gsub("_", " ")
      article = specname =~ /^[aeiou]/ ? "an" : "a"
      raise Error, "#{command} requires #{article} #{specname}"
    when "<options>"
      die_with_completions project, method, spec
    end

    case spec
    when :issue, :open_issue, :unstarted_issue, :started_issue, :assigned_issue
      ## issue completion sticks the title on there, so this will strip it off
      valr = val.sub(/\A(\w+-\d+)_.*$/,'\1')
      issues = project.issues_for valr
      case issues.size
      when 0; raise Error, "no issue with name #{val.inspect}"
      when 1; issues.first
      else
        raise Error, "multiple issues matching name #{val.inspect}"
      end
    when :release, :unreleased_release
      if val == "unassigned"
        :unassigned
      else
        project.release_for(val) or raise Error, "no release with name #{val}"
      end
    when :component
      project.component_for(val) or raise Error, "no component with name #{val}" if val
    else
      val # no translation for other types
    end
  end

  raise Error, "too many arguments for #{command}" unless args.empty?
  built_args
end

.build_opts(method, args) ⇒ Object



23
24
25
26
# File 'lib/ditzstr/operator.rb', line 23

def build_opts method, args
  options_blk = @operations[method][:options_blk]
  options_blk and options args, &options_blk or nil
end

.die_with_completions(project, method, spec) ⇒ Object



83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
# File 'lib/ditzstr/operator.rb', line 83

def die_with_completions project, method, spec
  puts(case spec
  when :issue, :open_issue, :unstarted_issue, :started_issue, :assigned_issue
    m = { :issue => nil,
          :open_issue => :open?,
          :unstarted_issue => :unstarted?,
          :started_issue => :in_progress?,
          :assigned_issue => :assigned?,
        }[spec]
    project.issues.select { |i| m.nil? || i.send(m) }.sort_by { |i| i.creation_time }.reverse.map { |i| "#{i.name}_#{i.title.gsub(/\W+/, '-')}" }
  when :release
    project.releases.map { |r| r.name } + ["unassigned"]
  when :unreleased_release
    project.releases.select { |r| r.unreleased? }.map { |r| r.name }
  when :component
    project.components.map { |c| c.name }
  when :command
    operations.map { |name, _| name }
  else
    ""
  end)
  exit 0
end

.has_operation?(op) ⇒ Boolean

Returns:

  • (Boolean)


21
# File 'lib/ditzstr/operator.rb', line 21

def has_operation? op; @operations.member? op_to_method(op) end

.method_to_op(meth) ⇒ Object



9
# File 'lib/ditzstr/operator.rb', line 9

def method_to_op meth; meth.to_s.gsub("_", "-") end

.op_to_method(op) ⇒ Object



10
# File 'lib/ditzstr/operator.rb', line 10

def op_to_method op; op.gsub("-", "_").intern end

.operation(method, desc, *args_spec, &options_blk) ⇒ Object



12
13
14
15
16
# File 'lib/ditzstr/operator.rb', line 12

def operation method, desc, *args_spec, &options_blk
  @operations ||= {}
  @operations[method] = { :desc => desc, :args_spec => args_spec,
                          :options_blk => options_blk }
end

.operationsObject



18
19
20
# File 'lib/ditzstr/operator.rb', line 18

def operations
  @operations.map { |k, v| [method_to_op(k), v] }.sort_by { |k, v| k }
end

Instance Method Details

#actually_do_todo(project, config, releases, full) ⇒ Object



336
337
338
339
340
341
342
343
344
345
346
# File 'lib/ditzstr/operator.rb', line 336

def actually_do_todo project, config, releases, full
  releases ||= project.unreleased_releases + [:unassigned]
  releases = [*releases]
  releases.each do |r|
    puts r == :unassigned ? "Unassigned:" : "#{r.name} (#{r.status}):"
    issues = project.issues_for_release r
    issues = issues.select { |i| i.open? } unless full
    puts(todo_list_for(issues.sort_by { |i| i.sort_order }) || "No open issues.")
    puts
  end
end

#add(project, config, opts) ⇒ Object



196
197
198
199
200
201
202
# File 'lib/ditzstr/operator.rb', line 196

def add project, config, opts
  issue = Issue.create_interactively(:args => [config, project]) or return
  issue.log "created", config.user, get_comment(opts)
  project.add_issue issue
  project.assign_issue_names!
  puts "Added issue #{issue.name}."
end

#add_component(project, config) ⇒ Object



223
224
225
226
227
# File 'lib/ditzstr/operator.rb', line 223

def add_component project, config
  component = Component.create_interactively(:args => [project, config]) or return
  project.add_component component
  puts "Added component #{component.name}."
end

#add_reference(project, config, opts, issue) ⇒ Object



233
234
235
236
237
238
239
# File 'lib/ditzstr/operator.rb', line 233

def add_reference project, config, opts, issue
  puts "Adding a reference to #{issue.name}: #{issue.title}."
  reference = ask "Reference"
  issue.add_reference reference
  issue.log "added reference #{issue.references.size}", config.user, get_comment(opts)
  puts "Added reference to #{issue.name}."
end

#add_release(project, config, opts, maybe_name) ⇒ Object



214
215
216
217
218
219
220
# File 'lib/ditzstr/operator.rb', line 214

def add_release project, config, opts, maybe_name
  puts "Adding release #{maybe_name}." if maybe_name
  release = Release.create_interactively(:args => [project, config], :with => { :name => maybe_name }) or return
  release.log "created", config.user, get_comment(opts)
  project.add_release release
  puts "Added release #{release.name}."
end

#archive(project, config, release, dir) ⇒ Object



554
555
556
557
558
559
560
561
562
563
# File 'lib/ditzstr/operator.rb', line 554

def archive project, config, release, dir
  dir ||= "ditz-archive-#{release.name}"
  FileUtils.mkdir dir
  FileUtils.cp project.pathname, dir
  project.issues_for_release(release).each do |i|
    FileUtils.cp i.pathname, dir
    project.drop_issue i
  end
  puts "Archived to #{dir}."
end

#assign(project, config, opts, issue, maybe_release) ⇒ Object



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
# File 'lib/ditzstr/operator.rb', line 388

def assign project, config, opts, issue, maybe_release
  if maybe_release && maybe_release.name == issue.release
    raise Error, "issue #{issue.name} already assigned to release #{issue.release}"
  end

  puts "Issue #{issue.name} currently " + if issue.release
    "assigned to release #{issue.release}."
  else
    "not assigned to any release."
  end

  release = maybe_release || begin
    releases = project.releases.sort_by { |r| (r.release_time || 0).to_i }
    releases -= [releases.find { |r| r.name == issue.release }] if issue.release
    ask_for_selection(releases, "release") do |r|
      r.name + if r.released?
        " (released #{r.release_time.pretty_date})"
      else
        " (unreleased)"
      end
    end
  end
  issue.assign_to_release release, config.user, get_comment(opts)
  puts "Assigned #{issue.name} to #{release.name}."
end

#brick(project, config, dir) ⇒ Object



241
242
243
244
245
246
247
248
# File 'lib/ditzstr/brick.rb', line 241

def brick project, config, dir
	brickview = BrickView.new(project,config, dir)
	sharedir = File.dirname DitzStr::find_ditz_file("index.rhtml")
	start_webrick { |server| 
		imgdir = File.dirname DitzStr::find_ditz_file("index.rhtml")
       		server.mount('/', DitzStrServlet, {:brickview => brickview, :dir =>sharedir, :user=>config.user, :project=>project, :config=>config })
	}
end

#changelog(project, config, r) ⇒ Object



487
488
489
490
491
492
493
494
495
496
497
498
# File 'lib/ditzstr/operator.rb', line 487

def changelog project, config, r
  puts "== #{r.name} / #{r.released? ? r.release_time.pretty_date : 'unreleased'}"
  project.group_issues(project.issues_for_release(r)).each do |type, issues|
    issues.select { |i| i.closed? }.each do |i|
      if type == :bugfix
        puts "* #{type}: #{i.title}"
      else
        puts "* #{i.title}"
      end
    end
  end
end

#close(project, config, opts, issue) ⇒ Object



377
378
379
380
381
382
# File 'lib/ditzstr/operator.rb', line 377

def close project, config, opts, issue
  puts "Closing issue #{issue.name}: #{issue.title}."
  disp = ask_for_selection Issue::DISPOSITIONS, "disposition", lambda { |x| Issue::DISPOSITION_STRINGS[x] || x.to_s }
  issue.close disp, config.user, get_comment(opts)
  puts "Closed issue #{issue.name} with disposition #{issue.disposition_string}."
end

#comment(project, config, opts, issue) ⇒ Object



457
458
459
460
461
462
463
464
465
466
# File 'lib/ditzstr/operator.rb', line 457

def comment project, config, opts, issue
  puts "Commenting on issue #{issue.name}: #{issue.title}."
  comment = get_comment opts
  if comment.blank?
    puts "Empty comment, aborted."
  else
    issue.log "commented", config.user, comment
    puts "Comments recorded for #{issue.name}."
  end
end

#do(op, project, config, args) ⇒ Object



109
110
111
112
113
114
115
116
117
118
119
# File 'lib/ditzstr/operator.rb', line 109

def do op, project, config, args
  meth = self.class.op_to_method(op)

  # Parse options, removing them from args
  opts = self.class.build_opts meth, args
  built_args = self.class.build_args project, meth, args

  built_args.unshift opts if opts

  send meth, project, config, *built_args
end

#drop(project, config, issue) ⇒ Object



205
206
207
208
# File 'lib/ditzstr/operator.rb', line 205

def drop project, config, issue
  project.drop_issue issue
  puts "Dropped #{issue.name}. Note that other issue names may have changed."
end

#edit(project, config, opts, issue) ⇒ Object



570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
# File 'lib/ditzstr/operator.rb', line 570

def edit project, config, opts, issue
  data = { :title => issue.title, :description => issue.desc,
           :reporter => issue.reporter }

  fn = run_editor { |f| f.puts data.to_yaml }

  unless fn
    puts "Aborted."
    return
  end

  begin
    edits = YAML.load_file fn
    comment = opts[:silent] ? nil : get_comment(opts)
    if issue.change edits, config.user, comment, opts[:silent]
      puts "Change recorded."
    else
      puts "No changes."
    end
  end
end

#grep(project, config, match) ⇒ Object



512
513
514
515
516
517
518
519
# File 'lib/ditzstr/operator.rb', line 512

def grep project, config, match
  re = /#{match}/
  issues = project.issues.select do |i|
    i.title =~ re || i.desc =~ re ||
      i.log_events.map { |time, who, what, comments| comments }.join(" ") =~ re
  end
  puts(todo_list_for(issues) || "No matching issues.")
end

#help(project, config, opts, command) ⇒ Object



133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
# File 'lib/ditzstr/operator.rb', line 133

def help project, config, opts, command
  if opts[:cow]
    puts "MOO!"
    puts "All is well with the world now. A bit more methane though."
    return
  end
  return help_single(command) if command
  puts <<EOS
DitzStr commands:

EOS
  ops = self.class.operations
  len = ops.map { |name, op| name.to_s.length }.max
  ops.each do |name, opts|
    printf "  %#{len}s: %s\n", name, opts[:desc]
  end
  puts <<EOS

Use 'ditz help <command>' for details.
EOS
end

#help_single(command) ⇒ Object

Raises:



155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
# File 'lib/ditzstr/operator.rb', line 155

def help_single command
  name, opts = self.class.operations.find { |name, spec| name == command }
  raise Error, "no such ditz command '#{command}'" unless name
  args = opts[:args_spec].map do |spec|
    case spec.to_s
    when /^maybe_(.*)$/
      "[#{$1}]"
    else
      "<#{spec.to_s}>"
    end
  end.join(" ")

  puts <<EOS
#{opts[:desc]}.
Usage: ditz #{name} #{args}
EOS
end

#html(project, config, dir) ⇒ Object



501
502
503
504
# File 'lib/ditzstr/operator.rb', line 501

def html project, config, dir
  dir ||= "html"
  HtmlView.new(project, config, dir).render_all
end

#initObject



126
127
128
# File 'lib/ditzstr/operator.rb', line 126

def init
  Project.create_interactively
end

#log(project, config) ⇒ Object



522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
# File 'lib/ditzstr/operator.rb', line 522

def log project, config
  project.issues.map { |i| i.log_events.map { |e| [e, i] } }.
    flatten_one_level.sort_by { |e| e.first.first }.reverse.
    each do |(date, author, what, comment), i|
    puts <<EOS
date  : #{date.localtime} (#{date.ago} ago)
author: #{author}
 issue: [#{i.name}] #{i.title}

#{what}
#{comment.gsub(/^/, "  > ") unless comment =~ /^\A\s*\z/}
EOS
  puts unless comment.blank?
  end
end


313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
# File 'lib/ditzstr/operator.rb', line 313

def print_todo_list_by_release_for project, issues
  by_release = issues.inject({}) do |h, i|
    r = project.release_for(i.release) || :unassigned
    h[r] ||= []
    h[r] << i
    h
  end

  (project.releases + [:unassigned]).each do |r|
    next unless by_release.member? r
    puts r == :unassigned ? "Unassigned:" : "#{r.name} (#{r.status}):"
    print todo_list_for(by_release[r])
    puts
  end
end

#reconfigure(project, config) ⇒ Object



186
187
188
189
190
# File 'lib/ditzstr/operator.rb', line 186

def reconfigure project, config
  new_config = Config.create_interactively :defaults_from => config
  new_config.save! $opts[:config_file]
  puts "Configuration written."
end

#release(project, config, opts, release) ⇒ Object



481
482
483
484
# File 'lib/ditzstr/operator.rb', line 481

def release project, config, opts, release
  release.release! project, config.user, get_comment(opts)
  puts "Release #{release.name} released!"
end

#releases(project, config) ⇒ Object



469
470
471
472
473
474
475
# File 'lib/ditzstr/operator.rb', line 469

def releases project, config
  a, b = project.releases.partition { |r| r.released? }
  (b + a.sort_by { |r| r.release_time }).each do |r|
    status = r.released? ? "released #{r.release_time.pretty_date}" : r.status
    puts "#{r.name} (#{status})"
  end
end

#set_component(project, config, opts, issue, maybe_component) ⇒ Object



418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
# File 'lib/ditzstr/operator.rb', line 418

def set_component project, config, opts, issue, maybe_component
  puts "Changing the component of issue #{issue.name}: #{issue.title}."

  if project.components.size == 1
    raise Error, "this project does not use multiple components"
  end

  if maybe_component && maybe_component.name == issue.component
    raise Error, "issue #{issue.name} already assigned to component #{issue.component}"
  end

  component = maybe_component || begin
    components = project.components
    components -= [components.find { |r| r.name == issue.component }] if issue.component
    ask_for_selection(components, "component") { |r| r.name }
  end
  issue.assign_to_component component, config.user, get_comment(opts)
  oldname = issue.name
  project.assign_issue_names!
  puts <<EOS
Issue #{oldname} is now #{issue.name}. Note that the names of other issues may
have changed as well.
EOS
end

#shortlog(project, config) ⇒ Object



539
540
541
542
543
544
545
546
547
548
549
550
551
# File 'lib/ditzstr/operator.rb', line 539

def shortlog project, config
  project.issues.map { |i| i.log_events.map { |e| [e, i] } }.
    flatten_one_level.sort_by { |e| e.first.first }.reverse.
    each do |(date, author, what, comment), i|
      shortauthor = if author =~ /<(.*?)@/
        $1
      else
        author
      end[0..15]
      printf "%13s|%13s|%13s|%s\n", date.ago, i.name, shortauthor,
        what
    end
end

#show(project, config, issue) ⇒ Object



349
350
351
# File 'lib/ditzstr/operator.rb', line 349

def show project, config, issue
  ScreenView.new(project, config).render_issue issue
end

#start(project, config, opts, issue) ⇒ Object



357
358
359
360
361
# File 'lib/ditzstr/operator.rb', line 357

def start project, config, opts, issue
  puts "Starting work on issue #{issue.name}: #{issue.title}."
  issue.start_work config.user, get_comment(opts)
  puts "Recorded start of work for #{issue.name}."
end

#start_webrick(config = {}) {|server| ... } ⇒ Object

Yields:

  • (server)


229
230
231
232
233
234
235
236
237
# File 'lib/ditzstr/brick.rb', line 229

def start_webrick(config = {})
      		config.update(:Port => 8080)
       	server = HTTPServer.new(config)
       	yield server if block_given?
       	['INT', 'TERM'].each {|signal| 
               	trap(signal) {server.shutdown}
       	}
       	server.start
end

#status(project, config, releases) ⇒ Object



242
243
244
245
246
247
248
249
250
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
# File 'lib/ditzstr/operator.rb', line 242

def status project, config, releases
  releases ||= project.unreleased_releases + [:unassigned]

  if releases.empty?
    puts "No releases."
    return
  end

  entries = releases.map do |r|
    title, issues = (r == :unassigned ? r.to_s : r.name), project.issues_for_release(r)

    middle = Issue::TYPES.map do |type|
      type_issues = issues.select { |i| i.type == type }
      num = type_issues.size
      nc = type_issues.count_of { |i| i.closed? }
      pc = 100.0 * (type_issues.empty? ? 1.0 : nc.to_f / num)
      "%2d/%2d %s" % [nc, num, type.to_s.pluralize(num, false)]
    end

    bar = if r == :unassigned
      ""
    elsif r.released?
      "(released)"
    elsif issues.empty?
      "(no issues)"
    elsif issues.all? { |i| i.closed? }
      "(ready for release)"
    else
      status_bar_for(issues)
    end

    [title, middle, bar]
  end

  title_size = 0
  middle_sizes = []

  entries.each do |title, middle, bar|
    title_size = [title_size, title.length].max
    middle_sizes = middle.zip(middle_sizes).map do |e, s|
      [s || 0, e.length].max
    end
  end

  entries.each do |title, middle, bar|
    printf "%-#{title_size}s ", title
    middle.zip(middle_sizes).each_with_index do |(e, s), i|
      sep = i < middle.size - 1 ? "," : ""
      printf "%-#{s + sep.length}s ", e + sep
    end
    puts bar
  end
end

#status_bar_for(issues) ⇒ Object



296
297
298
299
300
301
# File 'lib/ditzstr/operator.rb', line 296

def status_bar_for issues
  Issue::STATUS_WIDGET.
    sort_by { |k, v| -Issue::STATUS_SORT_ORDER[k] }.
    map { |k, v| v * issues.count_of { |i| i.status == k } }.
    join
end

#stop(project, config, opts, issue) ⇒ Object



367
368
369
370
371
# File 'lib/ditzstr/operator.rb', line 367

def stop project, config, opts, issue
  puts "Stopping work on issue #{issue.name}: #{issue.title}."
  issue.stop_work config.user, get_comment(opts)
  puts "Recorded work stop for #{issue.name}."
end

#todo(project, config, opts, releases) ⇒ Object



332
333
334
# File 'lib/ditzstr/operator.rb', line 332

def todo project, config, opts, releases
  actually_do_todo project, config, releases, opts[:all]
end

#todo_list_for(issues, opts = {}) ⇒ Object



303
304
305
306
307
308
309
310
311
# File 'lib/ditzstr/operator.rb', line 303

def todo_list_for issues, opts={}
  return if issues.empty?
  name_len = issues.max_of { |i| i.name.length }
  issues.map do |i|
    s = sprintf "%s %#{name_len}s: %s", i.status_widget, i.name, i.title
    s += " [#{i.release}]" if opts[:show_release] && i.release
    s + "\n"
  end.join
end

#unassign(project, config, opts, issue) ⇒ Object



447
448
449
450
451
# File 'lib/ditzstr/operator.rb', line 447

def unassign project, config, opts, issue
  puts "Unassigning issue #{issue.name}: #{issue.title}."
  issue.unassign config.user, get_comment(opts)
  puts "Unassigned #{issue.name}."
end

#validate(project, config) ⇒ Object



507
508
509
# File 'lib/ditzstr/operator.rb', line 507

def validate project, config
  ## a no-op
end