mirror of
https://github.com/GitJournal/GitJournal.git
synced 2025-07-15 07:56:11 +08:00
526 lines
13 KiB
C
526 lines
13 KiB
C
#include "gitjournal.h"
|
|
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <sys/stat.h>
|
|
#include <errno.h>
|
|
#include <stdarg.h>
|
|
|
|
#include <git2.h>
|
|
|
|
#define GJ_ERR_EMPTY_COMMIT -954
|
|
|
|
void gj_log_internal(const char *format, ...)
|
|
{
|
|
char buffer[1024];
|
|
va_list args;
|
|
va_start(args, format);
|
|
vsprintf(buffer, format, args);
|
|
gj_log(buffer);
|
|
va_end(args);
|
|
}
|
|
|
|
gj_error *gj_error_info(int err)
|
|
{
|
|
if (err == 0)
|
|
return NULL;
|
|
|
|
gj_error *error = (gj_error *)malloc(sizeof(gj_error));
|
|
error->message_allocated = false;
|
|
if (err == GJ_ERR_EMPTY_COMMIT)
|
|
{
|
|
error->code = 954;
|
|
error->message = "Empty Commit";
|
|
return error;
|
|
}
|
|
|
|
const git_error *e = git_error_last();
|
|
if (e)
|
|
{
|
|
error->code = e->klass;
|
|
error->message = (char *)malloc(strlen(e->message));
|
|
strcpy(error->message, e->message);
|
|
error->message_allocated = true;
|
|
}
|
|
else
|
|
{
|
|
error->code = 1000;
|
|
error->message = "Unknown Message";
|
|
}
|
|
|
|
return error;
|
|
}
|
|
|
|
void gj_error_free(const gj_error *err)
|
|
{
|
|
if (err->message_allocated)
|
|
free(err->message);
|
|
free((void *)err);
|
|
}
|
|
|
|
int match_cb(const char *path, const char *spec, void *payload)
|
|
{
|
|
gj_log_internal("Match: %s\n", path);
|
|
return 0;
|
|
}
|
|
|
|
int gj_git_add(const char *git_base_path, const char *pattern)
|
|
{
|
|
int err;
|
|
git_repository *repo = NULL;
|
|
git_index *index = NULL;
|
|
|
|
err = git_repository_open(&repo, git_base_path);
|
|
if (err < 0)
|
|
goto cleanup;
|
|
|
|
err = git_repository_index(&index, repo);
|
|
if (err < 0)
|
|
goto cleanup;
|
|
|
|
char *paths[] = {(char *)pattern};
|
|
git_strarray pathspec = {paths, 1};
|
|
|
|
err = git_index_add_all(index, &pathspec, GIT_INDEX_ADD_DEFAULT, match_cb, NULL);
|
|
if (err < 0)
|
|
goto cleanup;
|
|
|
|
err = git_index_write(index);
|
|
if (err < 0)
|
|
goto cleanup;
|
|
|
|
cleanup:
|
|
git_index_free(index);
|
|
git_repository_free(repo);
|
|
|
|
return err;
|
|
}
|
|
|
|
int rm_match_cb(const char *path, const char *spec, void *payload)
|
|
{
|
|
char *git_base_path = (char *)payload;
|
|
if (!git_base_path)
|
|
{
|
|
gj_log_internal("git_base_path not in payload. Why?\n");
|
|
return 1;
|
|
}
|
|
|
|
int full_path_length = strlen(git_base_path) + 1 + strlen(path);
|
|
char *full_path = (char *)malloc(full_path_length);
|
|
strcpy(full_path, git_base_path);
|
|
strcat(full_path, "/"); // FIXME: Will not work on windows!
|
|
strcat(full_path, path);
|
|
|
|
int err = remove(full_path);
|
|
if (err != 0)
|
|
{
|
|
gj_log_internal("File could not be deleted: %s %d\n", full_path, errno);
|
|
if (errno == ENOENT)
|
|
{
|
|
gj_log_internal("ENOENT\n");
|
|
}
|
|
}
|
|
|
|
free(full_path);
|
|
return 0;
|
|
}
|
|
|
|
int gj_git_rm(const char *git_base_path, const char *pattern)
|
|
{
|
|
int err;
|
|
git_repository *repo = NULL;
|
|
git_index *index = NULL;
|
|
|
|
err = git_repository_open(&repo, git_base_path);
|
|
if (err < 0)
|
|
goto cleanup;
|
|
|
|
err = git_repository_index(&index, repo);
|
|
if (err < 0)
|
|
goto cleanup;
|
|
|
|
char *paths[] = {(char *)pattern};
|
|
git_strarray pathspec = {paths, 1};
|
|
|
|
void *payload = (void *)git_base_path;
|
|
err = git_index_remove_all(index, &pathspec, rm_match_cb, (void *)git_base_path);
|
|
if (err < 0)
|
|
goto cleanup;
|
|
|
|
err = git_index_write(index);
|
|
if (err < 0)
|
|
goto cleanup;
|
|
|
|
cleanup:
|
|
git_index_free(index);
|
|
git_repository_free(repo);
|
|
|
|
return err;
|
|
}
|
|
|
|
int gj_git_init(const char *git_base_path)
|
|
{
|
|
int err = 0;
|
|
git_repository *repo = NULL;
|
|
|
|
git_repository_init_options initopts = GIT_REPOSITORY_INIT_OPTIONS_INIT;
|
|
initopts.flags = GIT_REPOSITORY_INIT_MKPATH;
|
|
initopts.workdir_path = git_base_path;
|
|
|
|
err = git_repository_init_ext(&repo, git_base_path, &initopts);
|
|
if (err < 0)
|
|
goto cleanup;
|
|
|
|
cleanup:
|
|
git_repository_free(repo);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int gj_git_reset_hard(const char *git_base_path, const char *ref)
|
|
{
|
|
int err = 0;
|
|
git_repository *repo = NULL;
|
|
git_object *obj = NULL;
|
|
|
|
err = git_repository_open(&repo, git_base_path);
|
|
if (err < 0)
|
|
goto cleanup;
|
|
|
|
err = git_revparse_single(&obj, repo, ref);
|
|
if (err < 0)
|
|
goto cleanup;
|
|
|
|
err = git_reset(repo, obj, GIT_RESET_HARD, NULL);
|
|
if (err < 0)
|
|
goto cleanup;
|
|
|
|
cleanup:
|
|
git_object_free(obj);
|
|
git_repository_free(repo);
|
|
|
|
return err;
|
|
}
|
|
|
|
// FIXME: Add a datetime str
|
|
int gj_git_commit(const char *git_base_path, const char *author_name,
|
|
const char *author_email, const char *message)
|
|
{
|
|
int err = 0;
|
|
git_signature *sig = NULL;
|
|
git_index *index = NULL;
|
|
git_oid tree_id, commit_id;
|
|
git_tree *tree = NULL;
|
|
git_repository *repo = NULL;
|
|
git_oid parent_id;
|
|
git_commit *parent_commit = NULL;
|
|
|
|
err = git_repository_open(&repo, git_base_path);
|
|
if (err < 0)
|
|
goto cleanup;
|
|
|
|
err = git_repository_index(&index, repo);
|
|
if (err < 0)
|
|
goto cleanup;
|
|
|
|
int numOps = git_index_entrycount(index);
|
|
if (numOps == 0)
|
|
{
|
|
err = GJ_ERR_EMPTY_COMMIT;
|
|
goto cleanup;
|
|
}
|
|
|
|
err = git_signature_now(&sig, author_name, author_email);
|
|
if (err < 0)
|
|
goto cleanup;
|
|
|
|
err = git_index_write_tree(&tree_id, index);
|
|
if (err < 0)
|
|
goto cleanup;
|
|
|
|
err = git_tree_lookup(&tree, repo, &tree_id);
|
|
if (err < 0)
|
|
goto cleanup;
|
|
|
|
err = git_reference_name_to_id(&parent_id, repo, "HEAD");
|
|
if (err < 0)
|
|
{
|
|
if (err != GIT_ENOTFOUND)
|
|
goto cleanup;
|
|
|
|
git_error_clear();
|
|
|
|
err = git_commit_create(&commit_id, repo, "HEAD", sig, sig, NULL, message, tree, 0, NULL);
|
|
if (err < 0)
|
|
goto cleanup;
|
|
}
|
|
else
|
|
{
|
|
err = git_commit_lookup(&parent_commit, repo, &parent_id);
|
|
if (err < 0)
|
|
goto cleanup;
|
|
|
|
const git_commit *parents = {parent_commit};
|
|
err = git_commit_create(&commit_id, repo, "HEAD", sig, sig, NULL, message, tree, 1, &parents);
|
|
if (err < 0)
|
|
goto cleanup;
|
|
}
|
|
|
|
cleanup:
|
|
git_index_free(index);
|
|
git_commit_free(parent_commit);
|
|
git_tree_free(tree);
|
|
git_repository_free(repo);
|
|
git_signature_free(sig);
|
|
|
|
return err;
|
|
}
|
|
|
|
int fetch_progress(const git_transfer_progress *stats, void *payload)
|
|
{
|
|
int fetch_percent =
|
|
(100 * stats->received_objects) /
|
|
stats->total_objects;
|
|
int index_percent =
|
|
(100 * stats->indexed_objects) /
|
|
stats->total_objects;
|
|
int kbytes = stats->received_bytes / 1024;
|
|
|
|
gj_log_internal("network %3d%% (%4d kb, %5d/%5d) /"
|
|
" index %3d%% (%5d/%5d)\n",
|
|
fetch_percent, kbytes,
|
|
stats->received_objects, stats->total_objects,
|
|
index_percent,
|
|
stats->indexed_objects, stats->total_objects);
|
|
return 0;
|
|
}
|
|
|
|
char *g_public_key_path = NULL;
|
|
char *g_private_key_path = NULL;
|
|
char *g_passcode = NULL;
|
|
|
|
void gj_set_ssh_keys_paths(char *public_key, char *private_key, char *passcode)
|
|
{
|
|
g_public_key_path = public_key;
|
|
g_private_key_path = private_key;
|
|
g_passcode = passcode;
|
|
}
|
|
|
|
typedef struct
|
|
{
|
|
bool first_time;
|
|
} gj_credentials_payload;
|
|
|
|
int credentials_cb(git_cred **out, const char *url, const char *username_from_url,
|
|
unsigned int allowed_types, void *payload)
|
|
{
|
|
if (!payload)
|
|
{
|
|
gj_log_internal("credentials_cb has no payload\n");
|
|
return -1;
|
|
}
|
|
gj_credentials_payload *gj_payload = (gj_credentials_payload *)payload;
|
|
if (!gj_payload->first_time)
|
|
{
|
|
gj_log_internal("GitJournal: Credentials have been tried and they failed\n");
|
|
return -1;
|
|
}
|
|
|
|
gj_log_internal("UsernameProvided: %s\n", username_from_url);
|
|
gj_log_internal("Allowed Types: %d\n", allowed_types);
|
|
gj_log_internal("Payload: %p\n", payload);
|
|
|
|
if (!(allowed_types & GIT_CREDTYPE_SSH_KEY))
|
|
{
|
|
gj_log_internal("Some other auth mechanism is being used: %d\n", allowed_types);
|
|
return -1;
|
|
}
|
|
|
|
gj_log_internal("gj_paylaod: %p\n", gj_payload);
|
|
gj_payload->first_time = false;
|
|
return git_cred_ssh_key_new(out, username_from_url,
|
|
g_public_key_path, g_private_key_path, g_passcode);
|
|
}
|
|
|
|
int certificate_check_cb(git_cert *cert, int valid, const char *host, void *payload)
|
|
{
|
|
gj_log_internal("Valid: %d\n", valid);
|
|
gj_log_internal("CertType: %d\n", cert->cert_type);
|
|
|
|
if (valid == 0)
|
|
{
|
|
gj_log_internal("%s: Invalid certificate\n", host);
|
|
}
|
|
|
|
if (cert->cert_type == GIT_CERT_HOSTKEY_LIBSSH2)
|
|
{
|
|
gj_log_internal("LibSSH2 Key: %p\n", payload);
|
|
return 0;
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
int gj_git_clone(const char *clone_url, const char *git_base_path)
|
|
{
|
|
int err;
|
|
git_repository *repo = NULL;
|
|
git_clone_options options = GIT_CLONE_OPTIONS_INIT;
|
|
options.fetch_opts.callbacks.transfer_progress = fetch_progress;
|
|
options.fetch_opts.callbacks.certificate_check = certificate_check_cb;
|
|
|
|
gj_credentials_payload gj_payload = {true};
|
|
options.fetch_opts.callbacks.payload = (void *)&gj_payload;
|
|
options.fetch_opts.callbacks.credentials = credentials_cb;
|
|
|
|
err = git_clone(&repo, clone_url, git_base_path, &options);
|
|
if (err < 0)
|
|
goto cleanup;
|
|
|
|
cleanup:
|
|
git_repository_free(repo);
|
|
|
|
return err;
|
|
}
|
|
|
|
// FIXME: What if the 'HEAD" does not point to 'master'
|
|
int gj_git_push(const char *git_base_path)
|
|
{
|
|
int err = 0;
|
|
git_repository *repo = NULL;
|
|
git_remote *remote = NULL;
|
|
git_oid head_id;
|
|
|
|
err = git_repository_open(&repo, git_base_path);
|
|
if (err < 0)
|
|
goto cleanup;
|
|
|
|
err = git_remote_lookup(&remote, repo, "origin");
|
|
if (err < 0)
|
|
goto cleanup;
|
|
|
|
char *name = "refs/heads/master";
|
|
const git_strarray refs = {&name, 1};
|
|
|
|
git_push_options options = GIT_PUSH_OPTIONS_INIT;
|
|
|
|
gj_credentials_payload gj_payload = {true};
|
|
options.callbacks.payload = (void *)&gj_payload;
|
|
options.callbacks.credentials = credentials_cb;
|
|
|
|
err = git_remote_push(remote, &refs, &options);
|
|
if (err < 0)
|
|
goto cleanup;
|
|
|
|
cleanup:
|
|
git_remote_free(remote);
|
|
git_repository_free(repo);
|
|
|
|
return err;
|
|
}
|
|
|
|
int gj_git_pull(const char *git_base_path, const char *author_name, const char *author_email)
|
|
{
|
|
int err = 0;
|
|
git_repository *repo = NULL;
|
|
git_remote *remote = NULL;
|
|
git_annotated_commit *annotated_commit = NULL;
|
|
git_reference *ref = NULL;
|
|
git_index *index = NULL;
|
|
git_index_conflict_iterator *conflict_iter = NULL;
|
|
|
|
err = git_repository_open(&repo, git_base_path);
|
|
if (err < 0)
|
|
goto cleanup;
|
|
|
|
err = git_remote_lookup(&remote, repo, "origin");
|
|
if (err < 0)
|
|
goto cleanup;
|
|
|
|
git_fetch_options options = GIT_FETCH_OPTIONS_INIT;
|
|
|
|
gj_credentials_payload gj_payload = {true};
|
|
options.callbacks.payload = (void *)&gj_payload;
|
|
options.callbacks.credentials = credentials_cb;
|
|
|
|
err = git_remote_fetch(remote, NULL, &options, NULL);
|
|
if (err < 0)
|
|
goto cleanup;
|
|
|
|
git_merge_options merge_options = GIT_MERGE_OPTIONS_INIT;
|
|
git_checkout_options checkout_options = GIT_CHECKOUT_OPTIONS_INIT;
|
|
|
|
// FIXME: Maybe I should be taking the head of the remote?
|
|
err = git_repository_head(&ref, repo);
|
|
if (err < 0)
|
|
goto cleanup;
|
|
|
|
err = git_annotated_commit_from_ref(&annotated_commit, repo, ref);
|
|
if (err < 0)
|
|
goto cleanup;
|
|
|
|
err = git_merge(repo, (const git_annotated_commit **)&annotated_commit, 1,
|
|
&merge_options, &checkout_options);
|
|
if (err < 0)
|
|
goto cleanup;
|
|
|
|
err = git_repository_index(&index, repo);
|
|
if (err < 0)
|
|
goto cleanup;
|
|
|
|
err = git_index_conflict_iterator_new(&conflict_iter, index);
|
|
if (err < 0)
|
|
goto cleanup;
|
|
|
|
// Handle Conflicts
|
|
while (1)
|
|
{
|
|
git_index_entry *ancestor_out;
|
|
git_index_entry *our_out;
|
|
git_index_entry *their_out;
|
|
err = git_index_conflict_next((const git_index_entry **)&ancestor_out,
|
|
(const git_index_entry **)&our_out,
|
|
(const git_index_entry **)&their_out,
|
|
conflict_iter);
|
|
|
|
if (err == GIT_ITEROVER)
|
|
{
|
|
gj_log_internal(" No Conflicts\n");
|
|
break;
|
|
}
|
|
if (err < 0)
|
|
goto cleanup;
|
|
|
|
// FIXME: This isn't what I want. I want 'theirs' to be applied!
|
|
// How to do that?
|
|
git_index_conflict_remove(index, their_out->path);
|
|
}
|
|
|
|
err = git_index_write(index);
|
|
if (err < 0)
|
|
goto cleanup;
|
|
|
|
//const git_commit *parents = {parent_commit};
|
|
//err = git_commit_create(&commit_id, repo, "HEAD", sig, sig, NULL, message, tree, 1, &parents);
|
|
|
|
cleanup:
|
|
git_repository_state_cleanup(repo);
|
|
git_index_conflict_iterator_free(conflict_iter);
|
|
git_index_free(index);
|
|
git_reference_free(ref);
|
|
git_annotated_commit_free(annotated_commit);
|
|
git_remote_free(remote);
|
|
git_repository_free(repo);
|
|
|
|
return err;
|
|
}
|
|
|
|
int gj_init()
|
|
{
|
|
return git_libgit2_init();
|
|
}
|
|
|
|
int gj_shutdown()
|
|
{
|
|
return git_libgit2_shutdown();
|
|
}
|