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

6 7 8 9 10 11
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 已提交
12
class HubTest < Test::Unit::TestCase
13 14 15 16 17
  if defined? WebMock::API
    include WebMock::API
  else
    include WebMock
  end
C
Chris Wanstrath 已提交
18

19 20
  COMMANDS = []

21 22 23 24
  Hub::Context.class_eval do
    remove_method :which
    define_method :which do |name|
      COMMANDS.include?(name) ? "/usr/bin/#{name}" : nil
25 26 27
    end
  end

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

33 34
    Hub::Context::GIT_CONFIG.executable = 'git'

35
    @git = Hub::Context::GIT_CONFIG.replace(Hash.new { |h, k|
M
Mislav Marohnić 已提交
36 37 38
      unless k.index('config alias.') == 0
        raise ArgumentError, "`git #{k}` not stubbed"
      end
39
    }).update(
40
      'remote' => "mislav\norigin",
41 42 43
      'symbolic-ref -q HEAD' => 'refs/heads/master',
      'config github.user'   => 'tpw',
      'config github.token'  => 'abc123',
44 45
      'config --get-all remote.origin.url' => 'git://github.com/defunkt/hub.git',
      'config --get-all remote.mislav.url' => 'git://github.com/mislav/hub.git',
46
      "name-rev refs/heads/master@{upstream} --name-only --refs='refs/remotes/*' --no-undefined" => 'remotes/origin/master',
47
      'config --bool hub.http-clone' => 'false',
48
      'rev-parse --git-dir' => '.git'
49
    )
50
    super
51 52
  end

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

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

65 66 67 68 69 70 71 72 73 74 75 76
  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

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

83 84 85 86
  def test_clone_with_arguments_and_destination
    assert_forwarded "clone --template=one/two git://github.com/tpw/resque.git --origin master resquetastic"
  end

87 88
  def test_your_private_clone_fails_without_config
    out = hub("clone -p mustache") do
89
      stub_github_user(nil)
90 91
    end

92
    assert_equal "** No GitHub user set. See http://help.github.com/git-email-settings/\n", out
93 94 95 96
  end

  def test_your_public_clone_fails_without_config
    out = hub("clone mustache") do
97
      stub_github_user(nil)
98 99
    end

100
    assert_equal "** No GitHub user set. See http://help.github.com/git-email-settings/\n", out
101 102
  end

103
  def test_private_clone_left_alone
104
    assert_forwarded "clone git@github.com:rtomayko/ronn.git"
105 106 107
  end

  def test_public_clone_left_alone
108
    assert_forwarded "clone git://github.com/rtomayko/ronn.git"
109
  end
C
Chris Wanstrath 已提交
110 111

  def test_normal_public_clone_with_path
112
    assert_forwarded "clone git://github.com/rtomayko/ronn.git ronn-dev"
C
Chris Wanstrath 已提交
113
  end
114 115

  def test_normal_clone_from_path
116
    assert_forwarded "clone ./test"
117
  end
118

M
Mislav Marohnić 已提交
119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137
  def test_alias_expand
    stub_alias 'c', 'clone --bare'
    input   = "c rtomayko/ronn"
    command = "git clone --bare git://github.com/rtomayko/ronn.git"
    assert_command input, command
  end

  def test_alias_expand_advanced
    stub_alias 'c', 'clone --template="white space"'
    input   = "c rtomayko/ronn"
    command = "git clone '--template=white space' git://github.com/rtomayko/ronn.git"
    assert_command input, command
  end

  def test_alias_doesnt_expand_for_unknown_commands
    stub_alias 'c', 'compute --fast'
    assert_forwarded "c rtomayko/ronn"
  end

138 139 140 141 142 143 144 145 146 147 148 149
  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

150 151 152 153 154 155
  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

156
  def test_remote_from_rel_path
157
    assert_forwarded "remote add origin ./path"
158 159 160
  end

  def test_remote_from_abs_path
161
    assert_forwarded "remote add origin /path"
162 163
  end

164
  def test_private_remote_origin_as_normal
165
    assert_forwarded "remote add origin git@github.com:defunkt/resque.git"
C
Chris Wanstrath 已提交
166 167
  end

S
Stephen Celis 已提交
168 169
  def test_public_submodule
    input   = "submodule add wycats/bundler vendor/bundler"
J
Justin Ridgewell 已提交
170 171
    command = "git submodule add git://github.com/wycats/bundler.git vendor/bundler"
    assert_command input, command
S
Stephen Celis 已提交
172 173 174 175
  end

  def test_private_submodule
    input   = "submodule add -p grit vendor/grit"
J
Justin Ridgewell 已提交
176 177 178 179 180 181 182 183
    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 已提交
184 185 186 187 188
  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 已提交
189
    assert_command input, command
S
Stephen Celis 已提交
190 191
  end

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

  def test_public_remote
199
    input   = "remote add rtomayko"
C
Chris Wanstrath 已提交
200 201
    command = "git remote add rtomayko git://github.com/rtomayko/hub.git"
    assert_command input, command
C
Chris Wanstrath 已提交
202 203
  end

C
Chris Wanstrath 已提交
204 205 206 207 208 209
  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

210 211 212 213 214 215
  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

216 217 218 219 220 221
  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 已提交
222
  def test_private_remote_with_repo
223 224
    input   = "remote add -p jashkenas/coffee-script"
    command = "git remote add jashkenas git@github.com:jashkenas/coffee-script.git"
C
Chris Wanstrath 已提交
225 226 227 228
    assert_command input, command
  end

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

  def test_public_remote_f_with_repo
235 236
    input   = "remote add -f jashkenas/coffee-script"
    command = "git remote add -f jashkenas git://github.com/jashkenas/coffee-script.git"
C
Chris Wanstrath 已提交
237 238 239
    assert_command input, command
  end

240
  def test_named_private_remote_with_repo
241 242
    input   = "remote add -p origin jashkenas/coffee-script"
    command = "git remote add origin git@github.com:jashkenas/coffee-script.git"
243 244 245
    assert_command input, command
  end

246
  def test_fetch_existing_remote
247
    assert_forwarded "fetch mislav"
248 249 250 251 252 253
  end

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

254 255 256
    assert_commands "git remote add xoebus git://github.com/xoebus/hub.git",
                    "git fetch xoebus",
                    "fetch xoebus"
257 258 259 260 261 262
  end

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

263 264 265
    assert_commands "git remote add xoebus git://github.com/xoebus/hub.git",
                    "git fetch --depth=1 --prune xoebus",
                    "fetch --depth=1 --prune xoebus"
266 267 268 269 270 271 272 273
  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')

274 275 276 277
    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"
278 279 280 281 282 283 284 285
  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')

286 287 288 289
    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"
290 291 292 293 294 295 296 297 298 299 300 301 302 303
  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
304 305 306
    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"
307 308
  end

309
  def test_cherry_pick
310
    assert_forwarded "cherry-pick a319d88"
311 312 313
  end

  def test_cherry_pick_url
314
    url = 'http://github.com/mislav/hub/commit/a319d88'
315
    assert_commands "git fetch mislav", "git cherry-pick a319d88", "cherry-pick #{url}"
316 317
  end

318 319
  def test_cherry_pick_url_with_fragment
    url = 'http://github.com/mislav/hub/commit/abcdef0123456789#comments'
320
    assert_commands "git fetch mislav", "git cherry-pick abcdef0123456789", "cherry-pick #{url}"
321 322
  end

323 324
  def test_cherry_pick_url_with_remote_add
    url = 'https://github.com/xoebus/hub/commit/a319d88'
325
    assert_commands "git remote add -f xoebus git://github.com/xoebus/hub.git",
326 327
                    "git cherry-pick a319d88",
                    "cherry-pick #{url}"
328 329 330 331
  end

  def test_cherry_pick_origin_url
    url = 'https://github.com/defunkt/hub/commit/a319d88'
332
    assert_commands "git fetch origin", "git cherry-pick a319d88", "cherry-pick #{url}"
333 334 335
  end

  def test_cherry_pick_github_user_notation
336
    assert_commands "git fetch mislav", "git cherry-pick 368af20", "cherry-pick mislav@368af20"
337 338 339 340
  end

  def test_cherry_pick_github_user_repo_notation
    # not supported
341
    assert_forwarded "cherry-pick mislav/hubbub@a319d88"
342 343 344
  end

  def test_cherry_pick_github_notation_too_short
345
    assert_forwarded "cherry-pick mislav@a319"
346 347 348
  end

  def test_cherry_pick_github_notation_with_remote_add
349 350 351
    assert_commands "git remote add -f xoebus git://github.com/xoebus/hub.git",
                    "git cherry-pick a319d88",
                    "cherry-pick xoebus@a319d88"
352 353
  end

354 355 356 357 358 359 360 361 362
  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"
363 364 365

      cmd = Hub("am https://github.com/defunkt/hub/pull/55/files").command
      assert_includes '/pull/55.patch', cmd
366 367 368 369 370 371 372 373 374 375 376 377 378
    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

379 380 381 382 383 384 385 386 387 388
  def test_am_gist
    with_tmpdir('/tmp/') do
      url = 'https://gist.github.com/8da7fb575debd88c54cf'

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

389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423
  def test_apply_untouched
    assert_forwarded "apply some.patch"
  end

  def test_apply_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 apply /tmp/55.patch -p2",
                      "apply https://github.com/defunkt/hub/pull/55 -p2"

      cmd = Hub("apply https://github.com/defunkt/hub/pull/55/files").command
      assert_includes '/pull/55.patch', cmd
    end
  end

  def test_apply_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 apply /tmp/fdb9921.patch -p2",
                      "apply #{url} -p2"
    end
  end

  def test_apply_gist
    with_tmpdir('/tmp/') do
      url = 'https://gist.github.com/8da7fb575debd88c54cf'

      assert_commands "curl -#LA 'hub #{Hub::Version}' #{url}.txt -o /tmp/gist-8da7fb575debd88c54cf.txt",
                      "git apply /tmp/gist-8da7fb575debd88c54cf.txt -p2",
                      "apply #{url} -p2"
    end
  end

C
Chris Wanstrath 已提交
424
  def test_init
425
    assert_commands "git init", "git remote add origin git@github.com:tpw/hub.git", "init -g"
426 427 428 429
  end

  def test_init_no_login
    out = hub("init -g") do
430
      stub_github_user(nil)
431 432
    end

433
    assert_equal "** No GitHub user set. See http://help.github.com/git-email-settings/\n", out
C
Chris Wanstrath 已提交
434 435
  end

436 437 438 439
  def test_push_untouched
    assert_forwarded "push"
  end

440
  def test_push_two
441 442
    assert_commands "git push origin cool-feature", "git push staging cool-feature",
                    "push origin,staging cool-feature"
443 444
  end

445 446 447 448 449 450
  def test_push_current_branch
    stub_branch('refs/heads/cool-feature')
    assert_commands "git push origin cool-feature", "git push staging cool-feature",
                    "push origin,staging"
  end

C
Chris Wanstrath 已提交
451
  def test_push_more
452 453 454 455
    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 已提交
456
  end
C
Chris Wanstrath 已提交
457

458
  def test_create
459
    stub_no_remotes
460
    stub_nonexisting_fork('tpw')
461 462
    stub_request(:post, "https://#{auth}github.com/api/v2/yaml/repos/create").
      with(:body => { 'name' => 'hub' })
463

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

469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490
  def test_create_custom_name
    stub_no_remotes
    stub_nonexisting_fork('tpw', 'hubbub')
    stub_request(:post, "https://#{auth}github.com/api/v2/yaml/repos/create").
      with(:body => { 'name' => 'hubbub' })

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

  def test_create_in_organization
    stub_no_remotes
    stub_nonexisting_fork('acme', 'hubbub')
    stub_request(:post, "https://#{auth}github.com/api/v2/yaml/repos/create").
      with(:body => { 'name' => 'acme/hubbub' })

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

491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506
  def test_create_no_openssl
    stub_no_remotes
    stub_nonexisting_fork('tpw')
    stub_request(:post, "http://#{auth}github.com/api/v2/yaml/repos/create").
      with(:body => { 'name' => 'hub' })

    expected = "remote add -f origin git@github.com:tpw/hub.git\n"
    expected << "created repository: tpw/hub\n"

    assert_equal expected, hub("create") {
      ENV['GIT'] = 'echo'
      require 'net/https'
      Object.send :remove_const, :OpenSSL
    }
  end

507 508 509
  def test_create_failed
    stub_no_remotes
    stub_nonexisting_fork('tpw')
510
    stub_request(:post, "https://#{auth}github.com/api/v2/yaml/repos/create").
511 512
      to_return(:status => [401, "Your token is fail"])

513 514
    expected = "Error creating repository: Your token is fail (HTTP 401)\n"
    expected << "Check your token configuration (`git config github.token`)\n"
515 516 517
    assert_equal expected, hub("create") { ENV['GIT'] = 'echo' }
  end

518 519 520 521 522 523 524 525 526
  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'

527 528
    stub_request(:post, "https://#{auth('mojombo', '123abc')}github.com/api/v2/yaml/repos/create").
      with(:body => { 'name' => 'hub' })
529 530 531 532 533 534 535 536 537 538

    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

539
  def test_create_private_repository
540
    stub_no_remotes
541
    stub_nonexisting_fork('tpw')
542 543
    stub_request(:post, "https://#{auth}github.com/api/v2/yaml/repos/create").
      with(:body => { 'name' => 'hub', 'public' => '0' })
544

545
    expected = "remote add -f origin git@github.com:tpw/hub.git\n"
546
    expected << "created repository: tpw/hub\n"
547 548 549
    assert_equal expected, hub("create -p") { ENV['GIT'] = 'echo' }
  end

550
  def test_create_with_description_and_homepage
551
    stub_no_remotes
552
    stub_nonexisting_fork('tpw')
553 554
    stub_request(:post, "https://#{auth}github.com/api/v2/yaml/repos/create").with(:body => {
      'name' => 'hub', 'description' => 'toyproject', 'homepage' => 'http://example.com'
555 556
    })

557
    expected = "remote add -f origin git@github.com:tpw/hub.git\n"
558
    expected << "created repository: tpw/hub\n"
559
    assert_equal expected, hub("create -d toyproject -h http://example.com") { ENV['GIT'] = 'echo' }
560 561
  end

562 563 564 565 566
  def test_create_with_invalid_arguments
    assert_equal "invalid argument: -a\n",   hub("create -a blah")   { ENV['GIT'] = 'echo' }
    assert_equal "invalid argument: bleh\n", hub("create blah bleh") { ENV['GIT'] = 'echo' }
  end

567
  def test_create_with_existing_repository
568
    stub_no_remotes
569 570 571 572
    stub_existing_fork('tpw')

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

577
  def test_create_no_user
578
    stub_no_remotes
579 580 581
    out = hub("create") do
      stub_github_token(nil)
    end
582
    assert_equal "** No GitHub token set. See http://help.github.com/git-email-settings/\n", out
583 584
  end

585
  def test_create_outside_git_repo
586
    stub_no_git_repo
587 588
    assert_equal "'create' must be run from inside a git repository\n", hub("create")
  end
589 590

  def test_create_origin_already_exists
C
Chris Wanstrath 已提交
591
    stub_nonexisting_fork('tpw')
592 593
    stub_request(:post, "https://#{auth}github.com/api/v2/yaml/repos/create").
      with(:body => { 'name' => 'hub' })
C
Chris Wanstrath 已提交
594 595

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

599
  def test_fork
C
Chris Wanstrath 已提交
600
    stub_nonexisting_fork('tpw')
601
    stub_request(:post, "https://#{auth}github.com/api/v2/yaml/repos/fork/defunkt/hub")
C
Chris Wanstrath 已提交
602

C
Chris Wanstrath 已提交
603
    expected = "remote add -f tpw git@github.com:tpw/hub.git\n"
604 605 606 607
    expected << "new remote: tpw\n"
    assert_equal expected, hub("fork") { ENV['GIT'] = 'echo' }
  end

608 609
  def test_fork_failed
    stub_nonexisting_fork('tpw')
610
    stub_request(:post, "https://#{auth}github.com/api/v2/yaml/repos/fork/defunkt/hub").
611 612
      to_return(:status => [500, "Your fork is fail"])

613
    expected = "Error creating fork: Your fork is fail (HTTP 500)\n"
614 615 616
    assert_equal expected, hub("fork") { ENV['GIT'] = 'echo' }
  end

617
  def test_fork_no_remote
618
    stub_nonexisting_fork('tpw')
619
    stub_request(:post, "https://#{auth}github.com/api/v2/yaml/repos/fork/defunkt/hub")
C
Chris Wanstrath 已提交
620

621 622 623 624
    assert_equal "", hub("fork --no-remote") { ENV['GIT'] = 'echo' }
  end

  def test_fork_already_exists
625
    stub_existing_fork('tpw')
C
Chris Wanstrath 已提交
626

627
    expected = "tpw/hub already exists on GitHub\n"
C
Chris Wanstrath 已提交
628
    expected << "remote add -f tpw git@github.com:tpw/hub.git\n"
629 630 631
    expected << "new remote: tpw\n"
    assert_equal expected, hub("fork") { ENV['GIT'] = 'echo' }
  end
C
Chris Wanstrath 已提交
632

C
Chris Wanstrath 已提交
633
  def test_version
634
    out = hub('--version')
635
    assert_includes "git version 1.7.0.4", out
C
Chris Wanstrath 已提交
636
    assert_includes "hub version #{Hub::Version}", out
C
Chris Wanstrath 已提交
637
  end
C
Chris Wanstrath 已提交
638

639 640 641 642 643 644 645
  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')
646
    assert_equal Hub::Commands.improved_help_text, out
647 648 649 650 651 652 653
  end

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

C
Chris Wanstrath 已提交
654 655 656 657 658 659 660
  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 已提交
661

S
Stephen Celis 已提交
662 663 664 665
  def test_help_with_pager
    assert_equal Hub::Commands.improved_help_text, hub("-p")
  end

C
Chris Wanstrath 已提交
666 667 668 669
  def test_help_hub
    help_manpage = hub("help hub")
    assert_includes "git + hub = github", help_manpage
    assert_includes <<-config, help_manpage
C
Chris Wanstrath 已提交
670
Use git-config(1) to display the currently configured GitHub username:
C
Chris Wanstrath 已提交
671 672 673 674
config
  end

  def test_help_hub_no_groff
675 676
    stub_available_commands()
    assert_equal "** Can't find groff(1)\n", hub("help hub")
C
Chris Wanstrath 已提交
677
  end
678

679 680 681 682 683
  def test_hub_standalone
    help_standalone = hub("hub standalone")
    assert_equal Hub::Standalone.build, help_standalone
  end

C
Chris Wanstrath 已提交
684 685
  def test_hub_compare
    assert_command "compare refactor",
686
      "open https://github.com/defunkt/hub/compare/refactor"
687
  end
C
Chris Wanstrath 已提交
688

689 690 691 692
  def test_hub_compare_nothing
    expected = "Usage: hub compare [USER] [<START>...]<END>\n"
    assert_equal expected, hub("compare")
  end
C
Chris Wanstrath 已提交
693

694 695 696 697 698 699 700 701
  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')
702
    stub_tracking('feature', 'mislav', 'experimental')
703 704

    assert_command "compare",
705
      "open https://github.com/mislav/hub/compare/experimental"
706 707
  end

708
  def test_hub_compare_range
C
Chris Wanstrath 已提交
709
    assert_command "compare 1.0...fix",
710
      "open https://github.com/defunkt/hub/compare/1.0...fix"
711
  end
C
Chris Wanstrath 已提交
712

713 714 715 716 717 718 719 720 721 722 723 724 725 726 727
  def test_hub_compare_range_fixes_two_dots_for_tags
    assert_command "compare 1.0..fix",
      "open https://github.com/defunkt/hub/compare/1.0...fix"
  end

  def test_hub_compare_range_fixes_two_dots_for_shas
    assert_command "compare 1234abc..3456cde",
      "open https://github.com/defunkt/hub/compare/1234abc...3456cde"
  end

  def test_hub_compare_range_ignores_two_dots_for_complex_ranges
    assert_command "compare @{a..b}..@{c..d}",
      "open https://github.com/defunkt/hub/compare/@{a..b}..@{c..d}"
  end

728 729 730 731 732 733
  def test_hub_compare_on_wiki
    stub_repo_url 'git://github.com/defunkt/hub.wiki.git'
    assert_command "compare 1.0...fix",
      "open https://github.com/defunkt/hub/wiki/_compare/1.0...fix"
  end

734
  def test_hub_compare_fork
C
Chris Wanstrath 已提交
735
    assert_command "compare myfork feature",
736
      "open https://github.com/myfork/hub/compare/feature"
737 738 739 740
  end

  def test_hub_compare_url
    assert_command "compare -u 1.0...1.1",
741
      "echo https://github.com/defunkt/hub/compare/1.0...1.1"
742 743 744
  end

  def test_hub_browse
745
    assert_command "browse mojombo/bert", "open https://github.com/mojombo/bert"
746 747
  end

748 749 750 751
  def test_hub_browse_commit
    assert_command "browse mojombo/bert commit/5d5582", "open https://github.com/mojombo/bert/commit/5d5582"
  end

752 753
  def test_hub_browse_tracking_nothing
    stub_tracking_nothing
754
    assert_command "browse mojombo/bert", "open https://github.com/mojombo/bert"
755 756
  end

757
  def test_hub_browse_url
758
    assert_command "browse -u mojombo/bert", "echo https://github.com/mojombo/bert"
759 760
  end

761
  def test_hub_browse_self
762
    assert_command "browse resque", "open https://github.com/tpw/resque"
763 764
  end

765 766
  def test_hub_browse_subpage
    assert_command "browse resque commits",
767
      "open https://github.com/tpw/resque/commits/master"
768
    assert_command "browse resque issues",
769
      "open https://github.com/tpw/resque/issues"
770
    assert_command "browse resque wiki",
771
      "open https://github.com/tpw/resque/wiki"
772 773 774 775
  end

  def test_hub_browse_on_branch
    stub_branch('refs/heads/feature')
776
    stub_tracking('feature', 'mislav', 'experimental')
777

778
    assert_command "browse resque", "open https://github.com/tpw/resque"
779
    assert_command "browse resque commits",
780
      "open https://github.com/tpw/resque/commits/master"
781 782

    assert_command "browse",
783
      "open https://github.com/mislav/hub/tree/experimental"
784
    assert_command "browse -- tree",
785
      "open https://github.com/mislav/hub/tree/experimental"
786
    assert_command "browse -- commits",
787
      "open https://github.com/mislav/hub/commits/experimental"
788
  end
789

790
  def test_hub_browse_current
791 792
    assert_command "browse", "open https://github.com/defunkt/hub"
    assert_command "browse --", "open https://github.com/defunkt/hub"
793 794
  end

795 796 797 798
  def test_hub_browse_commit_from_current
    assert_command "browse -- commit/6616e4", "open https://github.com/defunkt/hub/commit/6616e4"
  end

799 800 801 802 803 804 805
  def test_hub_browse_no_tracking
    stub_tracking_nothing
    assert_command "browse", "open https://github.com/defunkt/hub"
  end

  def test_hub_browse_no_tracking_on_branch
    stub_branch('refs/heads/feature')
806
    stub_tracking_nothing('feature')
807 808 809
    assert_command "browse", "open https://github.com/defunkt/hub"
  end

810 811 812 813 814 815 816 817 818
  def test_hub_browse_current_wiki
    stub_repo_url 'git://github.com/defunkt/hub.wiki.git'

    assert_command "browse", "open https://github.com/defunkt/hub/wiki"
    assert_command "browse -- wiki", "open https://github.com/defunkt/hub/wiki"
    assert_command "browse -- commits", "open https://github.com/defunkt/hub/wiki/_history"
    assert_command "browse -- pages", "open https://github.com/defunkt/hub/wiki/_pages"
  end

819 820
  def test_hub_browse_current_subpage
    assert_command "browse -- network",
821
      "open https://github.com/defunkt/hub/network"
822
    assert_command "browse -- anything/everything",
823
      "open https://github.com/defunkt/hub/anything/everything"
824 825
  end

826 827 828 829
  def test_hub_browse_deprecated_private
    with_browser_env('echo') do
      assert_includes "Warning: the `-p` flag has no effect anymore\n", hub("browse -p defunkt/hub")
    end
830 831
  end

832 833 834
  def test_hub_browse_no_repo
    stub_repo_url(nil)
    assert_equal "Usage: hub browse [<USER>/]<REPOSITORY>\n", hub("browse")
835
  end
836 837 838 839 840 841 842 843 844 845

  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
846
      with_host_os("i686-linux") do
847 848 849 850 851 852 853 854
        assert_browser("xdg-open")
      end
    end
  end

  def test_cygwin_browser
    stub_available_commands "open", "cygstart"
    with_browser_env(nil) do
855
      with_host_os("i686-linux") do
856 857 858 859 860 861 862 863 864
        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
865
      with_host_os("i686-linux") do
866 867 868 869 870
        assert_equal expected, hub("browse")
      end
    end
  end

871
  def test_context_method_doesnt_hijack_git_command
872
    assert_forwarded 'remotes'
873 874
  end

875 876 877 878 879
  def test_not_choking_on_ruby_methods
    assert_forwarded 'id'
    assert_forwarded 'name'
  end

880 881
  def test_multiple_remote_urls
    stub_repo_url("git://example.com/other.git\ngit://github.com/my/repo.git")
882
    assert_command "browse", "open https://github.com/my/repo"
883 884
  end

885 886 887 888 889 890
  def test_global_flags_preserved
    cmd = '--no-pager --bare -c core.awesome=true -c name=value --git-dir=/srv/www perform'
    assert_command cmd, 'git --bare -c core.awesome=true -c name=value --git-dir=/srv/www --no-pager perform'
    assert_equal %w[git --bare -c core.awesome=true -c name=value --git-dir=/srv/www], @git.executable
  end

891 892
  protected

893 894 895 896
    def stub_github_user(name)
      @git['config github.user'] = name
    end

897 898 899 900
    def stub_github_token(token)
      @git['config github.token'] = token
    end

901
    def stub_repo_url(value)
902
      @git['config --get-all remote.origin.url'] = value
903 904 905 906 907 908 909
      Hub::Context::REMOTES.clear
    end

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

910
    def stub_tracking(from, remote_name, remote_branch)
911 912
      value = remote_branch ? "remotes/#{remote_name}/#{remote_branch}" : nil
      @git["name-rev refs/heads/#{from}@{upstream} --name-only --refs='refs/remotes/*' --no-undefined"] = value
913 914
    end

915
    def stub_tracking_nothing
916
      stub_tracking('master', nil, nil)
917 918
    end

919 920 921 922
    def stub_remotes_group(name, value)
      @git["config remotes.#{name}"] = value
    end

923 924 925 926 927 928 929 930
    def stub_no_remotes
      @git['remote'] = ''
    end

    def stub_no_git_repo
      @git.replace({})
    end

M
Mislav Marohnić 已提交
931 932 933 934
    def stub_alias(name, value)
      @git["config alias.#{name}"] = value
    end

935 936
    def stub_existing_fork(user, repo = 'hub')
      stub_fork(user, repo, 200)
937 938
    end

939 940
    def stub_nonexisting_fork(user, repo = 'hub')
      stub_fork(user, repo, 404)
941 942
    end

943 944
    def stub_fork(user, repo, status)
      stub_request(:get, "github.com/api/v2/yaml/repos/show/#{user}/#{repo}").
945 946 947
        to_return(:status => status)
    end

948 949 950 951 952 953 954 955 956 957 958
    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

959 960 961 962 963 964 965
    def with_tmpdir(value)
      dir, ENV['TMPDIR'] = ENV['TMPDIR'], value
      yield
    ensure
      ENV['TMPDIR'] = dir
    end

966
    def assert_browser(browser)
967
      assert_command "browse", "#{browser} https://github.com/defunkt/hub"
968 969
    end

970 971 972 973 974 975 976 977
    def with_host_os(value)
      host_os = RbConfig::CONFIG['host_os']
      RbConfig::CONFIG['host_os'] = value
      begin
        yield
      ensure
        RbConfig::CONFIG['host_os'] = host_os
      end
978 979
    end

980 981 982 983
    def auth(user = @git['config github.user'], password = @git['config github.token'])
      "#{user}%2Ftoken:#{password}@"
    end

C
Chris Wanstrath 已提交
984
end