hub_test.rb 23.9 KB
Newer Older
1 2
$LOAD_PATH.unshift File.dirname(__FILE__)
require 'helper'
3
require 'webmock/test_unit'
C
Chris Wanstrath 已提交
4

5 6 7 8 9 10
WebMock::BodyPattern.class_eval do
  undef normalize_hash
  # override normalizing hash since it otherwise requires JSON
  def normalize_hash(hash) hash end
end

C
Chris Wanstrath 已提交
11
class HubTest < Test::Unit::TestCase
12 13 14 15 16
  if defined? WebMock::API
    include WebMock::API
  else
    include WebMock
  end
C
Chris Wanstrath 已提交
17

18 19 20 21 22 23 24 25 26
  COMMANDS = []

  Hub::Commands.class_eval do
    remove_method :command?
    define_method :command? do |name|
      COMMANDS.include?(name)
    end
  end

27
  def setup
28
    COMMANDS.replace %w[open groff]
29
    Hub::Context::DIRNAME.replace 'hub'
30
    Hub::Context::REMOTES.clear
31

32 33 34
    @git = Hub::Context::GIT_CONFIG.replace(Hash.new { |h, k|
      raise ArgumentError, "`git #{k}` not stubbed"
    }).update(
35
      'remote' => "mislav\norigin",
36 37 38
      'symbolic-ref -q HEAD' => 'refs/heads/master',
      'config github.user'   => 'tpw',
      'config github.token'  => 'abc123',
39 40
      'config --get-all remote.origin.url' => 'git://github.com/defunkt/hub.git',
      'config --get-all remote.mislav.url' => 'git://github.com/mislav/hub.git',
41 42 43 44
      'config branch.master.remote'  => 'origin',
      'config branch.master.merge'   => 'refs/heads/master',
      'config branch.feature.remote' => 'mislav',
      'config branch.feature.merge'  => 'refs/heads/experimental',
45 46
      'config --bool hub.http-clone' => 'false',
      'config core.repositoryformatversion' => '0'
47
    )
48
    super
49 50
  end

C
Chris Wanstrath 已提交
51
  def test_private_clone
52 53
    input   = "clone -p rtomayko/ronn"
    command = "git clone git@github.com:rtomayko/ronn.git"
C
Chris Wanstrath 已提交
54
    assert_command input, command
C
Chris Wanstrath 已提交
55 56 57
  end

  def test_public_clone
58 59
    input   = "clone rtomayko/ronn"
    command = "git clone git://github.com/rtomayko/ronn.git"
C
Chris Wanstrath 已提交
60
    assert_command input, command
C
Chris Wanstrath 已提交
61 62
  end

63 64 65 66 67 68 69 70 71 72 73 74
  def test_your_private_clone
    input   = "clone -p resque"
    command = "git clone git@github.com:tpw/resque.git"
    assert_command input, command
  end

  def test_your_public_clone
    input   = "clone resque"
    command = "git clone git://github.com/tpw/resque.git"
    assert_command input, command
  end

75 76 77 78 79 80
  def test_clone_with_arguments_and_path
    input   = "clone --bare -o master -- resque"
    command = "git clone --bare -o master -- git://github.com/tpw/resque.git"
    assert_command input, command
  end

81 82
  def test_your_private_clone_fails_without_config
    out = hub("clone -p mustache") do
83
      stub_github_user(nil)
84 85 86 87 88 89 90
    end

    assert_equal "** No GitHub user set. See http://github.com/guides/local-github-config\n", out
  end

  def test_your_public_clone_fails_without_config
    out = hub("clone mustache") do
91
      stub_github_user(nil)
92 93 94 95 96
    end

    assert_equal "** No GitHub user set. See http://github.com/guides/local-github-config\n", out
  end

97
  def test_private_clone_left_alone
98 99
    input   = "clone git@github.com:rtomayko/ronn.git"
    command = "git clone git@github.com:rtomayko/ronn.git"
100 101 102 103
    assert_command input, command
  end

  def test_public_clone_left_alone
104 105
    input   = "clone git://github.com/rtomayko/ronn.git"
    command = "git clone git://github.com/rtomayko/ronn.git"
106 107
    assert_command input, command
  end
C
Chris Wanstrath 已提交
108 109

  def test_normal_public_clone_with_path
110 111
    input   = "clone git://github.com/rtomayko/ronn.git ronn-dev"
    command = "git clone git://github.com/rtomayko/ronn.git ronn-dev"
C
Chris Wanstrath 已提交
112 113
    assert_command input, command
  end
114 115 116 117 118 119

  def test_normal_clone_from_path
    input   = "clone ./test"
    command = "git clone ./test"
    assert_command input, command
  end
120

121 122 123 124 125 126 127 128 129 130 131 132
  def test_remote_origin
    input   = "remote add origin"
    command = "git remote add origin git://github.com/tpw/hub.git"
    assert_command input, command
  end

  def test_private_remote_origin
    input   = "remote add -p origin"
    command = "git remote add origin git@github.com:tpw/hub.git"
    assert_command input, command
  end

133 134 135 136 137 138
  def test_public_remote_origin_as_normal
    input   = "remote add origin http://github.com/defunkt/resque.git"
    command = "git remote add origin http://github.com/defunkt/resque.git"
    assert_command input, command
  end

139 140 141 142 143 144 145 146 147 148 149 150
  def test_remote_from_rel_path
    input = "remote add origin ./path"
    command = "git remote add origin ./path"
    assert_command input, command
  end

  def test_remote_from_abs_path
    input = "remote add origin /path"
    command = "git remote add origin /path"
    assert_command input, command
  end

151
  def test_private_remote_origin_as_normal
C
Chris Wanstrath 已提交
152 153 154 155 156
    input   = "remote add origin git@github.com:defunkt/resque.git"
    command = "git remote add origin git@github.com:defunkt/resque.git"
    assert_command input, command
  end

S
Stephen Celis 已提交
157 158
  def test_public_submodule
    input   = "submodule add wycats/bundler vendor/bundler"
J
Justin Ridgewell 已提交
159 160
    command = "git submodule add git://github.com/wycats/bundler.git vendor/bundler"
    assert_command input, command
S
Stephen Celis 已提交
161 162 163 164
  end

  def test_private_submodule
    input   = "submodule add -p grit vendor/grit"
J
Justin Ridgewell 已提交
165 166 167 168 169 170 171 172
    command = "git submodule add git@github.com:tpw/grit.git vendor/grit"
    assert_command input, command
  end

  def test_submodule_branch
    input   = "submodule add -b ryppl ryppl/pip vendor/pip"
    command = "git submodule add -b ryppl git://github.com/ryppl/pip.git vendor/pip"
    assert_command input, command
S
Stephen Celis 已提交
173 174 175 176 177
  end

  def test_submodule_with_args
    input   = "submodule -q add --bare -- grit grit"
    command = "git submodule -q add --bare -- git://github.com/tpw/grit.git grit"
J
Justin Ridgewell 已提交
178
    assert_command input, command
S
Stephen Celis 已提交
179 180
  end

C
Chris Wanstrath 已提交
181
  def test_private_remote
182
    input   = "remote add -p rtomayko"
C
Chris Wanstrath 已提交
183 184
    command = "git remote add rtomayko git@github.com:rtomayko/hub.git"
    assert_command input, command
C
Chris Wanstrath 已提交
185 186 187
  end

  def test_public_remote
188
    input   = "remote add rtomayko"
C
Chris Wanstrath 已提交
189 190
    command = "git remote add rtomayko git://github.com/rtomayko/hub.git"
    assert_command input, command
C
Chris Wanstrath 已提交
191 192
  end

C
Chris Wanstrath 已提交
193 194 195 196 197 198
  def test_public_remote_f
    input   = "remote add -f rtomayko"
    command = "git remote add -f rtomayko git://github.com/rtomayko/hub.git"
    assert_command input, command
  end

199 200 201 202 203 204
  def test_named_public_remote
    input   = "remote add origin rtomayko"
    command = "git remote add origin git://github.com/rtomayko/hub.git"
    assert_command input, command
  end

205 206 207 208 209 210
  def test_named_public_remote_f
    input   = "remote add -f origin rtomayko"
    command = "git remote add -f origin git://github.com/rtomayko/hub.git"
    assert_command input, command
  end

C
Chris Wanstrath 已提交
211
  def test_private_remote_with_repo
212 213
    input   = "remote add -p jashkenas/coffee-script"
    command = "git remote add jashkenas git@github.com:jashkenas/coffee-script.git"
C
Chris Wanstrath 已提交
214 215 216 217
    assert_command input, command
  end

  def test_public_remote_with_repo
218 219
    input   = "remote add jashkenas/coffee-script"
    command = "git remote add jashkenas git://github.com/jashkenas/coffee-script.git"
220 221 222 223
    assert_command input, command
  end

  def test_public_remote_f_with_repo
224 225
    input   = "remote add -f jashkenas/coffee-script"
    command = "git remote add -f jashkenas git://github.com/jashkenas/coffee-script.git"
C
Chris Wanstrath 已提交
226 227 228
    assert_command input, command
  end

229
  def test_named_private_remote_with_repo
230 231
    input   = "remote add -p origin jashkenas/coffee-script"
    command = "git remote add origin git@github.com:jashkenas/coffee-script.git"
232 233 234
    assert_command input, command
  end

235 236 237 238 239 240 241 242
  def test_fetch_existing_remote
    assert_command "fetch mislav", "git fetch mislav"
  end

  def test_fetch_new_remote
    stub_remotes_group('xoebus', nil)
    stub_existing_fork('xoebus')

243 244 245
    assert_commands "git remote add xoebus git://github.com/xoebus/hub.git",
                    "git fetch xoebus",
                    "fetch xoebus"
246 247 248 249 250 251
  end

  def test_fetch_new_remote_with_options
    stub_remotes_group('xoebus', nil)
    stub_existing_fork('xoebus')

252 253 254
    assert_commands "git remote add xoebus git://github.com/xoebus/hub.git",
                    "git fetch --depth=1 --prune xoebus",
                    "fetch --depth=1 --prune xoebus"
255 256 257 258 259 260 261 262
  end

  def test_fetch_multiple_new_remotes
    stub_remotes_group('xoebus', nil)
    stub_remotes_group('rtomayko', nil)
    stub_existing_fork('xoebus')
    stub_existing_fork('rtomayko')

263 264 265 266
    assert_commands "git remote add xoebus git://github.com/xoebus/hub.git",
                    "git remote add rtomayko git://github.com/rtomayko/hub.git",
                    "git fetch --multiple xoebus rtomayko",
                    "fetch --multiple xoebus rtomayko"
267 268 269 270 271 272 273 274
  end

  def test_fetch_multiple_comma_separated_remotes
    stub_remotes_group('xoebus', nil)
    stub_remotes_group('rtomayko', nil)
    stub_existing_fork('xoebus')
    stub_existing_fork('rtomayko')

275 276 277 278
    assert_commands "git remote add xoebus git://github.com/xoebus/hub.git",
                    "git remote add rtomayko git://github.com/rtomayko/hub.git",
                    "git fetch --multiple xoebus rtomayko",
                    "fetch xoebus,rtomayko"
279 280 281 282 283 284 285 286 287 288 289 290 291 292
  end

  def test_fetch_multiple_new_remotes_with_filtering
    stub_remotes_group('xoebus', nil)
    stub_remotes_group('mygrp', 'one two')
    stub_remotes_group('typo', nil)
    stub_existing_fork('xoebus')
    stub_nonexisting_fork('typo')

    # mislav: existing remote; skipped
    # xoebus: new remote, fork exists; added
    # mygrp:  a remotes group; skipped
    # URL:    can't be a username; skipped
    # typo:   fork doesn't exist; skipped
293 294 295
    assert_commands "git remote add xoebus git://github.com/xoebus/hub.git",
                    "git fetch --multiple mislav xoebus mygrp git://example.com typo",
                    "fetch --multiple mislav xoebus mygrp git://example.com typo"
296 297
  end

298
  def test_cherry_pick
299
    assert_forwarded "cherry-pick a319d88"
300 301 302
  end

  def test_cherry_pick_url
303
    url = 'http://github.com/mislav/hub/commit/a319d88'
304
    assert_commands "git fetch mislav", "git cherry-pick a319d88", "cherry-pick #{url}"
305 306
  end

307 308
  def test_cherry_pick_url_with_fragment
    url = 'http://github.com/mislav/hub/commit/abcdef0123456789#comments'
309
    assert_commands "git fetch mislav", "git cherry-pick abcdef0123456789", "cherry-pick #{url}"
310 311
  end

312 313
  def test_cherry_pick_url_with_remote_add
    url = 'http://github.com/xoebus/hub/commit/a319d88'
314 315 316
    assert_commands "git remote add -f xoebus git://github.com/xoebus/hub.git",
                    "git cherry-pick a319d88",
                    "cherry-pick #{url}"
317 318 319 320
  end

  def test_cherry_pick_private_url_with_remote_add
    url = 'https://github.com/xoebus/hub/commit/a319d88'
321 322 323
    assert_commands "git remote add -f xoebus git@github.com:xoebus/hub.git",
                    "git cherry-pick a319d88",
                    "cherry-pick #{url}"
324 325 326 327
  end

  def test_cherry_pick_origin_url
    url = 'https://github.com/defunkt/hub/commit/a319d88'
328
    assert_commands "git fetch origin", "git cherry-pick a319d88", "cherry-pick #{url}"
329 330 331
  end

  def test_cherry_pick_github_user_notation
332
    assert_commands "git fetch mislav", "git cherry-pick a319d88", "cherry-pick mislav@a319d88"
333 334 335 336
  end

  def test_cherry_pick_github_user_repo_notation
    # not supported
337
    assert_forwarded "cherry-pick mislav/hubbub@a319d88"
338 339 340
  end

  def test_cherry_pick_github_notation_too_short
341
    assert_forwarded "cherry-pick mislav@a319"
342 343 344
  end

  def test_cherry_pick_github_notation_with_remote_add
345 346 347
    assert_commands "git remote add -f xoebus git://github.com/xoebus/hub.git",
                    "git cherry-pick a319d88",
                    "cherry-pick xoebus@a319d88"
348 349
  end

350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371
  def test_am_untouched
    assert_forwarded "am some.patch"
  end

  def test_am_pull_request
    with_tmpdir('/tmp/') do
      assert_commands "curl -#LA 'hub #{Hub::Version}' https://github.com/defunkt/hub/pull/55.patch -o /tmp/55.patch",
                      "git am --signoff /tmp/55.patch -p2",
                      "am --signoff https://github.com/defunkt/hub/pull/55 -p2"
    end
  end

  def test_am_commit_url
    with_tmpdir('/tmp/') do
      url = 'https://github.com/davidbalbert/hub/commit/fdb9921'

      assert_commands "curl -#LA 'hub #{Hub::Version}' #{url}.patch -o /tmp/fdb9921.patch",
                      "git am --signoff /tmp/fdb9921.patch -p2",
                      "am --signoff #{url} -p2"
    end
  end

C
Chris Wanstrath 已提交
372
  def test_init
373
    assert_commands "git init", "git remote add origin git@github.com:tpw/hub.git", "init -g"
374 375 376 377
  end

  def test_init_no_login
    out = hub("init -g") do
378
      stub_github_user(nil)
379 380 381
    end

    assert_equal "** No GitHub user set. See http://github.com/guides/local-github-config\n", out
C
Chris Wanstrath 已提交
382 383
  end

384
  def test_push_two
385 386
    assert_commands "git push origin cool-feature", "git push staging cool-feature",
                    "push origin,staging cool-feature"
387 388
  end

C
Chris Wanstrath 已提交
389
  def test_push_more
390 391 392 393
    assert_commands "git push origin cool-feature",
                    "git push staging cool-feature",
                    "git push qa cool-feature",
                    "push origin,staging,qa cool-feature"
C
Chris Wanstrath 已提交
394
  end
C
Chris Wanstrath 已提交
395

396
  def test_create
397
    stub_no_remotes
398
    stub_nonexisting_fork('tpw')
399 400 401
    stub_request(:post, "github.com/api/v2/yaml/repos/create").
      with(:body => { 'login'=>'tpw', 'token'=>'abc123', 'name' => 'hub' })

402
    expected = "remote add -f origin git@github.com:tpw/hub.git\n"
403
    expected << "created repository: tpw/hub\n"
404 405 406
    assert_equal expected, hub("create") { ENV['GIT'] = 'echo' }
  end

407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427
  def test_create_with_env_authentication
    stub_no_remotes
    stub_nonexisting_fork('mojombo')

    old_user  = ENV['GITHUB_USER']
    old_token = ENV['GITHUB_TOKEN']
    ENV['GITHUB_USER']  = 'mojombo'
    ENV['GITHUB_TOKEN'] = '123abc'

    stub_request(:post, "github.com/api/v2/yaml/repos/create").
      with(:body => { 'login'=>'mojombo', 'token'=>'123abc', 'name' => 'hub' })

    expected = "remote add -f origin git@github.com:mojombo/hub.git\n"
    expected << "created repository: mojombo/hub\n"
    assert_equal expected, hub("create") { ENV['GIT'] = 'echo' }

  ensure
    ENV['GITHUB_USER']  = old_user
    ENV['GITHUB_TOKEN'] = old_token
  end

428
  def test_create_private_repository
429
    stub_no_remotes
430
    stub_nonexisting_fork('tpw')
431 432 433
    stub_request(:post, "github.com/api/v2/yaml/repos/create").
      with(:body => { 'login'=>'tpw', 'token'=>'abc123', 'name' => 'hub', 'public' => '0' })

434
    expected = "remote add -f origin git@github.com:tpw/hub.git\n"
435
    expected << "created repository: tpw/hub\n"
436 437 438
    assert_equal expected, hub("create -p") { ENV['GIT'] = 'echo' }
  end

439
  def test_create_with_description_and_homepage
440
    stub_no_remotes
441
    stub_nonexisting_fork('tpw')
442 443 444 445 446
    stub_request(:post, "github.com/api/v2/yaml/repos/create").with(:body => {
      'login'=>'tpw', 'token'=>'abc123', 'name' => 'hub',
      'description' => 'toyproject', 'homepage' => 'http://example.com'
    })

447
    expected = "remote add -f origin git@github.com:tpw/hub.git\n"
448
    expected << "created repository: tpw/hub\n"
449
    assert_equal expected, hub("create -d toyproject -h http://example.com") { ENV['GIT'] = 'echo' }
450 451 452
  end

  def test_create_with_existing_repository
453
    stub_no_remotes
454 455 456 457
    stub_existing_fork('tpw')

    expected = "tpw/hub already exists on GitHub\n"
    expected << "remote add -f origin git@github.com:tpw/hub.git\n"
458
    expected << "set remote origin: tpw/hub\n"
459 460 461
    assert_equal expected, hub("create") { ENV['GIT'] = 'echo' }
  end

462
  def test_create_no_user
463
    stub_no_remotes
464 465 466 467 468 469
    out = hub("create") do
      stub_github_token(nil)
    end
    assert_equal "** No GitHub token set. See http://github.com/guides/local-github-config\n", out
  end

470
  def test_create_outside_git_repo
471
    stub_no_git_repo
472 473
    assert_equal "'create' must be run from inside a git repository\n", hub("create")
  end
474 475

  def test_create_origin_already_exists
C
Chris Wanstrath 已提交
476
    stub_nonexisting_fork('tpw')
477 478
    stub_request(:post, "github.com/api/v2/yaml/repos/create").
      with(:body => { 'login'=>'tpw', 'token'=>'abc123', 'name' => 'hub' })
C
Chris Wanstrath 已提交
479 480

    expected = "remote -v\ncreated repository: tpw/hub\n"
481 482
    assert_equal expected, hub("create") { ENV['GIT'] = 'echo' }
  end
C
Chris Wanstrath 已提交
483

484
  def test_fork
C
Chris Wanstrath 已提交
485
    stub_nonexisting_fork('tpw')
486 487
    stub_request(:post, "github.com/api/v2/yaml/repos/fork/defunkt/hub").
      with(:body => { 'login'=>'tpw', 'token'=>'abc123' })
C
Chris Wanstrath 已提交
488

C
Chris Wanstrath 已提交
489
    expected = "remote add -f tpw git@github.com:tpw/hub.git\n"
490 491 492 493 494
    expected << "new remote: tpw\n"
    assert_equal expected, hub("fork") { ENV['GIT'] = 'echo' }
  end

  def test_fork_no_remote
495
    stub_nonexisting_fork('tpw')
496
    stub_request(:post, "github.com/api/v2/yaml/repos/fork/defunkt/hub")
C
Chris Wanstrath 已提交
497

498 499 500 501
    assert_equal "", hub("fork --no-remote") { ENV['GIT'] = 'echo' }
  end

  def test_fork_already_exists
502
    stub_existing_fork('tpw')
C
Chris Wanstrath 已提交
503

504
    expected = "tpw/hub already exists on GitHub\n"
C
Chris Wanstrath 已提交
505
    expected << "remote add -f tpw git@github.com:tpw/hub.git\n"
506 507 508
    expected << "new remote: tpw\n"
    assert_equal expected, hub("fork") { ENV['GIT'] = 'echo' }
  end
C
Chris Wanstrath 已提交
509

C
Chris Wanstrath 已提交
510
  def test_version
511
    out = hub('--version')
512
    assert_includes "git version 1.7.0.4", out
C
Chris Wanstrath 已提交
513
    assert_includes "hub version #{Hub::Version}", out
C
Chris Wanstrath 已提交
514
  end
C
Chris Wanstrath 已提交
515

516 517 518 519 520 521 522 523 524 525 526 527 528 529 530
  def test_exec_path
    out = hub('--exec-path')
    assert_equal "/usr/lib/git-core\n", out
  end

  def test_exec_path_arg
    out = hub('--exec-path=/home/wombat/share/my-l33t-git-core')
    assert_equal Hub::Commands.improved_help_text, hub("")
  end

  def test_html_path
    out = hub('--html-path')
    assert_equal "/usr/share/doc/git-doc\n", out
  end

C
Chris Wanstrath 已提交
531 532 533 534 535 536 537
  def test_help
    assert_equal Hub::Commands.improved_help_text, hub("help")
  end

  def test_help_by_default
    assert_equal Hub::Commands.improved_help_text, hub("")
  end
C
Chris Wanstrath 已提交
538

S
Stephen Celis 已提交
539 540 541 542
  def test_help_with_pager
    assert_equal Hub::Commands.improved_help_text, hub("-p")
  end

C
Chris Wanstrath 已提交
543 544 545 546
  def test_help_hub
    help_manpage = hub("help hub")
    assert_includes "git + hub = github", help_manpage
    assert_includes <<-config, help_manpage
C
Chris Wanstrath 已提交
547
Use git-config(1) to display the currently configured GitHub username:
C
Chris Wanstrath 已提交
548 549 550 551
config
  end

  def test_help_hub_no_groff
552 553
    stub_available_commands()
    assert_equal "** Can't find groff(1)\n", hub("help hub")
C
Chris Wanstrath 已提交
554
  end
555

556 557 558 559 560
  def test_hub_standalone
    help_standalone = hub("hub standalone")
    assert_equal Hub::Standalone.build, help_standalone
  end

C
Chris Wanstrath 已提交
561 562 563
  def test_hub_compare
    assert_command "compare refactor",
      "open http://github.com/defunkt/hub/compare/refactor"
564
  end
C
Chris Wanstrath 已提交
565

566 567 568 569
  def test_hub_compare_nothing
    expected = "Usage: hub compare [USER] [<START>...]<END>\n"
    assert_equal expected, hub("compare")
  end
C
Chris Wanstrath 已提交
570

571 572 573 574 575 576 577 578 579 580 581 582 583
  def test_hub_compare_tracking_nothing
    stub_tracking_nothing
    expected = "Usage: hub compare [USER] [<START>...]<END>\n"
    assert_equal expected, hub("compare")
  end

  def test_hub_compare_tracking_branch
    stub_branch('refs/heads/feature')

    assert_command "compare",
      "open http://github.com/mislav/hub/compare/experimental"
  end

584
  def test_hub_compare_range
C
Chris Wanstrath 已提交
585 586
    assert_command "compare 1.0...fix",
      "open http://github.com/defunkt/hub/compare/1.0...fix"
587
  end
C
Chris Wanstrath 已提交
588

589
  def test_hub_compare_fork
C
Chris Wanstrath 已提交
590 591 592 593
    assert_command "compare myfork feature",
      "open http://github.com/myfork/hub/compare/feature"
  end

594 595 596 597 598 599 600 601 602 603 604
  def test_hub_compare_private
    assert_command "compare -p myfork topsecret",
      "open https://github.com/myfork/hub/compare/topsecret"
  end

  def test_hub_compare_url
    assert_command "compare -u 1.0...1.1",
      "echo http://github.com/defunkt/hub/compare/1.0...1.1"
  end

  def test_hub_browse
605
    assert_command "browse mojombo/bert", "open http://github.com/mojombo/bert"
606 607
  end

608 609
  def test_hub_browse_tracking_nothing
    stub_tracking_nothing
610
    assert_command "browse mojombo/bert", "open http://github.com/mojombo/bert"
611 612
  end

613 614 615 616 617
  def test_hub_browse_url
    assert_command "browse -u mojombo/bert", "echo http://github.com/mojombo/bert"
  end

  def test_hub_browse_private
C
Chris Wanstrath 已提交
618 619
    assert_command "browse -p bmizerany/sinatra",
      "open https://github.com/bmizerany/sinatra"
620 621
  end

622
  def test_hub_browse_self
623
    assert_command "browse resque", "open http://github.com/tpw/resque"
624 625
  end

626 627 628 629 630 631 632 633 634 635 636 637
  def test_hub_browse_subpage
    assert_command "browse resque commits",
      "open http://github.com/tpw/resque/commits/master"
    assert_command "browse resque issues",
      "open http://github.com/tpw/resque/issues"
    assert_command "browse resque wiki",
      "open http://wiki.github.com/tpw/resque/"
  end

  def test_hub_browse_on_branch
    stub_branch('refs/heads/feature')

638
    assert_command "browse resque", "open http://github.com/tpw/resque"
639 640 641 642 643 644 645 646 647
    assert_command "browse resque commits",
      "open http://github.com/tpw/resque/commits/master"

    assert_command "browse",
      "open http://github.com/mislav/hub/tree/experimental"
    assert_command "browse -- tree",
      "open http://github.com/mislav/hub/tree/experimental"
    assert_command "browse -- commits",
      "open http://github.com/mislav/hub/commits/experimental"
648 649
  end

650
  def test_hub_browse_self_private
651
    assert_command "browse -p github", "open https://github.com/tpw/github"
652
  end
653

654
  def test_hub_browse_current
655
    assert_command "browse", "open http://github.com/defunkt/hub"
656 657 658 659 660 661 662 663
    assert_command "browse --", "open http://github.com/defunkt/hub"
  end

  def test_hub_browse_current_subpage
    assert_command "browse -- network",
      "open http://github.com/defunkt/hub/network"
    assert_command "browse -- anything/everything",
      "open http://github.com/defunkt/hub/anything/everything"
664 665
  end

666
  def test_hub_browse_current_private
667 668 669
    assert_command "browse -p", "open https://github.com/defunkt/hub"
  end

670 671 672
  def test_hub_browse_no_repo
    stub_repo_url(nil)
    assert_equal "Usage: hub browse [<USER>/]<REPOSITORY>\n", hub("browse")
673
  end
674 675 676 677 678 679 680 681 682 683 684 685 686 687 688 689 690 691 692 693 694 695 696 697 698 699 700 701 702 703 704 705 706 707 708

  def test_custom_browser
    with_browser_env("custom") do
      assert_browser("custom")
    end
  end

  def test_linux_browser
    stub_available_commands "open", "xdg-open", "cygstart"
    with_browser_env(nil) do
      with_ruby_platform("i686-linux") do
        assert_browser("xdg-open")
      end
    end
  end

  def test_cygwin_browser
    stub_available_commands "open", "cygstart"
    with_browser_env(nil) do
      with_ruby_platform("i686-linux") do
        assert_browser("cygstart")
      end
    end
  end

  def test_no_browser
    stub_available_commands()
    expected = "Please set $BROWSER to a web launcher to use this command.\n"
    with_browser_env(nil) do
      with_ruby_platform("i686-linux") do
        assert_equal expected, hub("browse")
      end
    end
  end

709 710 711 712
  def test_context_method_doesnt_hijack_git_command
    assert_command 'remotes', 'git remotes'
  end

713 714 715 716 717
  def test_not_choking_on_ruby_methods
    assert_forwarded 'id'
    assert_forwarded 'name'
  end

718 719 720 721 722
  def test_multiple_remote_urls
    stub_repo_url("git://example.com/other.git\ngit://github.com/my/repo.git")
    assert_command "browse", "open http://github.com/my/repo"
  end

723 724
  protected

725 726 727 728
    def stub_github_user(name)
      @git['config github.user'] = name
    end

729 730 731 732
    def stub_github_token(token)
      @git['config github.token'] = token
    end

733
    def stub_repo_url(value)
734
      @git['config --get-all remote.origin.url'] = value
735 736 737 738 739 740 741 742 743 744 745 746
      Hub::Context::REMOTES.clear
    end

    def stub_branch(value)
      @git['symbolic-ref -q HEAD'] = value
    end

    def stub_tracking_nothing
      @git['config branch.master.remote'] = nil
      @git['config branch.master.merge'] = nil
    end

747 748 749 750
    def stub_remotes_group(name, value)
      @git["config remotes.#{name}"] = value
    end

751 752 753 754 755 756 757 758
    def stub_no_remotes
      @git['remote'] = ''
    end

    def stub_no_git_repo
      @git.replace({})
    end

759 760 761 762 763 764 765 766 767 768 769 770 771
    def stub_existing_fork(user)
      stub_fork(user, 200)
    end

    def stub_nonexisting_fork(user)
      stub_fork(user, 404)
    end

    def stub_fork(user, status)
      stub_request(:get, "github.com/api/v2/yaml/repos/show/#{user}/hub").
        to_return(:status => status)
    end

772 773 774 775 776 777 778 779 780 781 782
    def stub_available_commands(*names)
      COMMANDS.replace names
    end

    def with_browser_env(value)
      browser, ENV['BROWSER'] = ENV['BROWSER'], value
      yield
    ensure
      ENV['BROWSER'] = browser
    end

783 784 785 786 787 788 789
    def with_tmpdir(value)
      dir, ENV['TMPDIR'] = ENV['TMPDIR'], value
      yield
    ensure
      ENV['TMPDIR'] = dir
    end

790 791 792 793 794 795 796 797 798 799 800 801 802 803
    def assert_browser(browser)
      assert_command "browse", "#{browser} http://github.com/defunkt/hub"
    end

    def with_ruby_platform(value)
      platform = RUBY_PLATFORM
      Object.send(:remove_const, :RUBY_PLATFORM)
      Object.const_set(:RUBY_PLATFORM, value)
      yield
    ensure
      Object.send(:remove_const, :RUBY_PLATFORM)
      Object.const_set(:RUBY_PLATFORM, platform)
    end

C
Chris Wanstrath 已提交
804
end