Class: Pindo::Command::Dev::Tag

Inherits:
Pindo::Command::Dev show all
Defined in:
lib/pindo/command/dev/tag.rb

Class Method Summary collapse

Instance Method Summary collapse

Constructor Details

#initialize(argv) ⇒ Tag

Returns a new instance of Tag.



45
46
47
48
49
50
51
52
53
54
55
# File 'lib/pindo/command/dev/tag.rb', line 45

def initialize(argv)
  @mode = argv.option('mode') || 'minor'
  @force_retag = argv.flag?('retag', false)
  
  unless ['major', 'minor', 'patch'].include?(@mode)
    raise Informative, "mode参数必须是 major, minor 或 patch"
  end
  
  super
  @additional_args = argv.remainder!
end

Class Method Details

.optionsObject

命令的选项列表



38
39
40
41
42
43
# File 'lib/pindo/command/dev/tag.rb', line 38

def self.options
  [
    ['--mode',   '指定版本号增加模式(major/minor/patch),默认minor'],
    ['--retag',  '强制重新打最新的tag']
  ].concat(super)
end

Instance Method Details

#add_release_tag(project_dir: nil, increment_mode: "minor", force_retag: false) ⇒ Object

Raises:

  • (ArgumentError)


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
123
124
125
126
127
128
129
# File 'lib/pindo/command/dev/tag.rb', line 96

def add_release_tag(project_dir: nil, increment_mode: "minor", force_retag: false)
  raise ArgumentError, "项目目录不能为空" if project_dir.nil?
    
  Funlog.instance.fancyinfo_start("开始创建初tag")
  latest_tag = get_latest_version_tag(project_dir: project_dir)
  if latest_tag.nil?
    new_tag = create_next_version_tag(
      project_dir: project_dir,
      tag_prefix: "v",
      increment_mode: increment_mode,
      force_retag: false
    )
    Funlog.instance.fancyinfo_success("创建初始tag: #{new_tag}")
    Funlog.instance.fancyinfo_success("仓库路径: #{project_dir}")
    return
  end

  if is_tag_at_head?(git_root_dir: project_dir, tag_name: latest_tag)
    Funlog.instance.fancyinfo_success("当前commit已有tag: #{latest_tag},无需重新打tag")
    Funlog.instance.fancyinfo_success("仓库路径: #{project_dir}")
    return
  end

  Funlog.instance.fancyinfo_success("最近的上次tag是: #{latest_tag} ")
  new_tag = create_next_version_tag(
      project_dir: project_dir,
      tag_prefix: "v",
      increment_mode: increment_mode,
      force_retag: force_retag
  )

  Funlog.instance.fancyinfo_success("当前仓库的tag: #{new_tag}")
  Funlog.instance.fancyinfo_success("仓库路径: #{project_dir}")
end

#merge_to_release_branch(project_dir: nil, release_branch: nil, coding_branch: nil) ⇒ Object



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
175
176
177
178
179
180
181
182
183
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
# File 'lib/pindo/command/dev/tag.rb', line 131

def merge_to_release_branch(project_dir: nil, release_branch: nil, coding_branch: nil)
  current_project_dir = project_dir
  Funlog.instance.fancyinfo_start("开始合并到#{release_branch}分支")
  if !coding_branch.eql?(release_branch)
    coding_branch_commit_id = git!(%W(-C #{current_project_dir} rev-parse #{coding_branch})).strip
    release_branch_commit_id = nil
    
    if remote_branch_exists?(local_repo_dir: current_project_dir, branch: release_branch)
      Funlog.instance.fancyinfo_update("存在#{release_branch}远程分支")
      if local_branch_exists?(local_repo_dir: current_project_dir, branch: release_branch)
        Funlog.instance.fancyinfo_update("存在#{release_branch}本地分支")
        git!(%W(-C #{current_project_dir} checkout #{release_branch}))
      else
        Funlog.instance.fancyinfo_update("不存在#{release_branch}本地分支")
        git!(%W(-C #{current_project_dir} checkout -b #{release_branch} origin/#{release_branch}))
      end

      git!(%W(-C #{current_project_dir} branch --set-upstream-to=origin/#{release_branch} #{release_branch}))
      git!(%W(-C #{current_project_dir} fetch origin #{release_branch}))
      git!(%W(-C #{current_project_dir} merge origin/#{release_branch}))

      # 执行合并操作,捕获输出以便后续检查冲突
      stdout, exit_status = Executable.capture_command('git', %W(-C #{current_project_dir} merge #{coding_branch}), :capture => :out)

      # 检查是否有冲突(git merge 在有冲突时返回非0状态)
      conflict_filelist = git!(%W(-C #{current_project_dir} diff --name-only --diff-filter=U --relative))
      if !conflict_filelist.nil? && conflict_filelist.size > 0
        puts "合并代码冲突, 冲突文件如下:"
        raise Informative, "请手动处理冲突的文件!!!"
      else
        git!(%W(-C #{current_project_dir} push))
        Funlog.instance.fancyinfo_success("代码已经合并到#{release_branch}分支")
        # 获取 release_branch 的 commit ID(处理空分支情况)
        begin
          release_branch_commit_id = git!(%W(-C #{current_project_dir} rev-parse #{release_branch})).strip
        rescue => e
          # 分支可能存在但没有提交(空分支),此时使用 coding_branch 的 commit
          Funlog.instance.fancyinfo_update("#{release_branch}分支为空或获取commit失败,将使用当前分支commit")
          release_branch_commit_id = coding_branch_commit_id
        end
      end

    else
      if local_branch_exists?(local_repo_dir: current_project_dir, branch: release_branch)
        Funlog.instance.fancyinfo_update("不存在#{release_branch}远程分支")
        Funlog.instance.fancyinfo_update("存在#{release_branch}本地分支")
        git!(%W(-C #{current_project_dir} checkout #{release_branch}))
        git!(%W(-C #{current_project_dir} checkout -b #{release_branch}_temp))
        git!(%W(-C #{current_project_dir} checkout #{coding_branch}))
        git!(%W(-C #{current_project_dir} branch -D #{release_branch}))
      else
        Funlog.instance.fancyinfo_update("不存在#{release_branch}远程分支")
        Funlog.instance.fancyinfo_update("不存在#{release_branch}本地分支")
      end

      git!(%W(-C #{current_project_dir} checkout -b #{release_branch}))
      git!(%W(-C #{current_project_dir} push origin #{release_branch}))
      git!(%W(-C #{current_project_dir} branch --set-upstream-to=origin/#{release_branch} #{release_branch}))

      Funlog.instance.fancyinfo_success("代码已经合并到#{release_branch}分支")
      # 获取 release_branch 的 commit ID(处理空分支情况)
      begin
        release_branch_commit_id = git!(%W(-C #{current_project_dir} rev-parse #{release_branch})).strip
      rescue => e
        # 新创建的分支,commit ID 应该与 coding_branch 相同
        release_branch_commit_id = coding_branch_commit_id
      end
    end

    git!(%W(-C #{current_project_dir} checkout #{coding_branch}))
    if release_branch_commit_id && !release_branch_commit_id.eql?(coding_branch_commit_id)
      git!(%W(-C #{current_project_dir} merge #{release_branch}))
      Funlog.instance.fancyinfo_success("已将#{release_branch}合并到#{coding_branch}")
    end
    git!(%W(-C #{current_project_dir} push origin #{coding_branch}))
    Funlog.instance.fancyinfo_success("已推送#{coding_branch}分支到远程")
  else
    Funlog.instance.fancyinfo_success("代码处于#{coding_branch}分支,无需合并")
  end
end

#runObject



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
82
83
84
85
86
87
88
89
90
91
92
93
94
# File 'lib/pindo/command/dev/tag.rb', line 57

def run
  current_project_dir = Dir.pwd
  puts current_project_dir
  # 检查当前目录是否是git仓库
  unless is_git_directory?(local_repo_dir: current_project_dir)
    
    Funlog.instance.fancyinfo_error("当前目录不是git仓库")
    raise Informative, "请在git仓库目录下执行此命令"
  end

  # 获取git仓库根目录
  root_dir = git_root_directory(local_repo_dir: current_project_dir)
  if root_dir.nil?
    Funlog.instance.fancyinfo_error("无法获取git仓库根目录")
    raise Informative, "获取git仓库根目录失败"
  end

  

  release_branch = "master"
  process_need_add_files(project_dir: root_dir)

  current_branch = git!(%W(-C #{root_dir} rev-parse --abbrev-ref HEAD)).strip
  coding_branch = current_branch

  Funlog.instance.fancyinfo_start("开始合并到#{release_branch}分支")
  merge_to_release_branch(
    project_dir: root_dir, 
    release_branch: release_branch, 
    coding_branch: coding_branch
  )

  add_release_tag(
    project_dir: root_dir,
    increment_mode: @mode,
    force_retag: @force_retag
  )
end