git-pivotal-tracker-centro 1.0.0.rc1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (69) hide show
  1. data/.gitignore +6 -0
  2. data/.rspec +2 -0
  3. data/.rvmrc +1 -0
  4. data/.travis.yml +3 -0
  5. data/CHANGELOG +54 -0
  6. data/Gemfile +3 -0
  7. data/Gemfile.lock +82 -0
  8. data/LICENSE +21 -0
  9. data/Rakefile +49 -0
  10. data/VERSION +1 -0
  11. data/bin/git-finish +7 -0
  12. data/bin/git-info +7 -0
  13. data/bin/git-start +8 -0
  14. data/features/finish.feature +88 -0
  15. data/features/info.feature +99 -0
  16. data/features/start.feature +113 -0
  17. data/features/step_definitions/steps.rb +119 -0
  18. data/features/support/dsl/assertions.rb +11 -0
  19. data/features/support/dsl/data.rb +11 -0
  20. data/features/support/dsl/pivotal.rb +77 -0
  21. data/features/support/env.rb +6 -0
  22. data/features/support/git-pivotal.rb +68 -0
  23. data/features/test_repo/origin.git/COMMIT_EDITMSG +1 -0
  24. data/features/test_repo/origin.git/HEAD +1 -0
  25. data/features/test_repo/origin.git/config +8 -0
  26. data/features/test_repo/origin.git/description +1 -0
  27. data/features/test_repo/origin.git/hooks/applypatch-msg.sample +15 -0
  28. data/features/test_repo/origin.git/hooks/commit-msg.sample +24 -0
  29. data/features/test_repo/origin.git/hooks/post-commit.sample +8 -0
  30. data/features/test_repo/origin.git/hooks/post-receive.sample +15 -0
  31. data/features/test_repo/origin.git/hooks/post-update.sample +8 -0
  32. data/features/test_repo/origin.git/hooks/pre-applypatch.sample +14 -0
  33. data/features/test_repo/origin.git/hooks/pre-commit.sample +46 -0
  34. data/features/test_repo/origin.git/hooks/pre-rebase.sample +169 -0
  35. data/features/test_repo/origin.git/hooks/prepare-commit-msg.sample +36 -0
  36. data/features/test_repo/origin.git/hooks/update.sample +128 -0
  37. data/features/test_repo/origin.git/index +0 -0
  38. data/features/test_repo/origin.git/info/exclude +6 -0
  39. data/features/test_repo/origin.git/logs/HEAD +1 -0
  40. data/features/test_repo/origin.git/logs/refs/heads/master +1 -0
  41. data/features/test_repo/origin.git/objects/0c/6f7b1384910d1a2f137590095f008a06c7e00c +0 -0
  42. data/features/test_repo/origin.git/objects/10/ecf2b7ce989f01f3f7266e712b48d9275f2635 +0 -0
  43. data/features/test_repo/origin.git/objects/a5/71d56305df09fb060f6ccb730b46080d305beb +0 -0
  44. data/features/test_repo/origin.git/refs/heads/master +1 -0
  45. data/features/test_repo/readme +1 -0
  46. data/git-pivotal-tracker-centro.gemspec +29 -0
  47. data/lib/commands/base.rb +120 -0
  48. data/lib/commands/bug.rb +19 -0
  49. data/lib/commands/card.rb +32 -0
  50. data/lib/commands/chore.rb +19 -0
  51. data/lib/commands/feature.rb +19 -0
  52. data/lib/commands/finish.rb +67 -0
  53. data/lib/commands/info.rb +58 -0
  54. data/lib/commands/map.rb +10 -0
  55. data/lib/commands/pick.rb +94 -0
  56. data/lib/commands/start.rb +39 -0
  57. data/lib/git-pivotal-tracker.rb +11 -0
  58. data/readme.markdown +106 -0
  59. data/spec/commands/base_spec.rb +137 -0
  60. data/spec/commands/bug_spec.rb +24 -0
  61. data/spec/commands/chore_spec.rb +24 -0
  62. data/spec/commands/feature_spec.rb +24 -0
  63. data/spec/commands/finish_spec.rb +134 -0
  64. data/spec/commands/map_spec.rb +14 -0
  65. data/spec/commands/start_spec.rb +28 -0
  66. data/spec/factories.rb +13 -0
  67. data/spec/factory.rb +26 -0
  68. data/spec/spec_helper.rb +24 -0
  69. metadata +291 -0
@@ -0,0 +1 @@
1
+ 0000000000000000000000000000000000000000 10ecf2b7ce989f01f3f7266e712b48d9275f2635 Jeff Tucker <trydionel@gmail.com> 1279979728 -0400 commit (initial): A blank repo for testing
@@ -0,0 +1 @@
1
+ 10ecf2b7ce989f01f3f7266e712b48d9275f2635
@@ -0,0 +1 @@
1
+ Testing repo
@@ -0,0 +1,29 @@
1
+ # -*- encoding: utf-8 -*-
2
+ $:.push File.expand_path("../lib", __FILE__)
3
+
4
+ Gem::Specification.new do |s|
5
+ s.name = "git-pivotal-tracker-centro"
6
+ s.version = IO.read("VERSION")
7
+ s.platform = Gem::Platform::RUBY
8
+ s.authors = ["Zach Dennis", "Jeff Tucker", "Sam Stokes", "John Wood"]
9
+ s.email = "john@johnpwood.net"
10
+ s.homepage = "https://github.com/centro/git-pivotal"
11
+ s.summary = "A collection of git utilities to ease integration with Pivotal Tracker."
12
+ s.description = "A collection of git utilities to ease integration with Pivotal Tracker."
13
+
14
+ s.files = `git ls-files`.split("\n")
15
+ s.test_files = `git ls-files -- {test,spec,features}/*`.split("\n")
16
+ s.executables = `git ls-files -- bin/*`.split("\n").map{ |f| File.basename(f) }
17
+ s.require_paths = ["lib"]
18
+
19
+ s.add_runtime_dependency(%q<pivotal-tracker>, [">= 0"])
20
+
21
+ s.add_development_dependency(%q<rake>, [">= 0"])
22
+ s.add_development_dependency(%q<mocha>, [">= 0"])
23
+ s.add_development_dependency(%q<simplecov>, [">= 0"])
24
+ s.add_development_dependency(%q<aruba>, [">= 0"])
25
+ s.add_development_dependency(%q<rspec>, [">= 0"])
26
+ s.add_development_dependency(%q<cucumber>, [">= 0"])
27
+ s.add_development_dependency(%q<pry>, [">= 0"])
28
+ end
29
+
@@ -0,0 +1,120 @@
1
+ require 'rubygems'
2
+ require 'pivotal-tracker'
3
+ require 'optparse'
4
+
5
+ module Commands
6
+ class Base
7
+
8
+ attr_accessor :input, :output, :options
9
+
10
+ def initialize(*args)
11
+ @input = STDIN
12
+ @output = STDOUT
13
+
14
+ @options = {}
15
+
16
+ parse_gitconfig
17
+ parse_argv(*args)
18
+ end
19
+
20
+ def with(input, output)
21
+ tap do
22
+ @input = input
23
+ @output = output
24
+ end
25
+ end
26
+
27
+ def put(string, newline=true)
28
+ @output.print(newline ? string + "\n" : string) unless options[:quiet]
29
+ end
30
+
31
+ def sys(cmd)
32
+ if options[:verbose]
33
+ put cmd
34
+ system cmd
35
+ else
36
+ system "#{cmd} > /dev/null 2>&1"
37
+ end
38
+ end
39
+
40
+ def get(cmd)
41
+ put cmd if options[:verbose]
42
+ `#{cmd}`
43
+ end
44
+
45
+ def get_char
46
+ input.getc
47
+ end
48
+
49
+ def run!
50
+ unless options[:api_token] && options[:project_id]
51
+ put "Pivotal Tracker API Token and Project ID are required"
52
+ return 1
53
+ end
54
+
55
+ PivotalTracker::Client.token = options[:api_token]
56
+ PivotalTracker::Client.use_ssl = options[:use_ssl]
57
+
58
+ return 0
59
+ end
60
+
61
+ protected
62
+
63
+ def on_parse(opts)
64
+ # no-op, override in sub-class to provide command specific options
65
+ end
66
+
67
+ def current_branch
68
+ @current_branch ||= get('git symbolic-ref HEAD').chomp.split('/').last
69
+ end
70
+
71
+ def project
72
+ @project ||= PivotalTracker::Project.find(options[:project_id])
73
+ end
74
+
75
+ def full_name
76
+ options[:full_name]
77
+ end
78
+
79
+ def remote
80
+ options[:remote] || "origin"
81
+ end
82
+
83
+ private
84
+
85
+ def parse_gitconfig
86
+ token = get("git config --get pivotal.api-token").strip
87
+ name = get("git config --get pivotal.full-name").strip
88
+ id = get("git config --get pivotal.project-id").strip
89
+ remote = get("git config --get pivotal.remote").strip
90
+ use_ssl = get("git config --get pivotal.use-ssl").strip
91
+ verbose = get("git config --get pivotal.verbose").strip
92
+
93
+ options[:api_token] = token unless token == ""
94
+ options[:project_id] = id unless id == ""
95
+ options[:full_name] = name unless name == ""
96
+ options[:remote] = remote unless remote == ""
97
+ options[:use_ssl] = (/^true$/i.match(use_ssl))
98
+ options[:verbose] = verbose == "" ? true : (/^true$/i.match(verbose))
99
+ end
100
+
101
+ def parse_argv(*args)
102
+ OptionParser.new do |opts|
103
+ opts.banner = "Usage: git pick [options]"
104
+ opts.on("-k", "--api-key=", "Pivotal Tracker API key") { |k| options[:api_token] = k }
105
+ opts.on("-p", "--project-id=", "Pivotal Tracker project id") { |p| options[:project_id] = p }
106
+ opts.on("-n", "--full-name=", "Pivotal Tracker full name") { |n| options[:full_name] = n }
107
+ opts.on("-S", "--use-ssl", "Use SSL for connection to Pivotal Tracker (for private repos(?))") { |s| options[:use_ssl] = s }
108
+ opts.on("-D", "--defaults", "Accept default options. No-interaction mode") { |d| options[:defaults] = d }
109
+ opts.on("-q", "--quiet", "Quiet, no-interaction mode") { |q| options[:quiet] = q }
110
+ opts.on("-v", "--[no-]verbose", "Run verbosely") { |v| options[:verbose] = v }
111
+ opts.on("-f", "--force", "Do not prompt") { |f| options[:force] = f }
112
+
113
+ on_parse(opts)
114
+
115
+ opts.on_tail("-h", "--help", "This usage guide") { put opts.to_s; exit 0 }
116
+ end.parse!(args)
117
+ end
118
+
119
+ end
120
+ end
@@ -0,0 +1,19 @@
1
+ require 'commands/pick'
2
+
3
+ module Commands
4
+ class Bug < Pick
5
+
6
+ def type
7
+ "bug"
8
+ end
9
+
10
+ def plural_type
11
+ "bugs"
12
+ end
13
+
14
+ def branch_suffix
15
+ "bugfix"
16
+ end
17
+
18
+ end
19
+ end
@@ -0,0 +1,32 @@
1
+ require 'commands/pick'
2
+
3
+ module Commands
4
+ class Card < Pick
5
+ attr_accessor :story_id
6
+
7
+ def type
8
+ "story"
9
+ end
10
+
11
+ def plural_type
12
+ "cards"
13
+ end
14
+
15
+ def branch_suffix
16
+ if story.story_type == "bug"
17
+ "bugfix"
18
+ else
19
+ story.story_type
20
+ end
21
+ end
22
+
23
+ protected
24
+
25
+ def story
26
+ return @story if @story
27
+ raise ArgumentError, "No story id was given!" unless story_id
28
+ project.stories.find(story_id)
29
+ end
30
+
31
+ end
32
+ end
@@ -0,0 +1,19 @@
1
+ require 'commands/pick'
2
+
3
+ module Commands
4
+ class Chore < Pick
5
+
6
+ def type
7
+ "chore"
8
+ end
9
+
10
+ def plural_type
11
+ "chores"
12
+ end
13
+
14
+ def branch_suffix
15
+ "chore"
16
+ end
17
+
18
+ end
19
+ end
@@ -0,0 +1,19 @@
1
+ require 'commands/pick'
2
+
3
+ module Commands
4
+ class Feature < Pick
5
+
6
+ def type
7
+ "feature"
8
+ end
9
+
10
+ def plural_type
11
+ "features"
12
+ end
13
+
14
+ def branch_suffix
15
+ "feature"
16
+ end
17
+
18
+ end
19
+ end
@@ -0,0 +1,67 @@
1
+ require 'commands/base'
2
+
3
+ module Commands
4
+ class Finish < Base
5
+
6
+ def run!
7
+ super
8
+
9
+ unless story_id
10
+ put "Branch name must contain a Pivotal Tracker story id"
11
+ return 1
12
+ end
13
+
14
+ if confirm_should_finish_story?
15
+ topic_branch = current_branch
16
+
17
+ put "Switching to master"
18
+ sys "git checkout master"
19
+
20
+ put "Destroying local branch"
21
+ local_branch_deleted_successfully = sys "git branch -d #{topic_branch}"
22
+
23
+ if local_branch_deleted_successfully
24
+ put "Destroying remote branch"
25
+ sys "git push origin :#{topic_branch}"
26
+
27
+ put "Marking Story #{story_id} as finished..."
28
+ if !story.update(:current_state => finished_state)
29
+ put "Unable to mark Story #{story_id} as finished"
30
+ return 1
31
+ end
32
+ else
33
+ put "The local branch could not be deleted. Please make sure your changes have been merged."
34
+ return 1
35
+ end
36
+ end
37
+
38
+ return 0
39
+ end
40
+
41
+ protected
42
+
43
+ def confirm_should_finish_story?
44
+ return true if options[:force]
45
+
46
+ put "Finish story #{story_id} and delete local and remote branches? (y/N): ", false
47
+ finish_story = get_char.strip.downcase
48
+ finish_story == 'y'
49
+ end
50
+
51
+ def finished_state
52
+ if story.story_type == "chore"
53
+ "accepted"
54
+ else
55
+ "finished"
56
+ end
57
+ end
58
+
59
+ def story_id
60
+ match = current_branch[/\d+/] and match.to_i
61
+ end
62
+
63
+ def story
64
+ @story ||= project.stories.find(story_id)
65
+ end
66
+ end
67
+ end
@@ -0,0 +1,58 @@
1
+ require 'commands/base'
2
+
3
+ module Commands
4
+ class Info < Base
5
+ def initialize(*args)
6
+ @story_id = args.shift if args.first =~ /^(\d+)$/
7
+ super(*args)
8
+ end
9
+
10
+ def run!
11
+ super
12
+
13
+ unless story_id
14
+ put "No story id was supplied and you aren't on a topic branch!"
15
+ return 1
16
+ end
17
+
18
+ put "Story: #{story.name}"
19
+ put "URL: #{story.url}"
20
+ put "Labels: #{story.labels.split(',').join(', ')}" if story.labels
21
+ put "State: #{story.accepted_at ? 'accepted' : 'not accepted'}"
22
+
23
+ colwidth = 74
24
+
25
+ put "\nDescription:\n"
26
+ put wrap_text("#{story.description}\n", colwidth).gsub(/^/, ' ').chomp
27
+
28
+ if options[:comments]
29
+ put "\nComments:\n"
30
+ story.notes.all.each do |note|
31
+ @output.printf " %-37s%37s\n\n", note.author, note.noted_at.strftime("%b %e, %Y %k:%M%p")
32
+ put wrap_text(note.text, colwidth - 2).gsub(/^/, ' ')
33
+ end
34
+ end
35
+
36
+ return 0
37
+ end
38
+
39
+ protected
40
+
41
+ def wrap_text(txt, col = 80)
42
+ txt.gsub(/(.{1,#{col}})( +|$\n?)|(.{1,#{col}})/,
43
+ "\\1\\3\n")
44
+ end
45
+
46
+ def on_parse(opts)
47
+ opts.on("-c", "--comments", "Display comments"){ |v| options[:comments] = v }
48
+ end
49
+
50
+ def story_id
51
+ @story_id || current_branch[/\d+/]
52
+ end
53
+
54
+ def story
55
+ @story ||= project.stories.find(story_id)
56
+ end
57
+ end
58
+ end
@@ -0,0 +1,10 @@
1
+ module Commands
2
+ class Map < Hash
3
+ def [](arg)
4
+ super || (
5
+ key = keys.detect{ |e| !arg.nil? && e.respond_to?(:match) && e.match(arg) }
6
+ super key
7
+ )
8
+ end
9
+ end
10
+ end
@@ -0,0 +1,94 @@
1
+ require 'commands/base'
2
+
3
+ module Commands
4
+ class Pick < Base
5
+
6
+ def type
7
+ raise Error("must define in subclass")
8
+ end
9
+
10
+ def plural_type
11
+ raise Error("must define in subclass")
12
+ end
13
+
14
+ def branch_suffix
15
+ raise Error("must define in subclass")
16
+ end
17
+
18
+ def run!
19
+ response = super
20
+ return response if response > 0
21
+
22
+ if story
23
+ put "Story: #{story.name}"
24
+ put "URL: #{story.url}\n"
25
+
26
+ if confirm_should_start_story?
27
+ put "Updating #{type} status in Pivotal Tracker..."
28
+ story.update(:owned_by => options[:full_name], :current_state => :started)
29
+
30
+ if story.errors.empty?
31
+ branch_name = get_branch_name
32
+ create_branch(branch_name)
33
+ else
34
+ put "Unable to mark #{type} as started"
35
+ put "\t" + story.errors.to_a.join("\n\t")
36
+ return 1
37
+ end
38
+ end
39
+ else
40
+ put "No #{plural_type} available!"
41
+ end
42
+
43
+ return 0
44
+ end
45
+
46
+ protected
47
+
48
+ def confirm_should_start_story?
49
+ return true if options[:force]
50
+
51
+ put "Start #{type} #{story.id} - #{story.name}? (Y/n): ", false
52
+ start_story = get_char.strip.downcase
53
+ start_story == '' || start_story == 'y'
54
+ end
55
+
56
+ def story
57
+ if @story.nil?
58
+ msg = "Retrieving latest #{plural_type} from Pivotal Tracker"
59
+ msg += " for #{options[:full_name]}" if options[:only_mine]
60
+ put "#{msg}..."
61
+
62
+ conditions = { :story_type => type, :current_state => "unstarted", :limit => 1, :offset => 0 }
63
+ conditions[:owned_by] = options[:full_name] if options[:only_mine]
64
+ @story = project.stories.all(conditions).first
65
+ end
66
+ @story
67
+ end
68
+
69
+ def get_branch_name
70
+ branch_name = ""
71
+ suggested_branch_name = "#{story.id}-#{story.name.downcase.gsub(/\s+/, "-")}"
72
+
73
+ unless options[:quiet] || options[:defaults]
74
+ put "Enter branch name [#{suggested_branch_name}]: ", false
75
+ branch_name = input.gets.chomp.strip
76
+ end
77
+
78
+ branch_name == "" ? suggested_branch_name : branch_name
79
+ end
80
+
81
+ def create_branch(branch)
82
+ if get("git branch").match(branch).nil?
83
+ put "Creating remote branch '#{branch}'"
84
+ sys "git push origin origin:refs/heads/#{branch}"
85
+ sys "git fetch origin"
86
+
87
+ put "Switched to a new branch '#{branch}'"
88
+ sys "git branch #{branch} origin/#{branch}"
89
+ sys "git checkout #{branch}"
90
+ end
91
+ end
92
+
93
+ end
94
+ end
@@ -0,0 +1,39 @@
1
+ require 'commands/map'
2
+ require 'commands/bug'
3
+ require 'commands/card'
4
+ require 'commands/chore'
5
+ require 'commands/feature'
6
+
7
+ module Commands
8
+ class Start
9
+ COMMAND_MAP = Map.new.merge({
10
+ "bug" => Commands::Bug,
11
+ "chore" => Commands::Chore,
12
+ "feature" => Commands::Feature,
13
+ /^\d+$/ => Commands::Card
14
+ })
15
+
16
+ class << self
17
+ def for(*args)
18
+ identifier = args.shift
19
+ construct_instance_for(identifier, args) || display_usage_instructions_and_quit(identifier)
20
+ end
21
+
22
+ private
23
+
24
+ def construct_instance_for(identifier, args)
25
+ if klass=COMMAND_MAP[identifier]
26
+ instance = klass.new(*args)
27
+ instance.story_id = identifier if instance.respond_to?(:story_id=)
28
+ instance
29
+ end
30
+ end
31
+
32
+ def display_usage_instructions_and_quit(identifier)
33
+ puts "ERROR: Unknown card identifier given: '#{identifier}'. Valid options are 'bug', 'chore', 'feature', or the card number."
34
+ exit 1
35
+ end
36
+ end
37
+ end
38
+
39
+ end
@@ -0,0 +1,11 @@
1
+ $:.unshift(File.dirname(__FILE__))
2
+
3
+ require File.join('commands', 'base')
4
+ require File.join('commands', 'pick')
5
+ require File.join('commands', 'card')
6
+ require File.join('commands', 'feature')
7
+ require File.join('commands', 'bug')
8
+ require File.join('commands', 'chore')
9
+ require File.join('commands', 'finish')
10
+ require File.join('commands', 'info')
11
+ require File.join('commands', 'start')
data/readme.markdown ADDED
@@ -0,0 +1,106 @@
1
+ This is based on https://github.com/trydionel/git-pivotal but has an extended vision and set of goals to achieve.
2
+
3
+ Things like:
4
+
5
+ * Make the existing API more intuitive:
6
+ * git-bug, git-chore, git-feature should not be separate CLI commands. The common action is start, so goal here is to support "git-start _story\_type_"
7
+ * Users should be able to interact with specific stories when possible.
8
+ * e.g. "git start 123456" should start story 123456.
9
+ * being generic (interacting with next available or specific should apply to all applicable commands)
10
+ * Add more commands to interact with Pivotal:
11
+ * git comments
12
+ * git label
13
+ * git unstart
14
+ * Add verbosity and dry-run support to commands to communicate to the users what commands will be run
15
+ * Add before/after hooks extension points so people do not have to modify the project in order to do something custom.
16
+ * e.g. if you want to build a changelog as story's are finished, this should be able to be done by hooking into the "git finish" command and not require altering the code-base
17
+ * better support for handling merge conflicts (_this may be nothing more than communicating better to the user if a merge conflict happens when issuing a command_)
18
+
19
+ The main vision for this is simple: Encourage and support good practices and be flexible.
20
+
21
+ More README to come. See ISSUES for things I want to tackle in this project.
22
+
23
+ ## The workflow
24
+
25
+ 1. Assign the story to yourself (or have someone assign it to you)
26
+ 2. From _master_, `git start <story_type>`, where story\_type is either 'bug', 'chore', 'feature', or the story number. A new branch will be created, and you will be automatically switched to it. The story will also be marked as started.
27
+ 3. WRITE ALL THE CODE!, create pull request, merge into master.
28
+ 4. `git finish`. This will destroy the local and remote branches, and mark the story as finished.
29
+
30
+ ### git start - Starting the next available Feature/Bug/Chore
31
+
32
+ git start <story_type>
33
+
34
+ Replace story\_type in the above command to start the next available story in your Pivotal project, e.g.:
35
+
36
+ 1 git-pivotal:master % git start feature
37
+ Retrieving latest features from Pivotal Tracker for John Wood...
38
+ Story: Test the tracker gem
39
+ URL: http://www.pivotaltracker.com/story/show/38895179
40
+
41
+ Start feature 38895179 - Test the tracker gem? (Y/n): Y
42
+ Updating feature status in Pivotal Tracker...
43
+ Enter branch name [38895179-test-the-tracker-gem]:
44
+ git branch
45
+ Creating remote branch '38895179-test-the-tracker-gem'
46
+ git push origin origin:refs/heads/38895179-test-the-tracker-gem
47
+ Total 0 (delta 0), reused 0 (delta 0)
48
+ To git@github.com:centro/git-pivotal.git
49
+ * [new branch] origin/HEAD -> 38895179-test-the-tracker-gem
50
+ git fetch origin
51
+ Switched to a new branch '38895179-test-the-tracker-gem'
52
+ git branch 38895179-test-the-tracker-gem origin/38895179-test-the-tracker-gem
53
+ Branch 38895179-test-the-tracker-gem set up to track remote branch 38895179-test-the-tracker-gem from origin.
54
+ git checkout 38895179-test-the-tracker-gem
55
+ Switched to branch '38895179-test-the-tracker-gem'
56
+ 2 git-pivotal:38895179-test-the-tracker-gem %
57
+
58
+ ### git finish
59
+ When on a feature branch, this command will close the associated story in Pivotal Tracker and remove the feature branch.
60
+
61
+ 3 git-pivotal:38895179-test-the-tracker-gem % git finish
62
+ git symbolic-ref HEAD
63
+ Finish story 38895179 and delete local and remote branches? (y/N): y
64
+ Switching to master
65
+ git co master
66
+ Switched to branch 'master'
67
+ Destroying local branch
68
+ git branch -d 38895179-test-the-tracker-gem
69
+ Deleted branch 38895179-test-the-tracker-gem (was 99e52a3).
70
+ Destroying remote branch
71
+ git push origin :38895179-test-the-tracker-gem
72
+ To git@github.com:centro/git-pivotal.git
73
+ - [deleted] 38895179-test-the-tracker-gem
74
+ Marking Story 38895179 as finished...
75
+ 4 git-pivotal:master %
76
+
77
+ ### git info
78
+ When on a feature/bug/chore branch, this command will display the story information as recorded in Pivotal Tracker.
79
+
80
+ 5 git-pivotal:1234567-testing % git info
81
+ Story: Test git pivotal
82
+ URL: http://www.pivotaltracker.com/story/show/1234567
83
+ Description: The awesome story description
84
+ 6 git-pivotal:1234567-testing %
85
+
86
+ ## Installation
87
+
88
+ _This section is out of date and applies to the original project. It needs to be updated._
89
+
90
+ To install git-pivotal, simply run
91
+
92
+ [sudo] gem install git-pivotal
93
+
94
+ ## Configuration
95
+
96
+ Once installed, git pivotal needs three bits of info: your Pivotal Tracker API Token, your name as it appears in Pivotal Tracker and your Pivotal Tracker project id. The former two are best set as a global git config options:
97
+
98
+ git config --global pivotal.api-token 9a9a9a9a9a9a9a9a9a9a
99
+ git config --global pivotal.full-name "Jeff Tucker"
100
+
101
+ The project id is best placed within your project's git config:
102
+
103
+ git config -f .git/config pivotal.project-id 88888
104
+
105
+ If you're not interested in storing these options in git, you can pass them into git pivotal as command line arguments. See the usage guides for more details.
106
+