root/contrib/credential/osxkeychain/git-credential-osxkeychain.c

/* [<][>][^][v][top][bottom][index][help] */

DEFINITIONS

This source file includes following definitions.
  1. die
  2. xstrdup
  3. write_item
  4. find_username_in_item
  5. find_internet_password
  6. delete_internet_password
  7. add_internet_password
  8. read_credential
  9. main

#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <Security/Security.h>

static SecProtocolType protocol;
static char *host;
static char *path;
static char *username;
static char *password;
static UInt16 port;

static void die(const char *err, ...)
{
        char msg[4096];
        va_list params;
        va_start(params, err);
        vsnprintf(msg, sizeof(msg), err, params);
        fprintf(stderr, "%s\n", msg);
        va_end(params);
        exit(1);
}

static void *xstrdup(const char *s1)
{
        void *ret = strdup(s1);
        if (!ret)
                die("Out of memory");
        return ret;
}

#define KEYCHAIN_ITEM(x) (x ? strlen(x) : 0), x
#define KEYCHAIN_ARGS \
        NULL, /* default keychain */ \
        KEYCHAIN_ITEM(host), \
        0, NULL, /* account domain */ \
        KEYCHAIN_ITEM(username), \
        KEYCHAIN_ITEM(path), \
        port, \
        protocol, \
        kSecAuthenticationTypeDefault

static void write_item(const char *what, const char *buf, int len)
{
        printf("%s=", what);
        fwrite(buf, 1, len, stdout);
        putchar('\n');
}

static void find_username_in_item(SecKeychainItemRef item)
{
        SecKeychainAttributeList list;
        SecKeychainAttribute attr;

        list.count = 1;
        list.attr = &attr;
        attr.tag = kSecAccountItemAttr;

        if (SecKeychainItemCopyContent(item, NULL, &list, NULL, NULL))
                return;

        write_item("username", attr.data, attr.length);
        SecKeychainItemFreeContent(&list, NULL);
}

static void find_internet_password(void)
{
        void *buf;
        UInt32 len;
        SecKeychainItemRef item;

        if (SecKeychainFindInternetPassword(KEYCHAIN_ARGS, &len, &buf, &item))
                return;

        write_item("password", buf, len);
        if (!username)
                find_username_in_item(item);

        SecKeychainItemFreeContent(NULL, buf);
}

static void delete_internet_password(void)
{
        SecKeychainItemRef item;

        /*
         * Require at least a protocol and host for removal, which is what git
         * will give us; if you want to do something more fancy, use the
         * Keychain manager.
         */
        if (!protocol || !host)
                return;

        if (SecKeychainFindInternetPassword(KEYCHAIN_ARGS, 0, NULL, &item))
                return;

        SecKeychainItemDelete(item);
}

static void add_internet_password(void)
{
        /* Only store complete credentials */
        if (!protocol || !host || !username || !password)
                return;

        if (SecKeychainAddInternetPassword(
              KEYCHAIN_ARGS,
              KEYCHAIN_ITEM(password),
              NULL))
                return;
}

static void read_credential(void)
{
        char buf[1024];

        while (fgets(buf, sizeof(buf), stdin)) {
                char *v;

                if (!strcmp(buf, "\n"))
                        break;
                buf[strlen(buf)-1] = '\0';

                v = strchr(buf, '=');
                if (!v)
                        die("bad input: %s", buf);
                *v++ = '\0';

                if (!strcmp(buf, "protocol")) {
                        if (!strcmp(v, "imap"))
                                protocol = kSecProtocolTypeIMAP;
                        else if (!strcmp(v, "imaps"))
                                protocol = kSecProtocolTypeIMAPS;
                        else if (!strcmp(v, "ftp"))
                                protocol = kSecProtocolTypeFTP;
                        else if (!strcmp(v, "ftps"))
                                protocol = kSecProtocolTypeFTPS;
                        else if (!strcmp(v, "https"))
                                protocol = kSecProtocolTypeHTTPS;
                        else if (!strcmp(v, "http"))
                                protocol = kSecProtocolTypeHTTP;
                        else if (!strcmp(v, "smtp"))
                                protocol = kSecProtocolTypeSMTP;
                        else /* we don't yet handle other protocols */
                                exit(0);
                }
                else if (!strcmp(buf, "host")) {
                        char *colon = strchr(v, ':');
                        if (colon) {
                                *colon++ = '\0';
                                port = atoi(colon);
                        }
                        host = xstrdup(v);
                }
                else if (!strcmp(buf, "path"))
                        path = xstrdup(v);
                else if (!strcmp(buf, "username"))
                        username = xstrdup(v);
                else if (!strcmp(buf, "password"))
                        password = xstrdup(v);
        }
}

int main(int argc, const char **argv)
{
        const char *usage =
                "usage: git credential-osxkeychain <get|store|erase>";

        if (!argv[1])
                die(usage);

        read_credential();

        if (!strcmp(argv[1], "get"))
                find_internet_password();
        else if (!strcmp(argv[1], "store"))
                add_internet_password();
        else if (!strcmp(argv[1], "erase"))
                delete_internet_password();
        /* otherwise, ignore unknown action */

        return 0;
}

/* [<][>][^][v][top][bottom][index][help] */