mirror of
https://github.com/trezor/trezor-firmware.git
synced 2024-11-22 07:28:10 +00:00
monorepo: add notes and creation tool
This commit is contained in:
parent
fd2829a27b
commit
04bde880c6
76
README-monorepo.md
Normal file
76
README-monorepo.md
Normal file
@ -0,0 +1,76 @@
|
|||||||
|
Monorepo notes
|
||||||
|
==============
|
||||||
|
|
||||||
|
|
||||||
|
Generating
|
||||||
|
----------
|
||||||
|
|
||||||
|
Use the [create-monorepo] script to regenerate from current master(s).
|
||||||
|
|
||||||
|
[create-monorepo]: create-monorepo.py
|
||||||
|
|
||||||
|
|
||||||
|
Structure
|
||||||
|
---------
|
||||||
|
|
||||||
|
This is a result of Git merge of several unrelated histories, each of which is
|
||||||
|
moved to its own subdirectory during the merge.
|
||||||
|
|
||||||
|
That means that this is actually all the original repos at the same time. You can
|
||||||
|
check out any historical commit hash, or any historical tag.
|
||||||
|
|
||||||
|
All tags from the previous history still exist, and in addition, each has a version
|
||||||
|
named by its directory. I.e., for trezor-mcu tag `v1.6.3`, you can also check out
|
||||||
|
`legacy/v1.6.3`.
|
||||||
|
|
||||||
|
|
||||||
|
Merging pre-existing branches
|
||||||
|
-----------------------------
|
||||||
|
|
||||||
|
Because the repository shares all the histories, merging a branch or PR can be done
|
||||||
|
with a simple `git merge`. It's often necessary to add hints to git by specifying a merge strategy - especially when some commits add new files.
|
||||||
|
|
||||||
|
Use the following options: `-s subtree -X subtree=<destdir>`.
|
||||||
|
|
||||||
|
Example for your local checkout:
|
||||||
|
|
||||||
|
$ git remote add core-local ~/git/trezor-core
|
||||||
|
$ git fetch core-local
|
||||||
|
$ git merge core-local/wip -s subtree -X subtree=core
|
||||||
|
|
||||||
|
Same options should be used for `git rebase` of a pre-existing branch.
|
||||||
|
|
||||||
|
|
||||||
|
Sub-repositories
|
||||||
|
----------------
|
||||||
|
|
||||||
|
The monorepo has two subdirectories that can be exported to separate repos:
|
||||||
|
|
||||||
|
* **common** exports to https://github.com/trezor/trezor-common
|
||||||
|
* **crypto** exports to https://github.com/trezor/trezor-crypto
|
||||||
|
|
||||||
|
These exports are managed with [git-subrepo] tool. To export all commits that touch
|
||||||
|
one of these directories, run the following command:
|
||||||
|
|
||||||
|
$ git subrepo push <dirname>
|
||||||
|
|
||||||
|
You will need commit access to the respective GitHub repository.
|
||||||
|
|
||||||
|
For installation instructions and detailed usage info, refer to the [git-subrepo] README.
|
||||||
|
|
||||||
|
[git-subrepo]: https://github.com/ingydotnet/git-subrepo
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
Sketch of further details:
|
||||||
|
|
||||||
|
What git-subrepo does under the hood is create and fetch a remote for the export,
|
||||||
|
check out `parent` revision and replay all commits since `commit` using
|
||||||
|
something along the lines of `git filter-branch --subdirectory-filter`.
|
||||||
|
|
||||||
|
So basically a nicely tuned git-subtree.
|
||||||
|
|
||||||
|
This can all be done manually if need be (or if you need more advanced usecases like
|
||||||
|
importing changes from the repo commit-by-commit, because git-subrepo will squash
|
||||||
|
on import). See [this nice article](https://medium.com/@porteneuve/mastering-git-subtrees-943d29a798ec)
|
||||||
|
for hints.
|
137
create_monorepo.py
Normal file
137
create_monorepo.py
Normal file
@ -0,0 +1,137 @@
|
|||||||
|
#!/usr/bin/python3
|
||||||
|
import glob
|
||||||
|
import os
|
||||||
|
import subprocess
|
||||||
|
|
||||||
|
TREZOR_REPO = "https://github.com/trezor"
|
||||||
|
|
||||||
|
NAME="monorepo"
|
||||||
|
MAIN_REPO = "trezor-core"
|
||||||
|
SUBREPOS = {
|
||||||
|
"trezor-common": "common",
|
||||||
|
"trezor-crypto": "crypto",
|
||||||
|
"trezor-mcu": "legacy",
|
||||||
|
"trezor-storage": "storage",
|
||||||
|
"python-trezor": "python",
|
||||||
|
}
|
||||||
|
PUBLISHED_SUBREPOS = ["trezor-common", "trezor-crypto"]
|
||||||
|
|
||||||
|
KEEP_TAGS = ["trezor-core", "trezor-mcu", "python-trezor"]
|
||||||
|
|
||||||
|
GITSUBREPO_TEMPLATE = """\
|
||||||
|
; DO NOT EDIT (unless you know what you are doing)
|
||||||
|
;
|
||||||
|
; This subdirectory is a git "subrepo", and this file is maintained by the
|
||||||
|
; git-subrepo command. See https://github.com/git-commands/git-subrepo#readme
|
||||||
|
;
|
||||||
|
[subrepo]
|
||||||
|
remote = git+ssh://git@github.com/trezor/{remote}
|
||||||
|
branch = master
|
||||||
|
commit = {remote_head}
|
||||||
|
parent = {current_head}
|
||||||
|
method = rebase
|
||||||
|
cmdver = 0.4.0
|
||||||
|
"""
|
||||||
|
|
||||||
|
|
||||||
|
def lines(s):
|
||||||
|
yield from s.strip().split("\n")
|
||||||
|
|
||||||
|
|
||||||
|
def git(args):
|
||||||
|
print("+ git:", args)
|
||||||
|
return subprocess.check_output("git " + args, universal_newlines=True, shell=True)
|
||||||
|
|
||||||
|
|
||||||
|
def move_to_subtree(remote, dst):
|
||||||
|
os.makedirs(dst, exist_ok=True)
|
||||||
|
for fn in lines(git(f"ls-tree --name-only remotes/{remote}/master")):
|
||||||
|
if fn == ".gitmodules":
|
||||||
|
continue
|
||||||
|
git(f"mv {fn} {dst}/{fn}")
|
||||||
|
|
||||||
|
|
||||||
|
def rewrite_gitmodules(remote, dst):
|
||||||
|
master_gitmodules = git("show master:.gitmodules")
|
||||||
|
try:
|
||||||
|
gitmodules = git(f"show {remote}/master:.gitmodules")
|
||||||
|
except:
|
||||||
|
# no gitmodules
|
||||||
|
return
|
||||||
|
gitmodules = gitmodules.replace('submodule "', f'submodule "{dst}/')
|
||||||
|
with open(".gitmodules", "w") as f:
|
||||||
|
f.write(master_gitmodules + gitmodules)
|
||||||
|
git("add .gitmodules")
|
||||||
|
|
||||||
|
|
||||||
|
def merge_remote(remote, dst):
|
||||||
|
git(f"remote add {remote} {TREZOR_REPO}/{remote}")
|
||||||
|
git(f"fetch {remote}")
|
||||||
|
try:
|
||||||
|
git(f"merge --no-commit --allow-unrelated-histories {remote}/master")
|
||||||
|
except:
|
||||||
|
# this might fail because of .gitmodules conflict
|
||||||
|
pass
|
||||||
|
|
||||||
|
rewrite_gitmodules(remote, dst)
|
||||||
|
move_to_subtree(remote, dst)
|
||||||
|
|
||||||
|
|
||||||
|
def retag_remote(remote, dst):
|
||||||
|
for tagline in lines(git(f"ls-remote -t {remote}")):
|
||||||
|
commit, tagpath = tagline.split()
|
||||||
|
tagname = os.path.basename(tagpath)
|
||||||
|
git(f"tag {dst}/{tagname} {commit}")
|
||||||
|
git(f"tag -d {tagname}")
|
||||||
|
|
||||||
|
|
||||||
|
def generate_subrepo_file(remote):
|
||||||
|
current_head = git("rev-parse master").strip()
|
||||||
|
remote_head = git(f"rev-parse {remote}/master").strip()
|
||||||
|
dst = SUBREPOS[remote]
|
||||||
|
with open(f"{dst}/.gitrepo", "w") as f:
|
||||||
|
f.write(GITSUBREPO_TEMPLATE.format(remote=remote, current_head=current_head, remote_head=remote_head))
|
||||||
|
git(f"add {dst}/.gitrepo")
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
git(f"clone {TREZOR_REPO}/{MAIN_REPO} {NAME}")
|
||||||
|
os.chdir(NAME)
|
||||||
|
move_to_subtree("origin", "core")
|
||||||
|
git(f"commit -m 'MONOREPO CREATE FROM {MAIN_REPO}'")
|
||||||
|
retag_remote("origin", "core")
|
||||||
|
|
||||||
|
for remote, dst in SUBREPOS.items():
|
||||||
|
merge_remote(remote, dst)
|
||||||
|
|
||||||
|
if remote in PUBLISHED_SUBREPOS:
|
||||||
|
with open(f"{dst}/.gitmodules", "w") as f:
|
||||||
|
f.write(git(f"show {remote}/master:.gitmodules"))
|
||||||
|
git(f"add {dst}/.gitmodules")
|
||||||
|
|
||||||
|
git(f"commit -m 'MONOREPO MERGE {remote}'")
|
||||||
|
|
||||||
|
try:
|
||||||
|
retag_remote(remote, dst)
|
||||||
|
except:
|
||||||
|
pass
|
||||||
|
|
||||||
|
for submodule in glob.glob("*/vendor/*"):
|
||||||
|
modname = os.path.basename(submodule)
|
||||||
|
if modname not in SUBREPOS:
|
||||||
|
continue
|
||||||
|
|
||||||
|
git(f"rm {submodule}")
|
||||||
|
symlink_target = f"../../{SUBREPOS[modname]}"
|
||||||
|
os.symlink(symlink_target, submodule)
|
||||||
|
git(f"add {submodule}")
|
||||||
|
|
||||||
|
git(f"commit -m 'MONOREPO RELINK SUBMODULES'")
|
||||||
|
|
||||||
|
for remote in PUBLISHED_SUBREPOS:
|
||||||
|
generate_subrepo_file(remote)
|
||||||
|
git(f"commit -m 'MONOREPO SUBREPO FILES'")
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
main()
|
Loading…
Reference in New Issue
Block a user