2016-08-10 06:53:33 +00:00
|
|
|
/* -*- Mode: C; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
|
|
|
|
|
|
|
|
#include "common.h"
|
|
|
|
#include <glib/gstdio.h>
|
|
|
|
|
|
|
|
#include "utils.h"
|
|
|
|
#include "log.h"
|
|
|
|
|
|
|
|
#include "seafile-session.h"
|
|
|
|
#include "commit-mgr.h"
|
|
|
|
#include "branch-mgr.h"
|
|
|
|
#include "repo-mgr.h"
|
|
|
|
#include "fs-mgr.h"
|
|
|
|
#include "seafile-error.h"
|
|
|
|
|
|
|
|
#include "seaf-db.h"
|
|
|
|
|
|
|
|
#define INDEX_DIR "index"
|
|
|
|
|
|
|
|
struct _SeafRepoManagerPriv {
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
static SeafRepo *
|
|
|
|
load_repo (SeafRepoManager *manager, const char *repo_id);
|
|
|
|
|
|
|
|
gboolean
|
|
|
|
is_repo_id_valid (const char *id)
|
|
|
|
{
|
|
|
|
if (!id)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
return is_uuid_valid (id);
|
|
|
|
}
|
|
|
|
|
|
|
|
SeafRepo*
|
|
|
|
seaf_repo_new (const char *id, const char *name, const char *desc)
|
|
|
|
{
|
|
|
|
SeafRepo* repo;
|
|
|
|
|
|
|
|
/* valid check */
|
|
|
|
|
|
|
|
|
|
|
|
repo = g_new0 (SeafRepo, 1);
|
|
|
|
memcpy (repo->id, id, 36);
|
|
|
|
repo->id[36] = '\0';
|
|
|
|
|
|
|
|
repo->name = g_strdup(name);
|
|
|
|
repo->desc = g_strdup(desc);
|
|
|
|
|
|
|
|
repo->ref_cnt = 1;
|
|
|
|
|
|
|
|
return repo;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
seaf_repo_free (SeafRepo *repo)
|
|
|
|
{
|
|
|
|
if (repo->name) g_free (repo->name);
|
|
|
|
if (repo->desc) g_free (repo->desc);
|
|
|
|
if (repo->category) g_free (repo->category);
|
|
|
|
if (repo->head) seaf_branch_unref (repo->head);
|
|
|
|
g_free (repo);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
seaf_repo_ref (SeafRepo *repo)
|
|
|
|
{
|
|
|
|
g_atomic_int_inc (&repo->ref_cnt);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
seaf_repo_unref (SeafRepo *repo)
|
|
|
|
{
|
|
|
|
if (!repo)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (g_atomic_int_dec_and_test (&repo->ref_cnt))
|
|
|
|
seaf_repo_free (repo);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
set_head_common (SeafRepo *repo, SeafBranch *branch)
|
|
|
|
{
|
|
|
|
if (repo->head)
|
|
|
|
seaf_branch_unref (repo->head);
|
|
|
|
repo->head = branch;
|
|
|
|
seaf_branch_ref(branch);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
seaf_repo_from_commit (SeafRepo *repo, SeafCommit *commit)
|
|
|
|
{
|
|
|
|
repo->name = g_strdup (commit->repo_name);
|
|
|
|
repo->desc = g_strdup (commit->repo_desc);
|
|
|
|
repo->encrypted = commit->encrypted;
|
|
|
|
if (repo->encrypted) {
|
|
|
|
repo->enc_version = commit->enc_version;
|
|
|
|
if (repo->enc_version >= 1)
|
|
|
|
memcpy (repo->magic, commit->magic, 33);
|
|
|
|
}
|
|
|
|
repo->no_local_history = commit->no_local_history;
|
|
|
|
repo->version = commit->version;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
seaf_repo_to_commit (SeafRepo *repo, SeafCommit *commit)
|
|
|
|
{
|
|
|
|
commit->repo_name = g_strdup (repo->name);
|
|
|
|
commit->repo_desc = g_strdup (repo->desc);
|
|
|
|
commit->encrypted = repo->encrypted;
|
|
|
|
if (commit->encrypted) {
|
|
|
|
commit->enc_version = repo->enc_version;
|
|
|
|
if (commit->enc_version >= 1)
|
|
|
|
commit->magic = g_strdup (repo->magic);
|
|
|
|
}
|
|
|
|
commit->no_local_history = repo->no_local_history;
|
|
|
|
commit->version = repo->version;
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
collect_commit (SeafCommit *commit, void *vlist, gboolean *stop)
|
|
|
|
{
|
|
|
|
GList **commits = vlist;
|
|
|
|
|
|
|
|
/* The traverse function will unref the commit, so we need to ref it.
|
|
|
|
*/
|
|
|
|
seaf_commit_ref (commit);
|
|
|
|
*commits = g_list_prepend (*commits, commit);
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
GList *
|
|
|
|
seaf_repo_get_commits (SeafRepo *repo)
|
|
|
|
{
|
|
|
|
GList *branches;
|
|
|
|
GList *ptr;
|
|
|
|
SeafBranch *branch;
|
|
|
|
GList *commits = NULL;
|
|
|
|
|
|
|
|
branches = seaf_branch_manager_get_branch_list (seaf->branch_mgr, repo->id);
|
|
|
|
if (branches == NULL) {
|
|
|
|
seaf_warning ("Failed to get branch list of repo %s.\n", repo->id);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (ptr = branches; ptr != NULL; ptr = ptr->next) {
|
|
|
|
branch = ptr->data;
|
|
|
|
gboolean res = seaf_commit_manager_traverse_commit_tree (seaf->commit_mgr,
|
|
|
|
repo->id,
|
|
|
|
repo->version,
|
|
|
|
branch->commit_id,
|
|
|
|
collect_commit,
|
|
|
|
&commits,
|
|
|
|
FALSE);
|
|
|
|
if (!res) {
|
|
|
|
for (ptr = commits; ptr != NULL; ptr = ptr->next)
|
|
|
|
seaf_commit_unref ((SeafCommit *)(ptr->data));
|
|
|
|
g_list_free (commits);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
commits = g_list_reverse (commits);
|
|
|
|
|
|
|
|
out:
|
|
|
|
for (ptr = branches; ptr != NULL; ptr = ptr->next) {
|
|
|
|
seaf_branch_unref ((SeafBranch *)ptr->data);
|
|
|
|
}
|
|
|
|
return commits;
|
|
|
|
}
|
|
|
|
|
2021-06-09 03:03:17 +00:00
|
|
|
#if 0
|
2016-08-10 06:53:33 +00:00
|
|
|
static int
|
|
|
|
compare_repo (const SeafRepo *srepo, const SeafRepo *trepo)
|
|
|
|
{
|
|
|
|
return g_strcmp0 (srepo->id, trepo->id);
|
|
|
|
}
|
2021-06-09 03:03:17 +00:00
|
|
|
#endif
|
2016-08-10 06:53:33 +00:00
|
|
|
|
|
|
|
SeafRepoManager*
|
|
|
|
seaf_repo_manager_new (SeafileSession *seaf)
|
|
|
|
{
|
|
|
|
SeafRepoManager *mgr = g_new0 (SeafRepoManager, 1);
|
|
|
|
|
|
|
|
mgr->priv = g_new0 (SeafRepoManagerPriv, 1);
|
|
|
|
mgr->seaf = seaf;
|
|
|
|
|
|
|
|
return mgr;
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
seaf_repo_manager_init (SeafRepoManager *mgr)
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
seaf_repo_manager_start (SeafRepoManager *mgr)
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
repo_exists_in_db (SeafDB *db, const char *id)
|
|
|
|
{
|
|
|
|
char sql[256];
|
|
|
|
gboolean db_err = FALSE;
|
|
|
|
|
|
|
|
snprintf (sql, sizeof(sql), "SELECT repo_id FROM Repo WHERE repo_id = '%s'",
|
|
|
|
id);
|
|
|
|
return seaf_db_check_for_existence (db, sql, &db_err);
|
|
|
|
}
|
|
|
|
|
|
|
|
SeafRepo*
|
|
|
|
seaf_repo_manager_get_repo (SeafRepoManager *manager, const gchar *id)
|
|
|
|
{
|
|
|
|
SeafRepo repo;
|
|
|
|
int len = strlen(id);
|
|
|
|
|
|
|
|
if (len >= 37)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
memcpy (repo.id, id, len + 1);
|
|
|
|
|
|
|
|
if (repo_exists_in_db (manager->seaf->db, id)) {
|
|
|
|
SeafRepo *ret = load_repo (manager, id);
|
|
|
|
if (!ret)
|
|
|
|
return NULL;
|
|
|
|
/* seaf_repo_ref (ret); */
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
gboolean
|
|
|
|
seaf_repo_manager_repo_exists (SeafRepoManager *manager, const gchar *id)
|
|
|
|
{
|
|
|
|
SeafRepo repo;
|
|
|
|
memcpy (repo.id, id, 37);
|
|
|
|
|
|
|
|
return repo_exists_in_db (manager->seaf->db, id);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
load_repo_commit (SeafRepoManager *manager,
|
|
|
|
SeafRepo *repo,
|
|
|
|
SeafBranch *branch)
|
|
|
|
{
|
|
|
|
SeafCommit *commit;
|
|
|
|
|
|
|
|
commit = seaf_commit_manager_get_commit_compatible (manager->seaf->commit_mgr,
|
|
|
|
repo->id,
|
|
|
|
branch->commit_id);
|
|
|
|
if (!commit) {
|
|
|
|
seaf_warning ("Commit %s is missing\n", branch->commit_id);
|
|
|
|
repo->is_corrupted = TRUE;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
set_head_common (repo, branch);
|
|
|
|
seaf_repo_from_commit (repo, commit);
|
|
|
|
|
|
|
|
seaf_commit_unref (commit);
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
load_virtual_info (SeafDBRow *row, void *vrepo_id)
|
|
|
|
{
|
|
|
|
char *ret_repo_id = vrepo_id;
|
|
|
|
const char *origin_repo_id;
|
|
|
|
|
|
|
|
origin_repo_id = seaf_db_row_get_column_text (row, 0);
|
|
|
|
memcpy (ret_repo_id, origin_repo_id, 37);
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
char *
|
|
|
|
get_origin_repo_id (SeafRepoManager *mgr, const char *repo_id)
|
|
|
|
{
|
|
|
|
char sql[256];
|
|
|
|
char origin_repo_id[37];
|
|
|
|
|
|
|
|
memset (origin_repo_id, 0, 37);
|
|
|
|
|
|
|
|
snprintf (sql, 256,
|
|
|
|
"SELECT origin_repo FROM VirtualRepo "
|
|
|
|
"WHERE repo_id = '%s'", repo_id);
|
|
|
|
seaf_db_foreach_selected_row (seaf->db, sql, load_virtual_info, origin_repo_id);
|
|
|
|
|
|
|
|
if (origin_repo_id[0] != 0)
|
|
|
|
return g_strdup(origin_repo_id);
|
|
|
|
else
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static SeafRepo *
|
|
|
|
load_repo (SeafRepoManager *manager, const char *repo_id)
|
|
|
|
{
|
|
|
|
SeafRepo *repo;
|
|
|
|
SeafBranch *branch;
|
|
|
|
|
|
|
|
repo = seaf_repo_new(repo_id, NULL, NULL);
|
|
|
|
if (!repo) {
|
|
|
|
seaf_warning ("[repo mgr] failed to alloc repo.\n");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
repo->manager = manager;
|
|
|
|
|
|
|
|
branch = seaf_branch_manager_get_branch (seaf->branch_mgr, repo_id, "master");
|
|
|
|
if (!branch) {
|
|
|
|
seaf_warning ("Failed to get master branch of repo %.8s.\n", repo_id);
|
|
|
|
repo->is_corrupted = TRUE;
|
|
|
|
} else {
|
|
|
|
load_repo_commit (manager, repo, branch);
|
|
|
|
seaf_branch_unref (branch);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (repo->is_corrupted) {
|
|
|
|
seaf_warning ("Repo %.8s is corrupted.\n", repo->id);
|
|
|
|
seaf_repo_free (repo);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
char *origin_repo_id = get_origin_repo_id (manager, repo->id);
|
|
|
|
if (origin_repo_id)
|
|
|
|
memcpy (repo->store_id, origin_repo_id, 36);
|
|
|
|
else
|
|
|
|
memcpy (repo->store_id, repo->id, 36);
|
|
|
|
g_free (origin_repo_id);
|
|
|
|
|
|
|
|
return repo;
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
collect_repo_id (SeafDBRow *row, void *data)
|
|
|
|
{
|
|
|
|
GList **p_ids = data;
|
|
|
|
const char *repo_id;
|
|
|
|
|
|
|
|
repo_id = seaf_db_row_get_column_text (row, 0);
|
|
|
|
*p_ids = g_list_prepend (*p_ids, g_strdup(repo_id));
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
GList *
|
|
|
|
seaf_repo_manager_get_repo_id_list (SeafRepoManager *mgr)
|
|
|
|
{
|
|
|
|
GList *ret = NULL;
|
|
|
|
char sql[256];
|
|
|
|
|
|
|
|
snprintf (sql, 256, "SELECT repo_id FROM Repo");
|
|
|
|
|
|
|
|
if (seaf_db_foreach_selected_row (mgr->seaf->db, sql,
|
|
|
|
collect_repo_id, &ret) < 0)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
GList *
|
|
|
|
seaf_repo_manager_get_repo_list (SeafRepoManager *mgr, int start, int limit)
|
|
|
|
{
|
|
|
|
GList *id_list = NULL, *ptr;
|
|
|
|
GList *ret = NULL;
|
|
|
|
SeafRepo *repo;
|
|
|
|
char sql[256];
|
|
|
|
|
|
|
|
if (start == -1 && limit == -1)
|
|
|
|
snprintf (sql, 256, "SELECT repo_id FROM Repo");
|
|
|
|
else
|
|
|
|
snprintf (sql, 256, "SELECT repo_id FROM Repo LIMIT %d, %d", start, limit);
|
|
|
|
|
|
|
|
if (seaf_db_foreach_selected_row (mgr->seaf->db, sql,
|
|
|
|
collect_repo_id, &id_list) < 0)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
for (ptr = id_list; ptr; ptr = ptr->next) {
|
|
|
|
char *repo_id = ptr->data;
|
|
|
|
repo = seaf_repo_manager_get_repo (mgr, repo_id);
|
|
|
|
if (repo != NULL)
|
|
|
|
ret = g_list_prepend (ret, repo);
|
|
|
|
}
|
|
|
|
|
|
|
|
string_list_free (id_list);
|
|
|
|
return g_list_reverse (ret);
|
|
|
|
}
|
|
|
|
|
|
|
|
GList *
|
|
|
|
seaf_repo_manager_get_repos_by_owner (SeafRepoManager *mgr,
|
|
|
|
const char *email)
|
|
|
|
{
|
|
|
|
GList *id_list = NULL, *ptr;
|
|
|
|
GList *ret = NULL;
|
|
|
|
char sql[256];
|
|
|
|
|
|
|
|
snprintf (sql, 256, "SELECT repo_id FROM RepoOwner WHERE owner_id='%s'",
|
|
|
|
email);
|
|
|
|
|
|
|
|
if (seaf_db_foreach_selected_row (mgr->seaf->db, sql,
|
|
|
|
collect_repo_id, &id_list) < 0)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
for (ptr = id_list; ptr; ptr = ptr->next) {
|
|
|
|
char *repo_id = ptr->data;
|
|
|
|
SeafRepo *repo = seaf_repo_manager_get_repo (mgr, repo_id);
|
|
|
|
if (repo != NULL)
|
|
|
|
ret = g_list_prepend (ret, repo);
|
|
|
|
}
|
|
|
|
|
|
|
|
string_list_free (id_list);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
gboolean
|
|
|
|
seaf_repo_manager_is_virtual_repo (SeafRepoManager *mgr, const char *repo_id)
|
|
|
|
{
|
|
|
|
char sql[256];
|
|
|
|
gboolean db_err;
|
|
|
|
|
|
|
|
snprintf (sql, 256,
|
|
|
|
"SELECT 1 FROM VirtualRepo WHERE repo_id = '%s'", repo_id);
|
|
|
|
return seaf_db_check_for_existence (seaf->db, sql, &db_err);
|
|
|
|
}
|