bitbucket에서 github로 모든 분기가있는 git 저장소를 이동하는 방법은 무엇입니까?


334

모든 분기와 전체 히스토리가있는 git 저장소를 bitbucket에서 github로 옮기는 가장 좋은 방법은 무엇입니까? 사용해야하는 스크립트 나 명령 목록이 있습니까?


9
Github는 다음과 같은 툴과 문서를 제공합니다 : help.github.com/articles/… help.github.com/articles/…
mems

: 조금 늦게 파티에 그러나 여기에서 마이그레이션하는 스크립트입니다 gist.github.com/chinmaya-n/cff02f1277c811deab2e550f2aad9967
inblueswithu

답변:


394

당신은 GitHub의 페이지 "를 참조 할 수 있습니다 저장소를 복제 "

다음을 사용합니다.

그것은 줄 것이다 :

git clone --mirror https://bitbucket.org/exampleuser/repository-to-mirror.git
# Make a bare mirrored clone of the repository

cd repository-to-mirror.git
git remote set-url --push origin https://github.com/exampleuser/mirrored
# Set the push location to your mirror

git push --mirror

언급 한 바와 같이 주석에 의해이야 LS :


2
이 방법으로 인해 문제가 발생했습니다 (문제인지 확실하지 않음). 비트 버킷에서 github으로 리포지를 미러링하면 5 가지 분기에 대해 github에서 "비교 및 풀 요청"으로 표시됩니다. github에서 분기로 표시되지 않습니다. 그것에 대해 무엇을 할 수 있습니까?
Siddharth

2
그리고 이슈와 위키는 어떻습니까?
Vladislav Rastrusny

@FractalizeR wiki는 복제 할 수있는 또 다른 저장소입니다 ( github.com/blog/699-making-github-more-open-git-backed-wikis ). 그래도 문제를 쉽게 복제 할 수있는 방법은 없습니다. Api (다양한 GitHub 리포지토리 백업 프로그램은 addyosmani.com/blog/backing-up-a-github-account )를 사용해야 합니다.
VonC

3
리포지토리를 Github에서 Bitbucket으로 이동하려는 경우에도 작동합니다.
simeg

1
큰 파일 (> 100MB)이 포함되어 있기 때문에이 방법을 사용하여 리포지를 복사해야했습니다. GH는 저장소에 큰 파일을 허용하지 않습니다. 리포지를 GH로 푸시하기 전에 큰 파일을 제거했습니다. 그러나 GH의 " 저장소 가져 오기 "기능을 사용하는 다른 문제는 제외하고 다른 답변 에서 설명한대로 시간의 95 % 를 올바른 선택 입니다. 그러나 실패하면 유용한 오류 메시지가 표시되지 않으며 GH 지원 센터에 문의해야합니다. 파일 크기 제한을 알려주는 것은 GH 지원이었습니다.
LS

412

매우 간단합니다.

GitHub에서 빈 저장소를 새로 만듭니다 (readme 또는 licesne없이 이전에 추가 할 수 있음). 다음 화면이 표시됩니다.

내부에서 코드 옵션 을 가져 오기 위해 비트 버킷 URL의 저장소와 모음을 붙여 넣습니다!

가져 오기 코드를 클릭하십시오


12
허용 된 답변은 일반적인 Git 관련 방법 만 표시하므로 실제로 여기에서 질문에 대답하고 있습니다. 당신의 대답은 훨씬 간단합니다!
Emile Bergeron

6
이 답변은 리포지토리를 생성하는 사람이고 맨손 인 한 좋은 것입니다. 다른 사람이 당신을 위해 저장소를 만들면 (예 : 지점 생성 담당)과 같이 작동하지 않습니다. 그런 경우에 받아 들여질 수있는 방법이 있습니다.
Master.Aurora

4
이 방법을 사용하려는 다른 사람들을 위해이 서비스를 사용하기 전에 이전 리포지토리 호스트에서 2 단계 인증을 제거하십시오. 그것을 /import끝내는 유일한 방법 은 URL 끝에 있는를 바꾸고 /settings설정에 액세스하는 것입니다.
Dave

1
작은 경고-문제가 발생하면 설명 오류 메시지가 표시되지 않습니다.
RobertH

GH의 " 저장소 가져 오기 "기능을 사용하는 것이 시간의 95 % 에 맞는 선택 입니다. 그러나 실패하면 유용한 오류 메시지가 표시되지 않으며 GH 지원 센터에 문의해야합니다. 큰 파일 (> 100MB)이 포함되어 있기 때문에 하나의 저장소에 사용할 수 없습니다. 리포지토리를 GH로 푸시하기 직전에 큰 파일을 제거 하고 허용되는 CLI 방법 을 사용해야했습니다 .
LS

125

github에서 "코드 가져 오기"버튼을 찾을 수없는 경우 다음을 수행 할 수 있습니다.

  1. Github Importer를 직접 열고을 입력하십시오 url. 다음과 같이 보일 것입니다 :github importer의 스크린 샷
  2. 이름을 지정하십시오 (또는 이름을 자동으로 가져옵니다).
  3. 선택 Public또는 Private저장소
  4. 딸깍 하는 소리 Begin Import

업데이트 : 최근 Github에서는 "할 수있는 기능 발표 큰 파일 가져 오기 저장소를 "


불행히도 이것은 codebasehq에서 git-hub로 이동하려는 시도에서 작동하지 않았으며 "이 URL은 지원되지 않습니다". :(
sjmcdowall

@sjmcdowall 미안하지는 않았지만 codebasehq URL이 git 저장소를 가리 키기 때문에 작동해야한다고 생각합니다.
biniam

URL이 더 이상 존재하지 않습니다.
코딩 된 컨테이너

@CodedContainer 방금 확인했는데 작동합니다. 방금 스크린 샷을 업데이트했습니다.
biniam

1
GitHub에 로그인하지 않은 경우 URL은 404 만 반환합니다. 로그인 한 경우 가져 오기 도구가로드됩니다.
Jason이

41

http://www.blackdogfoundry.com/blog/moving-repository-from-bitbucket-to-github/

이것은 하나의 자식 공급자에서 다른 공급자로 이동하는 데 도움이되었습니다. 그 끝에서 모든 커밋은 대상 git에있었습니다. 간단하고 똑바로.

git remote rename origin bitbucket
git remote add origin https://github.com/edwardaux/Pipelines.git
git push origin master

푸시가 GitHub에 성공적으로 적용되었다는 사실에 만족하면 다음을 실행하여 기존 리모컨을 삭제할 수 있습니다.

git remote rm bitbucket

2
링크에 관련 될 수있는 부분은 답변에 포함 시키십시오.
k0pernikus

"오류 : som ref를 ' url.gi ' 로 푸시하지 못했습니다. 힌트 : 원격에 로컬에없는 작업이 포함되어 있기 때문에 업데이트가 거부되었습니다. 일반적으로 다른 리포지토리가 동일한 참조를 푸시했기 때문에 발생합니다. 다시 밀기 전에 먼저 원격 변경 사항 (예 : pull ...)을 통합하고 싶을 수 있습니다 자세한 내용은 git push --help에서 빨리 감기에 대한 참고 사항 참조
코딩 된 컨테이너

1
이것은 2019
Wang-Zhao-Liu QM

13

기존 저장소를 github에서 bitbucket으로 가져 오는 역 유스 케이스가있었습니다.

Bitbucket은 가져 오기 도구 도 제공합니다. 유일하게 필요한 단계는 리포지토리에 URL을 추가하는 것입니다.

다음과 같습니다.

비트 버킷 가져 오기 도구의 스크린 샷


5

나는 이것이 오래된 질문이라는 것을 알고 있습니다. 몇 달 전에 내가 똑같은 일을하려고 할 때 그것을 발견했으며 주어진 대답에 압도되었습니다. 그들은 모두 단품으로 발행 된 명령이나 GitHub 임포터를 통해 Bitbucket에서 GitHub로 한 번에 하나의 저장소를 가져 오는 것을 처리하는 것처럼 보였습니다.

gitter라는 GitHub 프로젝트에서 코드를 가져 와서 내 요구에 맞게 수정했습니다.

요점을 포크 하거나 여기에서 코드를 가져올 수 있습니다.

#!/usr/bin/env ruby
require 'fileutils'

# Originally  -- Dave Deriso        -- deriso@gmail.com
# Contributor -- G. Richard Bellamy -- rbellamy@terradatum.com
# If you contribute, put your name here!
# To get your team ID:
# 1. Go to your GitHub profile, select 'Personal Access Tokens', and create an Access token
# 2. curl -H "Authorization: token <very-long-access-token>" https://api.github.com/orgs/<org-name>/teams
# 3. Find the team name, and grabulate the Team ID
# 4. PROFIT!

#----------------------------------------------------------------------
#your particulars
@access_token = ''
@team_id = ''
@org = ''


#----------------------------------------------------------------------
#the verison of this app
@version = "0.2"

#----------------------------------------------------------------------
#some global params
@create = false
@add = false
@migrate = false
@debug = false
@done = false
@error = false

#----------------------------------------------------------------------
#fancy schmancy color scheme

class String; def c(cc); "\e[#{cc}m#{self}\e[0m" end end
#200.to_i.times{ |i| print i.to_s.c(i) + " " }; puts
@sep = "-".c(90)*95
@sep_pref = ".".c(90)*95
@sep_thick = "+".c(90)*95

#----------------------------------------------------------------------
# greetings

def hello
  puts @sep
  puts "BitBucket to GitHub migrator -- v.#{@version}".c(95)
  #puts @sep_thick
end

def goodbye
  puts @sep
  puts "done!".c(95)
  puts @sep
  exit
end

def puts_title(text)
   puts  @sep, "#{text}".c(36), @sep
end

#----------------------------------------------------------------------
# helper methods

def get_options
  require 'optparse'

  n_options = 0
  show_options = false

  OptionParser.new do |opts|
    opts.banner = @sep +"\nUsage: gitter [options]\n".c(36)
    opts.version = @version
    opts.on('-n', '--name [name]', String, 'Set the name of the new repo') { |value| @repo_name = value; n_options+=1 }
    opts.on('-c', '--create', String, 'Create new repo') { @create = true; n_options+=1 }
    opts.on('-m', '--migrate', String, 'Migrate the repo') { @migrate = true; n_options+=1 }
    opts.on('-a', '--add', String, 'Add repo to team') { @add = true; n_options+=1 }
    opts.on('-l', '--language [language]', String, 'Set language of the new repo') { |value| @language = value.strip.downcase; n_options+=1 }
    opts.on('-d', '--debug', 'Print commands for inspection, doesn\'t actually run them') { @debug = true; n_options+=1 }
    opts.on_tail('-h', '--help', 'Prints this little guide') { show_options = true; n_options+=1 }
    @opts = opts
  end.parse!

  if show_options || n_options == 0
    puts @opts
    puts "\nExamples:".c(36)
    puts 'create new repo: ' + "\t\tgitter -c -l javascript -n node_app".c(93)
    puts 'migrate existing to GitHub: ' + "\tgitter -m -n node_app".c(93)
    puts 'create repo and migrate to it: ' + "\tgitter -c -m -l javascript -n node_app".c(93)
    puts 'create repo, migrate to it, and add it to a team: ' + "\tgitter -c -m -a -l javascript -n node_app".c(93)
    puts "\nNotes:".c(36)
    puts "Access Token for repo is #{@access_token} - change this on line 13"
    puts "Team ID for repo is #{@team_id} - change this on line 14"
    puts "Organization for repo is #{@org} - change this on line 15"
    puts 'The assumption is that the person running the script has SSH access to BitBucket,'
    puts 'and GitHub, and that if the current directory contains a directory with the same'
    puts 'name as the repo to migrated, it will deleted and recreated, or created if it'
    puts 'doesn\'t exist - the repo to migrate is mirrored locally, and then created on'
    puts 'GitHub and pushed from that local clone.'
    puts 'New repos are private by default'
    puts "Doesn\'t like symbols for language (ex. use \'c\' instead of \'c++\')"
    puts @sep
    exit
  end
end

#----------------------------------------------------------------------
# git helper methods

def gitter_create(repo)
  if @language
    %q[curl https://api.github.com/orgs/] + @org + %q[/repos -H "Authorization: token ] + @access_token + %q[" -d '{"name":"] + repo + %q[","private":true,"language":"] + @language + %q["}']
  else
    %q[curl https://api.github.com/orgs/] + @org + %q[/repos -H "Authorization: token ] + @access_token + %q[" -d '{"name":"] + repo + %q[","private":true}']
  end
end

def gitter_add(repo)
  if @language
    %q[curl https://api.github.com/teams/] + @team_id + %q[/repos/] + @org + %q[/] + repo + %q[ -H "Accept: application/vnd.github.v3+json" -H "Authorization: token ] + @access_token + %q[" -d '{"permission":"pull","language":"] + @language + %q["}']
  else
    %q[curl https://api.github.com/teams/] + @team_id + %q[/repos/] + @org + %q[/] + repo + %q[ -H "Accept: application/vnd.github.v3+json" -H "Authorization: token ] + @access_token + %q[" -d '{"permission":"pull"}']
  end
end

def git_clone_mirror(bitbucket_origin, path)
  "git clone --mirror #{bitbucket_origin}"
end

def git_push_mirror(github_origin, path)
  "(cd './#{path}' && git push --mirror #{github_origin} && cd ..)"
end

def show_pwd
  if @debug
    Dir.getwd()
  end
end

def git_list_origin(path)
  "(cd './#{path}' && git config remote.origin.url && cd ..)"
end

# error checks

def has_repo
  File.exist?('.git')
end

def has_repo_or_error(show_error)
  @repo_exists = has_repo
  if !@repo_exists
    puts 'Error: no .git folder in current directory'.c(91) if show_error
    @error = true
  end
  "has repo: #{@repo_exists}"
end

def has_repo_name_or_error(show_error)
  @repo_name_exists = !(defined?(@repo_name)).nil?
  if !@repo_name_exists
    puts 'Error: repo name missing (-n your_name_here)'.c(91) if show_error
    @error = true
  end
end

#----------------------------------------------------------------------
# main methods
def run(commands)
  if @debug
    commands.each { |x| puts(x) }
  else
    commands.each { |x| system(x) }
  end
end

def set_globals

  puts_title 'Parameters'

  @git_bitbucket_origin =   "git@bitbucket.org:#{@org}/#{@repo_name}.git"
  @git_github_origin = "git@github.com:#{@org}/#{@repo_name}.git"

  puts 'debug: ' + @debug.to_s.c(93)
  puts 'working in: ' + Dir.pwd.c(93)
  puts 'create: ' + @create.to_s.c(93)
  puts 'migrate: ' + @migrate.to_s.c(93)
  puts 'add: ' + @add.to_s.c(93)
  puts 'language: ' + @language.to_s.c(93)
  puts 'repo name: '+ @repo_name.to_s.c(93)
  puts 'bitbucket: ' + @git_bitbucket_origin.to_s.c(93)
  puts 'github: ' + @git_github_origin.to_s.c(93)
  puts 'team_id: ' + @team_id.to_s.c(93)
  puts 'org: ' + @org.to_s.c(93)
end

def create_repo
  puts_title 'Creating'

  #error checks
  has_repo_name_or_error(true)
  goodbye if @error

  puts @sep

  commands = [
      gitter_create(@repo_name)
  ]

  run commands
end


def add_repo
  puts_title 'Adding repo to team'

  #error checks
  has_repo_name_or_error(true)
  goodbye if @error

  puts @sep

  commands = [
      gitter_add(@repo_name)
  ]

  run commands
end

def migrate_repo

  puts_title "Migrating Repo to #{@repo_provider}"

  #error checks
  has_repo_name_or_error(true)
  goodbye if @error

  if Dir.exists?("#{@repo_name}.git")
    puts "#{@repo_name} already exists... recursively deleting."
    FileUtils.rm_r("#{@repo_name}.git")
  end

  path = "#{@repo_name}.git"
  commands = [
    git_clone_mirror(@git_bitbucket_origin, path),
    git_list_origin(path),
    git_push_mirror(@git_github_origin, path)
  ]

  run commands
end

#----------------------------------------------------------------------
#sequence control
hello
get_options

#do stuff
set_globals
create_repo if @create
migrate_repo if @migrate
add_repo if @add

#peace out
goodbye

그런 다음 스크립트를 사용하십시오.

# create a list of repos
foo
bar
baz

# execute the script, iterating over your list
while read p; do ./bitbucket-to-github.rb -a -n $p; done<repos

# good nuff

5

있다 GitHub의 가져 오기와 가져 오기 저장소는

다른 버전 제어 시스템에서 Mercurial로 프로젝트를 호스팅하는 경우 GitHub 임포터 도구를 사용하여 자동으로 GitHub로 프로젝트를 가져올 수 있습니다.

  1. 페이지 오른쪽 상단에서을 클릭 한 다음 저장소 가져 오기를 클릭하십시오.
  2. "이전 저장소의 복제 URL"아래에서 가져올 프로젝트의 URL을 입력하십시오.
  3. 리포지토리를 소유 할 사용자 계정 또는 조직을 선택한 다음 GitHub에서 리포지토리 이름을 입력하십시오.
  4. 새 저장소가 공용인지 개인용인지 지정하십시오.
    • 공개 리포지토리는 GitHub의 모든 사용자가 볼 수 있으므로 GitHub의 공동 작업 커뮤니티를 활용할 수 있습니다.
    • 공용 또는 개인 저장소 단일 선택 단추 개인 저장소는 저장소 소유자 및 공유하기로 선택한 모든 협업 자만 사용할 수 있습니다.
  5. 입력 한 정보를 검토 한 다음 가져 오기 시작을 클릭하십시오.

리포지토리를 완전히 가져 오면 이메일이 전송됩니다.

  1. https://help.github.com/categories/importing-your-projects-to-github
  2. https://help.github.com/articles/importing-a-repository-with-github-importer/

2

로컬 git 저장소를 다른 업스트림으로 옮기려면 다음을 수행하십시오.

현재 원격 URL을 얻으려면 :

자식 원격 URL 원점

https://bitbucket.com/git/myrepo 와 같은 내용이 표시됩니다.

새 원격 저장소를 설정하려면 다음을 수행하십시오.

git remote set-url origin git@github.com : 폴더 /myrepo.git

이제 현재 (개발) 브랜치의 내용을 푸시하십시오.

git push-설정 업스트림 원점 개발

이제 새 리모컨에 지점의 전체 사본이 있습니다.

선택적 으로이 로컬 폴더의 원래 자식 원격으로 돌아갑니다.

git remote set-url origin https://bitbucket.com/git/myrepo

다른 폴더의 github에서 새 git-repository를 얻을 수 있으므로 이전 원격 (bitbucket)과 사용 가능한 새 원격 폴더를 가리키는 두 개의 로컬 폴더가 있습니다.


2

모든 Bitbucket (사용자) 리포지토리를 개인 리포지토리 로 GitHub 에 복제하기 위해 다음 bash 스크립트를 만들었습니다 .


요구 사항 :

  • jq (명령 줄 JSON 프로세서) | 맥 OS:brew install jq

단계 :

  1. https://github.com/settings/tokens로 이동하여 액세스 토큰을 만듭니다. "리포"범위 만 필요합니다.

  2. move_me.sh작업 폴더에 스크립트를 저장하고 필요에 따라 파일을 편집하십시오.

  3. 잊지 마세요 CHMOD 755

  4. 운영! ./move_me.sh

  5. 당신이 저장 한 시간을 즐기십시오.


노트:

  • 스크립트가있는 디렉토리 (작업 디렉토리) 내에 BitBucket 저장소를 복제합니다.

  • 이 스크립트 BitBucket 리포지토리를 삭제하지 않습니다 .


GitHub의 공개 저장소로 이동해야합니까?

찾아 변경 "private": true"private": false아래.

조직의 저장소를 옮기시겠습니까?

개발자 안내서 를 확인하십시오. 몇 가지 수정 사항이 있습니다.


행복한 이사.

#!/bin/bash

BB_USERNAME=your_bitbucket_username 
BB_PASSWORD=your_bitbucket_password

GH_USERNAME=your_github_username
GH_ACCESS_TOKEN=your_github_access_token

###########################

pagelen=$(curl -s -u $BB_USERNAME:$BB_PASSWORD https://api.bitbucket.org/2.0/repositories/$BB_USERNAME | jq -r '.pagelen')

echo "Total number of pages: $pagelen"

hr () {
  printf '%*s\n' "${COLUMNS:-$(tput cols)}" '' | tr ' ' -  
}

i=1

while [ $i -le $pagelen ]
do
  echo
  echo "* Processing Page: $i..."
  hr  
  pageval=$(curl -s -u $BB_USERNAME:$BB_PASSWORD https://api.bitbucket.org/2.0/repositories/$BB_USERNAME?page=$i)

  next=$(echo $pageval | jq -r '.next')
  slugs=($(echo $pageval | jq -r '.values[] | .slug'))
  repos=($(echo $pageval | jq -r '.values[] | .links.clone[1].href'))

  j=0
  for repo in ${repos[@]}
  do
    echo "$(($j + 1)) = ${repos[$j]}"
    slug=${slugs[$j]}
  git clone --bare $repo 
  cd "$slug.git"
  echo
  echo "* $repo cloned, now creating $slug on github..."  
  echo  

  read -r -d '' PAYLOAD <<EOP
  {
    "name": "$slug",
    "description": "$slug - moved from bitbucket",
    "homepage": "https://github.com/$slug",
    "private": true
  }
  EOP

  curl -H "Authorization: token $GH_ACCESS_TOKEN" --data "$PAYLOAD" \
      https://api.github.com/user/repos
  echo
  echo "* mirroring $repo to github..."  
  echo
  git push --mirror "git@github.com:$GH_USERNAME/$slug.git"
  j=$(( $j + 1 ))
  hr    
  cd ..
  done  
  i=$(( $i + 1 ))
done

1
재미있는 스크립트, 토큰과 결합. 산뜻한. 공감.
VonC

@VonC 감사합니다!
cenk December

1

개인 Git 저장소를 이동하는 단계는 다음과 같습니다.

1 단계 : Github 리포지토리 생성

먼저 Github.com에서 새로운 개인 저장소를 생성하십시오. 저장소를 비워 두는 것이 중요합니다. 예를 들어, 저장소를 작성할 때 README로이 저장소 초기화 옵션을 선택하지 마십시오.

2 단계 : 기존 컨텐츠 이동

다음으로 Github 리포지토리를 Bitbucket 리포지토리의 내용으로 채워야합니다.

  1. Bitbucket 에서 기존 리포지토리확인하십시오 .
    $ git clone https://USER@bitbucket.org/USER/PROJECT.git
  1. Bitbucket에서 체크 아웃 한 리포지토리의 업스트림 원격 으로 새 Github 리포지토리를 추가하십시오 .
    $ cd PROJECT
    $ git remote add upstream https://github.com:USER/PROJECT.git
  1. 모든 브랜치 (아래 : 마스터)와 태그를 Github 리포지토리에 푸시 합니다.
    $ git push upstream master
    $ git push --tags upstream

3 단계 : 이전 리포지토리 정리

마지막으로, 동일한 프로젝트에 두 개의 리포지토리를 두어 개발자가 혼동하지 않도록해야합니다. Bitbucket 리포지토리를 삭제하는 방법은 다음과 같습니다.

  1. Github 리포지토리에 모든 내용이 있는지 다시 확인하십시오.

  2. 이전 Bitbucket 리포지토리의 웹 인터페이스로 이동

  3. 메뉴 옵션 설정> 저장소 삭제를 선택하십시오.

  4. 새 Github 리포지토리의 URL을 리디렉션 URL로 추가

이를 통해 저장소는 Github의 새로운 집으로 완전히 자리 잡았습니다. 모든 개발자에게 알려주십시오!


-1

가장 간단한 방법 :

git remote rename origin repo_bitbucket

git remote add origin https://github.com/abc/repo.git

git push origin master

GitHub로 푸시가 성공하면 다음을 실행하여 이전 리모컨을 삭제하십시오.

git remote rm repo_bitbucket
당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.