# Git Pivotal Tracker Integration # Copyright (c) 2013 the original author or authors. # # Licensed under the Apache License, Version 2.0 (the "License"); # you may not use this file except in compliance with the License. # You may obtain a copy of the License at # # http://www.apache.org/licenses/LICENSE-2.0 # # Unless required by applicable law or agreed to in writing, software # distributed under the License is distributed on an "AS IS" BASIS, # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. # See the License for the specific language governing permissions and # limitations under the License. require 'git-pivotal-tracker-integration/command/command' require 'git-pivotal-tracker-integration/command/configuration' require 'git-pivotal-tracker-integration/util/git' require 'pivotal-tracker' require 'parseconfig' require 'logger' # An abstract base class for all commands # @abstract Subclass and override {#run} to implement command functionality class GitPivotalTrackerIntegration::Command::Base # Common initialization functionality for all command classes. This # enforces that: # * the command is being run within a valid Git repository # * the user has specified their Pivotal Tracker API token # * all communication with Pivotal Tracker will be protected with SSL # * the user has configured the project id for this repository def initialize self.start_logging self.check_version git_global_push_default = (GitPivotalTrackerIntegration::Util::Shell.exec "git config --global push.default").chomp if git_global_push_default != "simple" puts "git config --global push.default simple" puts GitPivotalTrackerIntegration::Util::Shell.exec "git config --global push.default simple" end @repository_root = GitPivotalTrackerIntegration::Util::Git.repository_root @configuration = GitPivotalTrackerIntegration::Command::Configuration.new @toggl = Toggl.new PivotalTracker::Client.token = @configuration.api_token PivotalTracker::Client.use_ssl = true @project = PivotalTracker::Project.find @configuration.project_id end def finish_toggle(configuration, time_spent) current_story = @configuration.story(@project) @toggl.create_task(parameters(configuration, time_spent)) @toggl.create_time_entry(parameters(configuration, time_spent)) end def start_logging $LOG = Logger.new("#{logger_filename}", 'weekly') end def logger_filename return "#{Dir.home}/.v2gpti_local.log" end def check_version gem_latest_version = (GitPivotalTrackerIntegration::Util::Shell.exec "gem list v2gpti --remote")[/\(.*?\)/].delete "()" gem_installed_version = Gem.loaded_specs["v2gpti"].version.version if (gem_installed_version == gem_latest_version) $LOG.info("v2gpti verison #{gem_installed_version} is up to date.") else $LOG.fatal("Out of date") abort "\n\nYou are using v2gpti version #{gem_installed_version}, but the current version is #{gem_latest_version}.\nPlease update your gem with the following command.\n\n sudo gem update v2gpti\n\n" end end # The main entry point to the command's execution # @abstract Override this method to implement command functionality def run raise NotImplementedError end # Toggl keys # name : The name of the task (string, required, unique in project) # pid : project ID for the task (integer, required) # wid : workspace ID, where the task will be saved (integer, project's workspace id is used when not supplied) # uid : user ID, to whom the task is assigned to (integer, not required) # estimated_seconds : estimated duration of task in seconds (integer, not required) # active : whether the task is done or not (boolean, by default true) # at : timestamp that is sent in the response for PUT, indicates the time task was last updated # -- Additional fields -- # done_seconds : duration (in seconds) of all the time entries registered for this task # uname : full name of the person to whom the task is assigned to TIMER_TOKENS = { "m" => (60), "h" => (60 * 60), "d" => (60 * 60 * 8) # a work day is 8 hours } def parameters(configuration, time_spent) current_story = configuration.story(@project) params = Hash.new params[:name] = "#{current_story.id}" + " - " + "#{current_story.name}" params[:estimated_seconds] = estimated_seconds current_story params[:pid] = configuration.toggl_project_id params[:uid] = @toggl.me["id"] params[:tags] = [current_story.story_type] params[:active] = false params[:description] = "#{current_story.id}" + " commit:" + "#{(GitPivotalTrackerIntegration::Util::Shell.exec "git rev-parse HEAD").chomp[0..6]}" params[:created_with] = "v2gpti" params[:duration] = seconds_spent(time_spent) params[:start] = (Time.now - params[:duration]).iso8601 task = @toggl.get_project_task_with_name(configuration.toggl_project_id, "#{current_story.id}") if !task.nil? params[:tid] = task['id'] end params end def seconds_spent(time_spent) seconds = 0 time_spent.scan(/(\d+)(\w)/).each do |amount, measure| seconds += amount.to_i * TIMER_TOKENS[measure] end seconds end def estimated_seconds(story) estimate = story.estimate seconds = 0 case estimate when 0 estimate = 15 * 60 when 1 estimate = 1.25 * 60 * 60 when 2 estimate = 3 * 60 * 60 when 3 estimate = 8 * 60 * 60 else estimate = -1 * 60 * 60 end estimate end end