ชำระเงินไดเรกทอรีย่อยใน Git หรือไม่


160

เป็นไปได้ไหมที่จะตรวจสอบไดเรกทอรีย่อยของที่เก็บใน Git?

ลองนึกภาพฉันกำลังตั้งค่าการติดตั้ง WordPress ใหม่ ฉันจะสร้างสองไดเรกทอรีใหม่สำหรับปลั๊กอินและการปรับแต่งธีมของฉัน:

  • wordpress/wp-content/plugins/myplugins/
  • wordpress/wp-content/themes/mytheme/

ฉันต้องการบำรุงรักษาไดเรกทอรีเหล่านี้ผ่าน Git ในการโค่นล้มฉันจะทำสิ่งนี้โดยมีtrunk/myplugins/และทำtrunk/mytheme/ไดเรกทอรีและตรวจสอบไดเรกทอรีย่อย Git มีวิธีการในการทำงานเดียวกันให้สำเร็จด้วยการใช้แหล่งเก็บข้อมูลเดียวหรือไม่?

ฉันอาจพลาดเรือในกระบวนทัศน์ Git บางอย่างซึ่งเป็นเวลานานสำหรับผู้ใช้ SVN ที่มีการเปิดรับ Git น้อย

แก้ไข: หลายสาขาจัดเก็บเนื้อหาที่แตกต่างกันเป็นวิธีที่น่าสนใจในการจัดการเรื่องนี้


2
ทำไมคุณไม่เช็คเอาต์ repo ทั้งหมดและสร้างลิงก์สัญลักษณ์ไปยังไดเรกทอรีย่อยที่คุณต้องการใช้?
randomness2077


คำตอบง่ายๆที่นี่
Peter Krauss

เป็นไปได้ไหมที่จะทำการเช็คเอาต์และอ้างอิงที่เก็บ Git?
luka5z

คำตอบ:


121

checkouts กระจัดกระจายอยู่ในขณะนี้ใน Git 1.7

ดูคำถาม“ เป็นไปได้ไหมที่จะทำการเช็คเอาต์แบบเบาบางโดยไม่ต้องตรวจสอบที่เก็บทั้งหมดก่อน?

โปรดทราบว่าการชำระเงินแบบเบาบางยังคงต้องการให้คุณดาวน์โหลดพื้นที่เก็บข้อมูลทั้งหมดแม้ว่าไฟล์บางไฟล์การดาวน์โหลด Git จะไม่จบลงในแผนผังการทำงานของคุณ


1
ที่ไหนgit cloneคำสั่งง่ายๆ? ฉันใช้คำตอบนี้ใช้งานได้!
Peter Krauss

4
และมีวิธีการเปลี่ยนชื่อโฟลเดอร์เหล่านั้นหรือไม่? หากฉันเช็คเอาต์กระจัดกระจาย/foo/bar/foobarเป็นไปได้หรือไม่ที่จะเห็นเฉพาะ/foobarในที่เก็บท้องถิ่นของฉัน
graywolf

17

ไม่มีวิธีที่แท้จริงในการทำเช่นนั้นในคอมไพล์ และถ้าคุณจะไม่ทำการเปลี่ยนแปลงที่มีผลต่อต้นไม้ทั้งสองในคราวเดียวเป็นหน่วยงานเดียวไม่มีเหตุผลที่ดีที่จะใช้ที่เก็บเดียวสำหรับทั้งสอง ฉันคิดว่าฉันจะพลาดคุณสมบัติการโค่นล้มนี้ แต่ฉันพบว่าการสร้างที่เก็บมีค่าใช้จ่ายด้านการบริหารเล็กน้อย (เนื่องจากความจริงที่ว่าที่เก็บถูกเก็บไว้ถัดจากสำเนาการทำงานของพวกเขาแทนที่จะกำหนดให้ฉันเลือกสถานที่นอก Working copy) ที่ฉันคุ้นเคยกับการสร้างคลังเก็บข้อมูลขนาดเล็กที่มีจุดประสงค์เดียวจำนวนมาก

หากคุณยืนยัน (หรือต้องการจริงๆ) คุณสามารถสร้างพื้นที่เก็บข้อมูลคอมไพล์ด้วย just mythemeและmypluginsไดเรกทอรีและ symlink จากภายในการติดตั้ง WordPress


MDCore เขียนว่า:

ทำให้การกระทำเช่นตำนานจะเพิ่มจำนวนการแก้ไขสำหรับmyplugin

โปรดทราบว่านี่ไม่ใช่ข้อกังวลสำหรับ git หากคุณตัดสินใจที่จะวางทั้งสองไดเรกทอรีในที่เก็บเดียวเพราะ git ไม่ได้อยู่กับแนวคิดของการเพิ่มจำนวนการแก้ไขที่ซ้ำซากจำเจของรูปแบบใด ๆ

เกณฑ์เพียงอย่างเดียวสำหรับสิ่งที่จะรวมเข้าด้วยกันในแหล่งเก็บข้อมูลเดียวในคอมไพล์คือไม่ว่าจะประกอบด้วยหน่วยเดียวคือ ในกรณีของคุณไม่ว่าจะมีการเปลี่ยนแปลงที่ไม่เหมาะสมที่จะดูการแก้ไขในแต่ละไดเรกทอรีแยก หากคุณมีการเปลี่ยนแปลงที่คุณต้องการแก้ไขไฟล์ในทั้งสองไดเรกทอรีพร้อมกันและการแก้ไขอยู่ด้วยกันพวกเขาควรจะเป็นหนึ่งที่เก็บ ถ้าไม่เช่นนั้นก็อย่าไปเยาะเย้ยพวกเขาด้วยกัน

Git ต้องการให้คุณใช้ที่เก็บแยกต่างหากสำหรับเอนทิตีที่แยกต่างหาก

submodules

Submodules ไม่ได้ระบุความต้องการที่จะเก็บทั้งสองไดเรกทอรีไว้ในที่เก็บเดียวเพราะจริง ๆ แล้วพวกเขาจะบังคับให้มีที่เก็บแยกต่างหากสำหรับแต่ละไดเรกทอรีซึ่งนำมารวมกันในที่เก็บอื่นโดยใช้ submodules เลวร้ายยิ่งกว่าเนื่องจากไดเรกทอรีภายในการติดตั้ง WordPress ไม่ใช่ไดเรกทอรีย่อยโดยตรงของไดเรกทอรีเดียวกันและยังเป็นส่วนหนึ่งของลำดับชั้นที่มีไฟล์อื่น ๆ อีกมากมายการใช้ที่เก็บต่อไดเรกทอรีเป็นไดเรกทอรีย่อยในที่เก็บแบบรวมจะไม่มีประโยชน์ใด ๆ เลย พื้นที่เก็บข้อมูลจะไม่สะท้อนถึงกรณี / ความต้องการใช้งานใด ๆ


ในกรณีที่git cloneลำดับคำสั่งง่าย ๆ ฉันใช้คำตอบนี้ใช้งานได้!
Peter Krauss

16

สิ่งหนึ่งที่ฉันไม่ชอบเกี่ยวกับการจ่ายเงินเบาบางคือถ้าคุณต้องการเช็คเอาต์ไดเรกทอรีย่อยที่มีบางไดเรกทอรีที่ลึกโครงสร้างไดเรกทอรีของคุณจะต้องมีไดเรกทอรีทั้งหมดที่นำไปสู่

ฉันจะหลีกเลี่ยงสิ่งนี้ได้อย่างไรเพื่อโคลน repo ในสถานที่ที่ไม่ใช่พื้นที่ทำงานของฉันแล้วสร้างลิงค์สัญลักษณ์ในไดเรกทอรีพื้นที่ทำงานของฉันไปยังไดเรกทอรีย่อยในพื้นที่เก็บข้อมูล Git ทำงานได้ค่อนข้างดีเพราะสิ่งต่าง ๆ เช่นสถานะ git จะแสดงไฟล์การเปลี่ยนแปลงที่สัมพันธ์กับไดเรกทอรีการทำงานปัจจุบันของคุณ


ใช้งานได้ในระบบปฏิบัติการที่รองรับการเชื่อมโยงสัญลักษณ์ พวกเขาจำเป็นต้องเปลี่ยนวิธีการชำระเงินเบาบาง
Anders Lindén

1
+1 สำหรับแนวคิดด้วยลิงก์สัญลักษณ์ในไดเรกทอรีที่เช็คเอาต์ อย่างไรก็ตามการชำระเงินแบบกระจัดกระจายและลิงก์สัญลักษณ์ไม่ได้เกิดขึ้นพร้อมกัน: คุณไม่จำเป็นต้องมีการโคลนแบบเต็ม
apitsch

10

ที่จริงแล้วการชำระเงิน "แคบ" หรือ "บางส่วน" หรือ "กระจัดกระจาย" อยู่ภายใต้การพัฒนาในปัจจุบันซึ่งเป็นเรื่องหนักสำหรับ Git .gitหมายเหตุคุณจะยังคงมีพื้นที่เก็บข้อมูลภายใต้เต็มรูปแบบ ดังนั้นการโพสต์อีกสองโพสต์จึงเป็นสถานะปัจจุบันของ Git แต่ดูเหมือนว่าเราจะสามารถทำการตรวจสอบแบบเบาบางได้ในที่สุด ชำระเงินรายชื่อผู้รับจดหมายหากคุณสนใจรายละเอียดเพิ่มเติม - พวกเขากำลังเปลี่ยนแปลงอย่างรวดเร็ว


ดีแล้วที่รู้! ฉันชอบมีไดเรกทอรีที่เกี่ยวข้องอย่างใกล้ชิดภายใต้ที่เก็บเดียวและจะทำมันถ้าเป็นไปได้ทั้งหมด
Annika Backstrom

5

git clone --filter จาก Git 2.19

ตัวเลือกนี้จะข้ามการดึงข้อมูลวัตถุที่ไม่ต้องการจากเซิร์ฟเวอร์:

git clone --depth 1 --no-checkout --filter=blob:none \
  "file://$(pwd)/server_repo" local_repo
cd local_repo
git checkout master -- mdir/

เซิร์ฟเวอร์ควรกำหนดค่าด้วย:

git config --local uploadpack.allowfilter 1
git config --local uploadpack.allowanysha1inwant 1

ไม่มีการสนับสนุนเซิร์ฟเวอร์ตั้งแต่ v2.19.0 แต่สามารถทดสอบในเครื่องได้แล้ว

file://$(path)จำเป็นต้องเอาชนะgit cloneshenanigans ของโปรโตคอล: วิธีการโคลนที่เก็บ git ในพื้นที่ที่มีเส้นทางสัมพัทธ์อย่างไร

จำไว้ว่า--depth 1มีนัยแล้ว--single-branchโปรดดู: ฉันจะโคลนกิ่งเดียวใน Git ได้อย่างไร

สิ่งที่ต้องทำ: --filter=blob:noneข้าม blobs ทั้งหมด แต่ยังคงดึงวัตถุต้นไม้ทั้งหมด แต่ใน repo ปกติมันควรจะเล็กเมื่อเทียบกับไฟล์ของตัวเองดังนั้นมันก็ดีพออยู่แล้ว ถามได้ที่: https://www.spinics.net/lists/git/msg342006.html Devs ตอบ a --filter=tree:0อยู่ในผลงานที่ต้องทำ

รูปแบบของการจัดทำเอกสารเกี่ยวกับ--filterman git-rev-list

มีการขยายส่วนขยายไปยังโปรโตคอลระยะไกล Git เพื่อสนับสนุนคุณสมบัตินี้

เอกสารบนต้นไม้ Git:

ทดสอบมันออกมา

#!/usr/bin/env bash
set -eu

list-objects() (
  git rev-list --all --objects
  echo "master commit SHA: $(git log -1 --format="%H")"
  echo "mybranch commit SHA: $(git log -1 --format="%H")"
  git ls-tree master
  git ls-tree mybranch | grep mybranch
  git ls-tree master~ | grep root
)

# Reproducibility.
export GIT_COMMITTER_NAME='a'
export GIT_COMMITTER_EMAIL='a'
export GIT_AUTHOR_NAME='a'
export GIT_AUTHOR_EMAIL='a'
export GIT_COMMITTER_DATE='2000-01-01T00:00:00+0000'
export GIT_AUTHOR_DATE='2000-01-01T00:00:00+0000'

rm -rf server_repo local_repo
mkdir server_repo
cd server_repo

# Create repo.
git init --quiet
git config --local uploadpack.allowfilter 1
git config --local uploadpack.allowanysha1inwant 1

# First commit.
# Directories present in all branches.
mkdir d1 d2
printf 'd1/a' > ./d1/a
printf 'd1/b' > ./d1/b
printf 'd2/a' > ./d2/a
printf 'd2/b' > ./d2/b
# Present only in root.
mkdir 'root'
printf 'root' > ./root/root
git add .
git commit -m 'root' --quiet

# Second commit only on master.
git rm --quiet -r ./root
mkdir 'master'
printf 'master' > ./master/master
git add .
git commit -m 'master commit' --quiet

# Second commit only on mybranch.
git checkout -b mybranch --quiet master~
git rm --quiet -r ./root
mkdir 'mybranch'
printf 'mybranch' > ./mybranch/mybranch
git add .
git commit -m 'mybranch commit' --quiet

echo "# List and identify all objects"
list-objects
echo

# Restore master.
git checkout --quiet master
cd ..

# Clone. Don't checkout for now, only .git/ dir.
git clone --depth 1 --quiet --no-checkout --filter=blob:none "file://$(pwd)/server_repo" local_repo
cd local_repo

# List missing objects from master.
echo "# Missing objects after --no-checkout"
git rev-list --all --quiet --objects --missing=print
echo

echo "# Git checkout fails without internet"
mv ../server_repo ../server_repo.off
! git checkout master
echo

echo "# Git checkout fetches the missing directory from internet"
mv ../server_repo.off ../server_repo
git checkout master -- d1/
echo

echo "# Missing objects after checking out d1"
git rev-list --all --quiet --objects --missing=print

GitHub ต้นน้ำ

เอาต์พุตใน Git v2.19:

# List and identify all objects
c6fcdfaf2b1462f809aecdad83a186eeec00f9c1
fc5e97944480982cfc180a6d6634699921ee63ec
7251a83be9a03161acde7b71a8fda9be19f47128
62d67bce3c672fe2b9065f372726a11e57bade7e
b64bf435a3e54c5208a1b70b7bcb0fc627463a75 d1
308150e8fddde043f3dbbb8573abb6af1df96e63 d1/a
f70a17f51b7b30fec48a32e4f19ac15e261fd1a4 d1/b
84de03c312dc741d0f2a66df7b2f168d823e122a d2
0975df9b39e23c15f63db194df7f45c76528bccb d2/a
41484c13520fcbb6e7243a26fdb1fc9405c08520 d2/b
7d5230379e4652f1b1da7ed1e78e0b8253e03ba3 master
8b25206ff90e9432f6f1a8600f87a7bd695a24af master/master
ef29f15c9a7c5417944cc09711b6a9ee51b01d89
19f7a4ca4a038aff89d803f017f76d2b66063043 mybranch
1b671b190e293aa091239b8b5e8c149411d00523 mybranch/mybranch
c3760bb1a0ece87cdbaf9a563c77a45e30a4e30e
a0234da53ec608b54813b4271fbf00ba5318b99f root
93ca1422a8da0a9effc465eccbcb17e23015542d root/root
master commit SHA: fc5e97944480982cfc180a6d6634699921ee63ec
mybranch commit SHA: fc5e97944480982cfc180a6d6634699921ee63ec
040000 tree b64bf435a3e54c5208a1b70b7bcb0fc627463a75    d1
040000 tree 84de03c312dc741d0f2a66df7b2f168d823e122a    d2
040000 tree 7d5230379e4652f1b1da7ed1e78e0b8253e03ba3    master
040000 tree 19f7a4ca4a038aff89d803f017f76d2b66063043    mybranch
040000 tree a0234da53ec608b54813b4271fbf00ba5318b99f    root

# Missing objects after --no-checkout
?f70a17f51b7b30fec48a32e4f19ac15e261fd1a4
?8b25206ff90e9432f6f1a8600f87a7bd695a24af
?41484c13520fcbb6e7243a26fdb1fc9405c08520
?0975df9b39e23c15f63db194df7f45c76528bccb
?308150e8fddde043f3dbbb8573abb6af1df96e63

# Git checkout fails without internet
fatal: '/home/ciro/bak/git/test-git-web-interface/other-test-repos/partial-clone.tmp/server_repo' does not appear to be a git repository
fatal: Could not read from remote repository.

Please make sure you have the correct access rights
and the repository exists.

# Git checkout fetches the missing directory from internet
remote: Enumerating objects: 1, done.
remote: Counting objects: 100% (1/1), done.
remote: Total 1 (delta 0), reused 0 (delta 0)
Receiving objects: 100% (1/1), 45 bytes | 45.00 KiB/s, done.
remote: Enumerating objects: 1, done.
remote: Counting objects: 100% (1/1), done.
remote: Total 1 (delta 0), reused 0 (delta 0)
Receiving objects: 100% (1/1), 45 bytes | 45.00 KiB/s, done.

# Missing objects after checking out d1
?8b25206ff90e9432f6f1a8600f87a7bd695a24af
?41484c13520fcbb6e7243a26fdb1fc9405c08520
?0975df9b39e23c15f63db194df7f45c76528bccb

สรุป: blobs ทั้งหมดจากด้านนอกd1/หายไป

โปรดทราบว่าroot/rootและmybranch/mybranchจะหายไป แต่--depth 1ซ่อนไว้จากรายการไฟล์ที่หายไป หากคุณลบไฟล์--depth 1เหล่านั้นจะปรากฏในรายการไฟล์ที่หายไป


1

เมื่อคุณแก้ไขจุดคุณสามารถใช้สองสาขาแยกเพื่อจัดเก็บสองไดเรกทอรีแยก สิ่งนี้จะเก็บทั้งคู่ไว้ในที่เก็บเดียวกัน แต่คุณยังคงไม่สามารถคอมมิชชันแผนผังไดเรคทอรีทั้งสองได้ หากคุณมีการเปลี่ยนแปลงอย่างใดอย่างหนึ่งที่ต้องมีการเปลี่ยนแปลงในอีกคุณจะต้องดำเนินการเหล่านั้นเป็นสองคอมมิชชันที่แยกจากกันและคุณเปิดโอกาสที่คู่ของเช็คเอาต์ของทั้งสองไดเรกทอรีสามารถซิงค์

หากคุณต้องการรักษาคู่ของไดเรกทอรีเป็นหนึ่งหน่วยคุณสามารถใช้ 'wordpress / wp-content' เป็นรากของ repo ของคุณและใช้ไฟล์. gitignore ที่ระดับบนสุดเพื่อละเว้นทุกสิ่งยกเว้นสองไดเรกทอรีย่อยที่น่าสนใจ นี่อาจเป็นทางออกที่สมเหตุสมผลที่สุดในตอนนี้

มีการกล่าวหาว่าการชำระเงินน้อยลงเป็นเวลาสองปีแล้ว แต่ก็ยังไม่มีวี่แววของพวกเขาในการพัฒนาระบบคอมไพล์และไม่มีสิ่งบ่งชี้ว่าการเปลี่ยนแปลงที่จำเป็นจะมาถึงที่นั่น ฉันไม่เชื่อใจพวกเขา


1

คุณไม่สามารถชำระเงินไดเรกทอรีเดียวของที่เก็บได้เนื่องจากพื้นที่เก็บข้อมูลทั้งหมดได้รับการจัดการโดยโฟลเดอร์. git เดียวในรูทของโครงการแทนไดเรกทอรีย่อยมากมายของ. svn

ปัญหาเกี่ยวกับการทำงานกับปลั๊กอินในที่เก็บเดียวคือการทำให้คอมมิชชันเช่นmythemeจะเพิ่มจำนวนการแก้ไขสำหรับmypluginดังนั้นแม้ในการโค่นล้มมันจะดีกว่าที่จะใช้ที่เก็บแยกต่างหาก

กระบวนทัศน์การโค่นล้มสำหรับโครงการย่อยคือsvn: externalsซึ่งแปลว่าค่อนข้างเป็นsubmodulesใน git (แต่ไม่ใช่ในกรณีที่คุณเคยใช้ svn: externals มาก่อน)


0

มีแรงบันดาลใจอยู่ที่นี่ เพียงแค่ใช้หรือshell regexgit regex

git checkout commit_id */*.bat  # *.bat in 1-depth subdir exclude current dir, shell regex  
git checkout commit_id '*.bat'  # *.bat in all subdir include current dir, git regex

ใช้ใบเสนอราคาเพื่อหลีกเลี่ยงการตีความเชลล์ regex และส่งสัญลักษณ์แทนไปยังคอมไพล์

ไฟล์แรกไม่สามารถเรียกซ้ำได้เฉพาะไฟล์ที่มีความลึก 1 ระดับsubdirเท่านั้น แต่อันที่สองกลับมาซ้ำอีก

สำหรับสถานการณ์ของคุณสิ่งต่อไปนี้อาจเพียงพอ

git checkout master */*/wp-content/*/*
git checkout master '*/wp-content/*'

เพียงแฮ็กบรรทัดตามต้องการ


0

คุณสามารถย้อนกลับการเปลี่ยนแปลงที่ไม่ได้รับอนุญาตได้เฉพาะกับไฟล์หรือไดเรกทอรีเฉพาะ:

git checkout [some_dir|file.txt]
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.