An alternate git workflow tool
Project description
gitty - a simple alternative to git flow with less rigidity
This project provides an easy way to deal with git branches in an environment with multiple parallel development streams.
View download stats here: https://pypistats.org/packages/gitty
It is worth noting that (like git flow), this is a tool to simplify the implementation of a workflow - you can do everything manually if you want to, so don't get too hung up on the tooling.
Some sample work flows...
To get started, we'll make a new empty project using node.
But first, some assumptions!
- the
master
branch is the cutting edge of new development (we will see how to change that later) - release stabilization branches are created for each major and minor release
- semantic versioning will be used to track where we are
- projects are using a supported project type:
- maven - solid support
- python - I use it for this project, so it is probably adequate
- nodejs - experimental - try it and let me know
- gradle - expreimental - I am using it, but still with dev builds
Next, we'll walk through some common scenarios and see how to use this thing.
% mkdir test-project
% cd test-project
% npm init --yes
% git init
% git add package.json
% git commit -m "initial commit"
Now, we have a blank project with a package.json file in it and nothing else.
current state of affairs (and some help)
To see some useful info about your project, you can just run the gitty command.
I'm assuming that you installed gitty using pip from https://pypi.org/project/gitty/
-> % gitty
$ git rev-parse --abbrev-ref HEAD
current branch: master
project type: node
current version: 1.0.0
command name: head
for help, try "gitty help"
This is telling you what gitty thinks the state of your project is, if you want more information, try gitty help
.
new release stabilization branch
Let's say that we're ready to snap a 1.0.0 release of our project. From the output above, you can see the "release"
command and it can be run using either gitty release
or just gitty r
:
-> % gitty r
$ git rev-parse --abbrev-ref HEAD
current branch: master
project type: node
current version: 1.0.0
command name: r
$ git checkout -b 1.0/master
Switched to a new branch '1.0/master'
$ git checkout -b 1.0/releases
Switched to a new branch '1.0/releases'
bump version to 1.0.0
$ git add package.json
$ git commit -m "bumped version to 1.0.0"
b'On branch 1.0/releases\nnothing to commit, working tree clean\n'
$ git tag 1.0.0
$ git checkout 1.0/master
Switched to branch '1.0/master'
$ git merge --strategy=ours 1.0/releases
bump version to 1.0.1
$ git add package.json
$ git commit -m "bumped version to 1.0.1"
$ git checkout master
Switched to branch 'master'
$ git merge --strategy=ours 1.0/master
bump version to 1.1.0
$ git add package.json
$ git commit -m "bumped version to 1.1.0"
Wow, that's a lot of crap.
That's showing you what's happening when you create a release - why all that stuff? I'll fill in more details later, but for now: Because.
At the end of that, you'll be back on master
and there will be 2 new branches:
-> % git branch
1.0/master
1.0/releases
* master
Those branches have purpose.
The 1.0/master
branch is used to stabilize the 1.0 release; The 1.0/releases
branch is used to give us a stable
point of reference in the future for things like hot fixes, etc.
There's also a new tag:
-> % git tag
1.0.0
That is on the releases branch, and specifically, on the commit where the release candidate should be built from.
new task for forward development
This is a common thing - so it's pretty simple:
-> % gitty t some_new_thing_here
$ git rev-parse --abbrev-ref HEAD
current branch: master
project type: node
current version: 1.1.0
command name: t
$ git checkout -b tasks/some_new_thing_here
Switched to a new branch 'tasks/some_new_thing_here'
Chuck some code here and commit it and push it - eventually, you'll merge it back to master
and it'll be real code. Yay.
new task for release stabilization branch
Yeah, this doesn't sound like a great idea, but sometimes, it's needed.
-> % gco 1.0/master
Switched to branch '1.0/master'
-> % gitty t some_new_task_for_10
$ git rev-parse --abbrev-ref HEAD
current branch: 1.0/master
project type: node
current version: 1.0.1
command name: t
$ git checkout -b 1.0/tasks/some_new_task_for_10
Switched to a new branch '1.0/tasks/some_new_task_for_10'
Now, you've created a branch from the 1.0/master
branch named 1.0/tasks/some_new_task_for_10
- the naming indicates
where it will end up (although eventually, you'll most likely want to merge it to master
too, for now, we'll just work on 1.0).
new release candidate for 1.0
To make a new RC for the stabilization branch, do this:
% gco 1.0/master
-> % gitty r
$ git rev-parse --abbrev-ref HEAD
current branch: 1.0/master
project type: node
current version: 1.0.1
command name: r
$ git checkout 1.0/releases
Switched to branch '1.0/releases'
$ git merge 1.0/master
bump version to 1.0.1
$ git add package.json
$ git commit -m "bumped version to 1.0.1"
b'On branch 1.0/releases\nnothing to commit, working tree clean\n'
$ git tag 1.0.1
$ git checkout 1.0/master
Switched to branch '1.0/master'
$ git merge 1.0/releases
bump version to 1.0.2
$ git add package.json
$ git commit -m "bumped version to 1.0.2"
Again, that's a lot of output...but it's what's happening.
Now, the 1.0/master
branch got merged to 1.0/release
, we have a new tag for the release candidate (1.0.1
), and
the 1.0/master
project file reflects that we're now working toward 1.0.2
for the next release candidate.
NOTE: Because we merge back to
1.0/master
, we have a single place to look for changes to be merged to version 1.1 (onmaster
).
wrapping up a release to prepare for the next one
Once release 1.0 has been released and we're ready to start on the road to 1.1 (and 1.2), we'll want to get all of the 1.0 changes merged back to master
(where 1.1 lives). There's not much to do there, so we just use git:
% git merge 1.0/master
Auto-merging package.json
CONFLICT (content): Merge conflict in package.json
Automatic merge failed; fix conflicts and then commit the result.
% (do what you need to do to merge the conflicts here)
% git add package.json
% git commit
Creating the next release is exactly the same as before:
% gcm
% gitty r
You'll see a pile of output again, and at the end, there will be 1.1/master
and 1.1/release
branches and a new 1.1.0
tag for the release candidate.
Versioning for python projects (i.e., eating my own dog food)
I'm using this tool to manage the git repository for this tool, so its versioning approach is the one that it supports.
The versioning scheme is based on semver 2.0.0, and will use a suffix of "dev0" to indicate that a version is not a release (and partly because of how setuptools behaves).
For this project, non-release builds will not be deployed to pypi - only actual releases will be deployed.
This versioning will start as of version 1.2.0.dev0 (currently master).
plugins
You can now make plugins for gitty - there are samples in the oh so cleverly named "plugin_samples" directory.
I'll be documenting them more in the future, unless I die or win the lottery or get busy with other things.
EXPERIMENTAL: tab completion plugin (zsh)
I'm working on making this easier, but for now, to enable tab completion:
- install gitty 1.5 or later
- download https://raw.githubusercontent.com/lmeadors/gitty/master/plugin_samples/zsh_complete.py to ~/.gitty/
- create a symlink to
/usr/local/bin/gitty_completion.zsh
in your zsh custom directory
This is a new feature starting with the 1.4 release. It seems to be working for me, but your mileage may vary - if it's broken, let me know, and I'll try to make it better.
UPDATE: I have been using this for a few months - it seems to work fine.
EXPERIMENTAL: How can I use different branch names? I do not have a "master" branch...
I have a few projects like this, too -
This should work, but I may have missed a few spots.
Create a file named .gittyrc
in your project directory (or your home directory if you want this to be a global
default) - you can override some settings with it. For example, if you just want to call your master
branch main
instead:
trunk=main
Your working (trunk) branch(es) would be named "main"; tasks would be named "tasks/...", and releases would live on "release" branches.
If you really miss git flow, you can even do this:
trunk=develop
tasks=features
release=master
This will give you a hybrid of gitflow and gitty:
- your "working branches" will be either
develop
orx.y/develop
- releases will live on
x.y/master
- task branches will live on either
features/...
orx.y/features/...
Because the gitty process has no concept of a "hotfix" (every release can be a hotfix; every version can be patched), there is no need for those extra branch names.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
File details
Details for the file gitty-1.8.0.tar.gz
.
File metadata
- Download URL: gitty-1.8.0.tar.gz
- Upload date:
- Size: 30.0 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/5.1.0 CPython/3.11.2
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | 0d0e25829b7da92c7aa825b1ee016def31ec6689f921a82399489ec6416ae19f |
|
MD5 | 7c444784db604a5c27a81aa66c22b0f5 |
|
BLAKE2b-256 | 62b6c3b81f3f787b9614c5fe1415519312047e32b1e1559c25e9ebee463e5282 |
File details
Details for the file gitty-1.8.0-py3-none-any.whl
.
File metadata
- Download URL: gitty-1.8.0-py3-none-any.whl
- Upload date:
- Size: 30.2 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/5.1.0 CPython/3.11.2
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | e29a04576810d19041b559252346e6c889ebf630ee7193da4c60466c68bc53d7 |
|
MD5 | 86dd2e53d5eb636a21b1a448497e1ee8 |
|
BLAKE2b-256 | 0f02ba745ed4eb38cca8c4c8ad86a1104f2f4757a4567c839a30e13a47d8341b |