/smog

manage local git repositories in style of package manager

Primary LanguageShellOtherNOASSERTION

smog

smog (source manager over git) is a source-code manager with some features of a package manager.

Why smog? I download and build software from git-repositories quite often and it is common that I modify the source-code. I needed a tool to solve the following:

  • store each repo in a unique path on my computer (à la go get)
  • easily check for remote updates
  • easily update while maintaining my local changes

Meanwhile, I dislike sudo make install, especially for packages that're not stable or I don't fully trust. So, additionally, I wanted the tool to:

  • create symlinks to binaries and add to my PATH
  • create symlinks to ELF libraries and enable ldconfig to cache them

and abviously:

  • easily cleanup and remove packages.

smog solves these issues by creating plain-text metadata files for each repository you clone, and by employing git + basic bash commands for maintaining them.

smog is configured by a few bash variables and it manages each repository (smog package) in either tag or branch mode. If you run the bootstrapper to install smog, it will manage itself as a "smog package", meaning that you upgrade smog with the command smog update smog.

Features

smog is best described by its main commands:

  • smog get URL - create package from a git repository
  • smog show PKG - show metadata of a package
  • smog list - list all packages
  • smog sync - list all packages that have remote updates
  • smog update PKG - update local repository with remote changes
  • smog link PKG - create symlinks for executable files in repository

unstable and experimental commands:

  • smog get -r URL - add git repository with submodules
  • smog build PKG - build package (requires manually defined buildcmd)
  • smog upgrade PKG - update and upgrade package
  • smog search KEYWORD - search for repositories (currently only on github.com)

Installation

Requirements

  • git
  • bash (>= 4.3) and a few common bash programs:
    • test, cat, cut, ln, readlink, tr, etc.
    • grep, sed, file, xargs (for asynchronous syncing)

optional:

  • vim or any other text editor
  • wget for bootstrapping
  • curl for searching

Bootstrapping

This is the recommended way of installation. The bootstrapper will install and configure smog as a package of itself. It will integrate smog to your bash environment and add the goto command for navigating to directories of smog packages.

Just download the bootstrapper script and run with bash:

wget -nv https://raw.githubusercontent.com/vgratian/smog/master/bootstrap
bash bootstrap
rm bootstrap

If you don't want to be asked too many questions, use the silent argument:

bash bootstrap silent

You can also safely remove smog with the bootstrapper:

bash bootstrap undo

Manual Installation

Clone the latest tag in a location of your choice and copy the default config file to config:

git clone -b 0.0.1 https://github.com/vgratian/smog
cd smog
cp config-def config

Edit config: at the very least, you should make sure the following:

  • $SMOG matches the the directory where smog is cloned
  • the directories $PKG and $MDD exist and are empty
  • the directories $BIN and $LIB exist, or these variables are unset ("")

Additionally, you might want to:

  • add the autocompletion script to your .bashrc
  • update your $PATH to include the directory $BIN (if set)
  • update ldconfig to include the directory $LIB (if set)

Note that in this case, smog is not a package of itself, and you have to update it manually with git as well.

Usage

For a comprehensive list of commands and options, simply run smog help.

Examples

Add and link a package

Here is a typical example of using smog. I add dwm, my favorite window-manager:

smog get git://git.suckless.org/dwm

Typically I modify the source-code, then build the package. This is the part that I do myself:

# modify the source code
goto dwm
vim dwm.c
# tell smog how to build dwm
smog set dwm buildcmd='make all'
smog build

Finally, I want to add the binary dwm to my PATH:

smog link dwm

(If smog detects more binaries in the repostory, it will ask if you want all of them to be linked).

Later on, I can upgrade dwm with:

test -n "$(smog sync dwm)" && smog upgrade dwm -f

Query packages

To get a plain list of packages, run smog list. For a more detailed list, run smog list -m. To show details of a package, run smog show PKG or smog show PKG -m.

Similarly, for a list of updatable packages, run smog sync. For a list of all packages and their status, run smog sync -m (packages that are up-to-date, will be printed in grey). To sync a single package, run smog sync PKG.

Remove a package

To remove a package and all of its files, run smog remove PKG. If smog created symlinks for PKG, it will first unlink them, delete the local repostory and the metadata file of the package.

If you only want to remove symlinks of a package, run smog unlink PKG.

Config

smog is configured by a set of bash variables that are sourced from the file config. Default values are provided in config-def.

variable type description
GIT command name git command invoked to clone, manage and query git repositories.
normally this is just git, I use Void's chroot-git.
EDITOR command name text editor to allow user to modify a generated list.
default is vim, but you shoud be able to use vi, nano or a GUI text editor.
ROOT absolute path The parent of the directories that smog operates on (described below).
this can't be empty, should be writable and is normally $HOME.
allows you to chroot or sandbox smog.
BIN (optional) path in $ROOT directory where symlinks are created to binaries and exectubles (when you run smog link PKG).
LIB (optional) path in $ROOT directory where symlinks are created to shared libraries (when you run smog link PKG).
PKG path in $ROOT directory where repositories are cloned (when you run smog get URL).
SMOG path in $ROOT smog home directory - containing the source-code and config.
if you bootstrapped smog, this directory is $PKG/github.com/vgratian/smog.
MDD path in $ROOT location of metadata files
LOCALBRANCH string name of the local working branch - smog will create this branch after cloning a repostory, this helps to isolate your local changes from the upstream source code and update the repository smoothly.
note: avoid names that might conflict remote branch names, such as master, main.
NPROCS integer number of processes when syncing packages (passed to xargs).
default is number of CPUs * 2, use 0 to run as many as possible.
BASHRC (optional) path in $ROOT only used for bootstrapping:
if not empty, the bootstrapper will edit $BASHRC to:
LDSOCONF (optional) absolute path only used for bootstrapping:
if not empty and if $LIB is defined, the bootstrapper will, create a config file for ldconfig allowing it to cache sharedlibs in smog packages.
note: this is normally in /etc/ld.so.conf.d/ and the bootstrapper might ask for sudo to create LDSOCONF.

Modes

Each smog package, a.k.a. your local repository, is associated with a remote git reference: either tag or branch. This decides how the package is updated:

  • branch mode: package is updatable if new commit(s) are available on the remote branch. smog update PKG will pull these commits and merge to $LOCALBRANCH.
  • tag mode: package is updatable if tag(s) are available on the remote repository that appear to be more recent. smog update PKG will pull the most recent tag and merge to $LOCALBRANCH.

Since tags tend to be more stable, this is the preferred mode when new package is created. If no tags are available, smog will prefer the branch master or main. You can control this behaviour with the -t, -b, -T <TAG> and -B <BRANCH>. Examples:

  • Add package dino in branch mode:
smog get https://github.com/dino/dino -b
  • Add dino in branch mode and use branch feature/handy:
smog get https://github.com/dino/dino -B feature/handy
  • Add package dwm in tag mode and use tag 6.0:
smog get git://git.suckless.org/dwm -T 6.0
  • Update dwm to tag 6.2:
smog update dwm -T 6.2

Metadata

Each package is defined by a plain-text metadata file in the directory $MDD. The basic variables are described below:

variable type description
url string the address of the remote repository
path string relative path of local repository in $PKG
mode string tag or branch
ref string name of the remote reference
sha string (only in branch-mode) SHA-1 of the last commit we have pulled from the remote branch

Manipulating metadata

Normally you should not want to touch metadata files. But there are a few optional variables, that help to control how the package is managed (warning: these might change in the future):

variable type description
builddir string subdirectory to scan when running smog link PKG
tag_pattern regex only check matching tags when running smog update PKG

These variables can be changed with the unadvertised command smog set PKG key=value.

Contributing

If you use smog, I would appreciate if you provide feedback or open an issue for a bug or feature request.

You are also welcome to open a merge request, if you want to contribute to this project.

If the project grows, it will be likely that bash is replaced by a compiled language.