Automatically push an updated file whenever it is changed
Linux
- Make sure
inotify-tools
is installed (https://github.com/rvoicilas/inotify-tools) - Configure git as usual
- Clone the git repository of interest from github and, if necessary, add file you want to monitor
- Allow username/password to be cached so you aren’t asked everytime
git config credential.helper store
- Open a terminal and navigate, as necessary; issue the following command
# <<branch>> = branch you are pushing to
# <<file>> = file you want to monitor
inotifywait -q -m -e CLOSE_WRITE --format="git commit -m 'auto commit' %w && git push origin <<branch>>" <<file>> | bash
- In a separate shell, do whatever you want and when monitored file is updated, it will automatically get committed and pushed (as long as the shell with the
inotifywait
command is still active)
Mac
- Make sure
fswatch
is installed (https://github.com/emcrisostomo/fswatch) - Configure git as usual
- Clone the git repository of interest from github and, if necessary, add file you want to monitor
- Allow username/password to be cached so you aren’t asked everytime
git config credential.helper store
- Create a script that performs the commit and push (auto_commit_push.sh)
#!/bin/bash
# <<branch>> = branch you are pushing to
git commit -m "auto commit" $1
git push origin <<branch>>
- Open a terminal and navigate, as necessary; issue the following command
# <<file>> = file you want to monitor
# <<path/to/auto_commit_push.sh>> = path to the script created above
fswatch -0 <<file>> | xargs -0 -n 1 bash <<path/to/auto_commit_push.sh>>
- In a separate shell, do whatever you want and when monitored file is updated, it will automatically get committed and pushed (as long as the shell with the
fswatch
command is still active)
https://gist.github.com/darencard/5d42319abcb6ec32bebf6a00ecf99e86
https://github.com/gitwatch/gitwatch
gitwatch
A bash script to watch a file or folder and commit changes to a git repo
What to use it for?
That’s really up to you, but here are some examples:
- config files: some programs auto-write their config files, without waiting for you to click an ‘Apply’ button; or even if there is such a button, most programs offer you no way of going back to an earlier version of your settings. If you commit your config file(s) to a git repo, you can track changes and go back to older versions. This script makes it convenient, to have all changes recorded automatically.
- document files: if you use an editor that does not have built-in git support (or maybe if you don’t like the git support it has), you can use gitwatch to automatically commit your files when you save them, or combine it with the editor’s auto-save feature to fully automatically and regularly track your changes
- more stuff! If you have any other uses, or can think of ones, please let us know, and we can add them to this list!
Installation
gitwatch
can be installed in various ways.
From Source
gitwatch
can be installed from source by simply cloning the repository and putting the shell script into your $PATH
. The commands below will do that for you if /usr/local/bin
is in your $PATH
. You may need to invoke install
with sudo
.
$ git clone https://github.com/gitwatch/gitwatch.git
$ cd gitwatch
$ [sudo] install -b gitwatch.sh /usr/local/bin/gitwatch
Update
If you installed gitwatch
from source, you can update it by following the exact same steps (or git pull
rather than clone if you kept the repository around).
bpkg
gitwatch
can be installed with bpkg. Make sure you have bpkg installed before running the command below. You may need to invoke bpkg
with sudo
when using the -g
flag.
$ [sudo] bpkg install -g gitwatch/gitwatch
Requirements
To run this script, you must have installed and globally available:
git
( git/git | http://www.git-scm.com )inotifywait
(part of inotify-tools: rvoicilas/inotify-tools )
Notes for Mac
If running on OS X, you’ll need to install the following Homebrew tools:
$ brew install fswatch
$ brew install coreutils
What it does
When you start the script, it prepares some variables and checks if the file [a] or directory [b] given as input really exists.
Then it goes into the main loop (which will run forever, until the script is forcefully stopped/killed), which will:
- watch for changes to the file/directory using
inotifywait
(inotifywait
will block until something happens) - wait 2 seconds
cd
into the directory [b] / the directory containing the file [a] (becausegit
likes to operate locally)git add <file>
[a] /git add .
[b]git commit -m "Scripted auto-commit on change (<date>)"
[a] /git commit -a -m"Scripted auto-commit on change (<date>)"
[b]- if a remote is defined (with
-r
) do a push after the commit (a specific branch can be selected with-b
)
Notes:
- the waiting period of 2 sec is added to allow for several changes to be written out completely before committing; depending on how fast the script is executed, this might otherwise cause race conditions when watching a folder
- currently, folders are always watched recursively
Usage
gitwatch.sh [-r <remote> [-b <branch>]] <file or directory to watch>
It is expected that the watched file/directory are already in a git repository (the script will not create a repository). If a folder is being watched, this will be watched fully recursively; this also means that all files and sub-folders added and removed from the directory will always be added and removed in the next commit. The .git
folder will be excluded from the inotifywait
call so changes to it will not cause unnecessary triggering of the script.
Starting on Boot
If you want to have the script auto-started upon boot, the method to do this depends on your operating system and distribution. If you have a GUI dialog to set up startup launches, you might want to use that, so you can more easily find and change the startup script calls later on.
Please also note that if either of the paths involved (script or target) contains spaces or special characters, you need to escape them accordingly; if you don’t know how to do that, the internet will help you, or feel free to ask here or contact me directly.
SysVInit
A central place to put startup scripts on Linux is generally /etc/rc.local
(to my knowledge; only tested and confirmed on Ubuntu). This file, if it has the +x bit, will be executed upon startup, by the root user account. If you want to start gitwatch
from rc.local
, the recommended way to call it is:
su -c "/absolute/path/to/script/gitwatch.sh /absolute/path/to/watched/file/or/folder" -l <username> &
The <username>
bit should be replaced with your username or that of any other (non-root) user account; it only needs write-access to the git repository of the file/folder you want to watch. The ampersand (&
) at the end sends the launched process into the background (this is important if you have other calls in rc.local
after the mentioned line, because the gitwatch
call does not usually return).
systemd
- If installed to a path other than
/usr/bin/gitwatch
, modify[email protected]
to suit - Create dir if it does not exist and copy systemd service file with
mkdir -p "$HOME/.config/systemd/user" && cp [email protected] $HOME/.config/systemd/user
- Start and enable the service for a given path by running
systemctl --user --now enable gitwatch@$(systemd-escape "'-r url/to/repository' /path/to/folder").service