Local git command wrappers
JavaScript
Latest commit 9829091 Jul 19, 2016 @bahmutov svg for badges
Failed to load latest commit information.
bin
docs
spec
src
test
.gitignore
.jshintrc
.npmrc
.travis.yml
Gruntfile.js
History.md
LICENSE
README.md
complexity.json
index.js
package.json

README.md

ggit

Local promise-returning git command wrappers

NPM

Build status dependencies devdependencies semantic-release manpm

Stand alone tool

You can install and run this tool as a stand alone CLI application.

npm install -g ggit
ggit --help
# get last commit id in the current folder, save into json file
ggit last -f build.json

API

cloneRepo

var clone = require('ggit').cloneRepo;
clone({
    url: 'git@github.com:bahmutov/test-next-updater.git',
    folder: 'folder to create, should not exist yet'
}).then(function () {
    console.log('cloned repo to destination folder');
});

exec

var exec = require('ggit').exec;
var cmd = 'rm -rf folder';
var verbose = true;
exec(cmd, verbose).then(function () {
    console.log('removed folder');
});

blame

Finds last person who has touched specific line in a file

  • filename - full or partial filename (from the repo's root)
  • lineNumber - starts with 1
var blame = require('ggit').blame;
blame(filename, lineNumber).then(function (info) {
  /*
    info is object with fields like
    { commit: '6e65f8ec5ed63cac92ed130b1246d9c23223c04e',
      author: 'Gleb Bahmutov',
      committer: 'Gleb Bahmutov',
      summary: 'adding blame feature',
      filename: 'test/blame.js',
      line: 'var blame = require(\'../index\').blame;' }
  */
});

Equivalent to porcelain git output: see git-blame

isTracked

Returns true if given path is tracked in the repo.

  • path
var isTracked = require('ggit').isTracked;
isTracked(filename).then(function (result) {
    // result is true or false
});

hasChanges

Returns true if there are local uncommitted stages

var changed = require('ggit').hasChanges;
changed().then(function (result) {
    // result is true or false
});

commit

Commit any changes with a given message. Second argument is optional and will be added after a blank line to the short main message.

var commit = require('ggit').commit;
commit('added foo', 'long text').then(function () {
    // after commit
});

You can pass the entire message if wanted as first argument

var fullMessage = 'first line\n\nbody of message\n';
commit(fullMessage).then(...);

push

Push commits to the remote

var psuh = require('ggit').psuh;
psuh().then(function () {
    // after the push
});

commits

Returns list of commits in the given folder as a list or object

// commits.all - gets all commits
var commits = require('ggit').commits;
commits.all(gitRepoFolder)
    .then(R.take(2))
    .then(console.table)
    .done();
// commits.byId - transforms list of commits into object
// where keys = ids, values = messages
// For example to get an object with 2 commit ids as keys
commits.all(gitRepoFolder)
    .then(R.take(2))
    .then(commits.byId)
    .then(console.log)
    .done();

Each object has at least 'id', 'message' and (maybe empty) 'body' properties.

trackedFiles

Returns all tracked source files in the given folder matching pattern. Both folder and pattern are optional.

require('ggit')
    .trackedFiles(__dirname, '*.js', options)
    .then(function (list) {
        console.log('javascript tracked in the current folder are');
        console.log(list);
    })
    .done();

The options argument is optional, and is passed directly to the glob package that does file discovery. The only important option to use is { dot: true } - if you want to find the filenames that start with .. For example to find ALL files in the repo call

require('ggit')
    .trackedFiles(__dirname, '**', { dot: true })
// returns .gitignore, .travis.yml, index.js etc

untrackedFiles

Returns all untracked source files in the repo.

require('ggit')
    .untrackedFiles()
    .then(function (list) {
        // list is Array of strings, could be empty
        console.log('untracked files are');
        console.log(list);
    })
    .done();

commitPerLine

Returns an object where for each key (filename) there is a list of commits for each line.

  • list of filenames
var perLine = require('ggit').commitPerLine;
perLine(['foo.js', 'bar.js']).then(function (result) {
    /*
    {
        'foo.js': [{
            commit: '3c6b01eb3c96db1cbdf277904545107ef97cbb56',
            author: 'Gleb Bahmutov',
            committer: 'Gleb Bahmutov',
            summary: 'cool commit',
            filename: 'foo.js',
            line: '// actual source line'
        },
            ...
        }],
        'bar.js': [...]
    }
    */
});

numstat

Returns info for a specific commit - number of lines changed, deleted. Same as $ git show --numstat <id>.

require('ggit')
    .numstat('5d3ee3')
    .then(function (result) {
        /* result is
            {
                commit: <full commit SHA>,
                author:
                message:
                date:
                changes: {
                    'filename 1': {
                        filename: 'filename 1',
                        added: 10,
                        deleted: 3
                    },
                    ...
                }
            }
        */
    })
    .done();

lastCommitId

Returns last commit id

require('ggit')
    .lastCommitId()
    .then(function (str) {
        // str is full SHA id string
    })
    .done();

You can pass options object as in lastCommitId(options) where

  • file - save id into the JSON file with the given file name.

branchName

Resolves with the current branch name

require('ggit').branchName()
    .then(function (name) {
        // name = "master" or whatever
    });

changed-files

Returns list of modified files

var changedFiles = require('ggit').changedFiles;
changedFiles()
    .then(function (files) {})
    .catch(function (error) {});

The object files groups filenames by modification property

{
    A: [...], // list of added files
    C: [...], // list of copied files
    M: [...], // list of modified files
    D: [...]  // list of deleted files
}
// each item in the list is
{
    diff: 'A' // or C, M, D
    name: 'src/something.js' // relative to the repo root
    filename: 'full path',
    before: 'file contents', // if available (for example M, D)
    after: 'file contents' // if available (for A, M)
}

This is a wrapper around two commands git diff --name-status --diff-filter=ACMD and git status --porcelain

fileContents

Returns the contents of a file at some point

  • filename - full or partial filename (from the repo's root)
  • at (optional) - checkpoint, HEAD by default
var fileContents = require('ggit').fileContents;
fileContents(filename).then(function (text) { ... });

Same as git show <at>:<name>

commitMessage

Returns the contents of the Git current commit message, usually for validation before the commit.

require('ggit').commitMessage()
    .then(function (text) {
      // do something with the message
    },
    function () {
      // file not found
    });

getGitFolder

const folder = require('ggit').getGitFolder();

Handles Git submodules

Development

Edit source, run unit tests, run end to end tests and push the code back to Github. The NPM publishing happens automatically using semantic release

npm test
npm run commit
git push

To debug problems, run the command with DEBUG=ggit environment variable enabled to see verbose logging.

Small print

Author: Gleb Bahmutov © 2015

License: MIT - do anything with the code, but don't blame uTest if it does not work.

Spread the word: tweet, star on github, etc.

Support: if you find any problems with this module, email / tweet / open issue on Github