mirror of
https://github.com/haiwen/seafile-server.git
synced 2025-06-26 15:11:34 +00:00
455 lines
13 KiB
C
455 lines
13 KiB
C
/* -*- Mode: C; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
|
|
|
|
#include "common.h"
|
|
|
|
#include <stdint.h>
|
|
#include <dirent.h>
|
|
#include <unistd.h>
|
|
#include <sys/types.h>
|
|
#include <sys/stat.h>
|
|
#include <fcntl.h>
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <errno.h>
|
|
|
|
#include <glib.h>
|
|
|
|
#include <ccnet/cevent.h>
|
|
#include "utils.h"
|
|
|
|
#include "seafile-session.h"
|
|
|
|
#include "seaf-db.h"
|
|
#include "seaf-utils.h"
|
|
|
|
#include "log.h"
|
|
|
|
#define CONNECT_INTERVAL_MSEC 10 * 1000
|
|
|
|
#define DEFAULT_THREAD_POOL_SIZE 500
|
|
#define DEFAULT_RPC_THREAD_POOL_SIZE 50
|
|
|
|
static int
|
|
load_thread_pool_config (SeafileSession *session);
|
|
|
|
SeafileSession *
|
|
seafile_session_new(const char *central_config_dir,
|
|
const char *seafile_dir,
|
|
CcnetClient *ccnet_session)
|
|
{
|
|
char *abs_central_config_dir = NULL;
|
|
char *abs_seafile_dir;
|
|
char *tmp_file_dir;
|
|
char *config_file_path;
|
|
GKeyFile *config;
|
|
SeafileSession *session = NULL;
|
|
|
|
if (!ccnet_session)
|
|
return NULL;
|
|
|
|
abs_seafile_dir = ccnet_expand_path (seafile_dir);
|
|
tmp_file_dir = g_build_filename (abs_seafile_dir, "tmpfiles", NULL);
|
|
if (central_config_dir) {
|
|
abs_central_config_dir = ccnet_expand_path (central_config_dir);
|
|
}
|
|
config_file_path = g_build_filename(
|
|
abs_central_config_dir ? abs_central_config_dir : abs_seafile_dir,
|
|
"seafile.conf", NULL);
|
|
|
|
if (checkdir_with_mkdir (abs_seafile_dir) < 0) {
|
|
seaf_warning ("Config dir %s does not exist and is unable to create\n",
|
|
abs_seafile_dir);
|
|
goto onerror;
|
|
}
|
|
|
|
if (checkdir_with_mkdir (tmp_file_dir) < 0) {
|
|
seaf_warning ("Temp file dir %s does not exist and is unable to create\n",
|
|
tmp_file_dir);
|
|
goto onerror;
|
|
}
|
|
|
|
GError *error = NULL;
|
|
config = g_key_file_new ();
|
|
if (!g_key_file_load_from_file (config, config_file_path,
|
|
G_KEY_FILE_NONE, &error)) {
|
|
seaf_warning ("Failed to load config file.\n");
|
|
g_key_file_free (config);
|
|
goto onerror;
|
|
}
|
|
g_free (config_file_path);
|
|
|
|
session = g_new0(SeafileSession, 1);
|
|
session->seaf_dir = abs_seafile_dir;
|
|
session->tmp_file_dir = tmp_file_dir;
|
|
session->session = ccnet_session;
|
|
session->config = config;
|
|
|
|
if (load_database_config (session) < 0) {
|
|
seaf_warning ("Failed to load database config.\n");
|
|
goto onerror;
|
|
}
|
|
|
|
session->cfg_mgr = seaf_cfg_manager_new (session);
|
|
if (!session->cfg_mgr)
|
|
goto onerror;
|
|
|
|
if (load_thread_pool_config (session) < 0) {
|
|
seaf_warning ("Failed to load thread pool config.\n");
|
|
goto onerror;
|
|
}
|
|
|
|
session->fs_mgr = seaf_fs_manager_new (session, abs_seafile_dir);
|
|
if (!session->fs_mgr)
|
|
goto onerror;
|
|
session->block_mgr = seaf_block_manager_new (session, abs_seafile_dir);
|
|
if (!session->block_mgr)
|
|
goto onerror;
|
|
session->commit_mgr = seaf_commit_manager_new (session);
|
|
if (!session->commit_mgr)
|
|
goto onerror;
|
|
session->repo_mgr = seaf_repo_manager_new (session);
|
|
if (!session->repo_mgr)
|
|
goto onerror;
|
|
session->branch_mgr = seaf_branch_manager_new (session);
|
|
if (!session->branch_mgr)
|
|
goto onerror;
|
|
|
|
session->cs_mgr = seaf_cs_manager_new (session);
|
|
if (!session->cs_mgr)
|
|
goto onerror;
|
|
|
|
session->share_mgr = seaf_share_manager_new (session);
|
|
if (!session->share_mgr)
|
|
goto onerror;
|
|
|
|
session->web_at_mgr = seaf_web_at_manager_new (session);
|
|
if (!session->web_at_mgr)
|
|
goto onerror;
|
|
|
|
session->token_mgr = seaf_token_manager_new (session);
|
|
if (!session->token_mgr)
|
|
goto onerror;
|
|
|
|
session->passwd_mgr = seaf_passwd_manager_new (session);
|
|
if (!session->passwd_mgr)
|
|
goto onerror;
|
|
|
|
session->quota_mgr = seaf_quota_manager_new (session);
|
|
if (!session->quota_mgr)
|
|
goto onerror;
|
|
|
|
session->listen_mgr = seaf_listen_manager_new (session);
|
|
if (!session->listen_mgr)
|
|
goto onerror;
|
|
|
|
session->copy_mgr = seaf_copy_manager_new (session);
|
|
if (!session->copy_mgr)
|
|
goto onerror;
|
|
|
|
session->job_mgr = ccnet_job_manager_new (session->sync_thread_pool_size);
|
|
ccnet_session->job_mgr = ccnet_job_manager_new (session->rpc_thread_pool_size);
|
|
|
|
session->size_sched = size_scheduler_new (session);
|
|
|
|
session->ev_mgr = cevent_manager_new ();
|
|
if (!session->ev_mgr)
|
|
goto onerror;
|
|
|
|
session->mq_mgr = seaf_mq_manager_new (session);
|
|
if (!session->mq_mgr)
|
|
goto onerror;
|
|
|
|
session->http_server = seaf_http_server_new (session);
|
|
if (!session->http_server)
|
|
goto onerror;
|
|
|
|
session->zip_download_mgr = zip_download_mgr_new ();
|
|
if (!session->zip_download_mgr)
|
|
goto onerror;
|
|
|
|
session->index_blocks_mgr = index_blocks_mgr_new (session);
|
|
if (!session->index_blocks_mgr)
|
|
goto onerror;
|
|
|
|
return session;
|
|
|
|
onerror:
|
|
free (abs_seafile_dir);
|
|
g_free (tmp_file_dir);
|
|
g_free (config_file_path);
|
|
g_free (session);
|
|
return NULL;
|
|
}
|
|
|
|
int
|
|
seafile_session_init (SeafileSession *session)
|
|
{
|
|
if (seaf_commit_manager_init (session->commit_mgr) < 0)
|
|
return -1;
|
|
|
|
if (seaf_fs_manager_init (session->fs_mgr) < 0)
|
|
return -1;
|
|
|
|
if (seaf_branch_manager_init (session->branch_mgr) < 0) {
|
|
seaf_warning ("Failed to init branch manager.\n");
|
|
return -1;
|
|
}
|
|
|
|
if (seaf_repo_manager_init (session->repo_mgr) < 0) {
|
|
seaf_warning ("Failed to init repo manager.\n");
|
|
return -1;
|
|
}
|
|
|
|
if (seaf_quota_manager_init (session->quota_mgr) < 0) {
|
|
seaf_warning ("Failed to init quota manager.\n");
|
|
return -1;
|
|
}
|
|
|
|
if ((session->create_tables || seaf_db_type(session->db) != SEAF_DB_TYPE_MYSQL)
|
|
&& seaf_cfg_manager_init (session->cfg_mgr) < 0) {
|
|
seaf_warning ("Failed to init config manager.\n");
|
|
return -1;
|
|
}
|
|
|
|
seaf_mq_manager_init (session->mq_mgr);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
seafile_session_start (SeafileSession *session)
|
|
{
|
|
if (cevent_manager_start (session->ev_mgr) < 0) {
|
|
seaf_warning ("Failed to start event manager.\n");
|
|
return -1;
|
|
}
|
|
|
|
if (seaf_cs_manager_start (session->cs_mgr) < 0) {
|
|
seaf_warning ("Failed to start chunk server manager.\n");
|
|
return -1;
|
|
}
|
|
|
|
if (seaf_share_manager_start (session->share_mgr) < 0) {
|
|
seaf_warning ("Failed to start share manager.\n");
|
|
return -1;
|
|
}
|
|
|
|
if (seaf_web_at_manager_start (session->web_at_mgr) < 0) {
|
|
seaf_warning ("Failed to start web access check manager.\n");
|
|
return -1;
|
|
}
|
|
|
|
if (seaf_passwd_manager_start (session->passwd_mgr) < 0) {
|
|
seaf_warning ("Failed to start password manager.\n");
|
|
return -1;
|
|
}
|
|
|
|
if (seaf_mq_manager_start (session->mq_mgr) < 0) {
|
|
seaf_warning ("Failed to start mq manager.\n");
|
|
return -1;
|
|
}
|
|
|
|
if (seaf_listen_manager_start (session->listen_mgr) < 0) {
|
|
seaf_warning ("Failed to start listen manager.\n");
|
|
return -1;
|
|
}
|
|
|
|
if (size_scheduler_start (session->size_sched) < 0) {
|
|
seaf_warning ("Failed to start size scheduler.\n");
|
|
return -1;
|
|
}
|
|
|
|
if (seaf_copy_manager_start (session->copy_mgr) < 0) {
|
|
seaf_warning ("Failed to start copy manager.\n");
|
|
return -1;
|
|
}
|
|
|
|
if (seaf_http_server_start (session->http_server) < 0) {
|
|
seaf_warning ("Failed to start http server thread.\n");
|
|
return -1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int
|
|
load_thread_pool_config (SeafileSession *session)
|
|
{
|
|
int rpc_tp_size, sync_tp_size;
|
|
|
|
rpc_tp_size = g_key_file_get_integer (session->config,
|
|
"thread pool size", "rpc",
|
|
NULL);
|
|
sync_tp_size = g_key_file_get_integer (session->config,
|
|
"thread pool size", "sync",
|
|
NULL);
|
|
|
|
if (rpc_tp_size > 0)
|
|
session->rpc_thread_pool_size = rpc_tp_size;
|
|
else
|
|
session->rpc_thread_pool_size = DEFAULT_RPC_THREAD_POOL_SIZE;
|
|
|
|
if (sync_tp_size > 0)
|
|
session->sync_thread_pool_size = sync_tp_size;
|
|
else
|
|
session->sync_thread_pool_size = DEFAULT_THREAD_POOL_SIZE;
|
|
|
|
return 0;
|
|
}
|
|
|
|
char *
|
|
get_system_default_repo_id (SeafileSession *session)
|
|
{
|
|
char *sql = "SELECT info_value FROM SystemInfo WHERE info_key='default_repo_id'";
|
|
return seaf_db_get_string (session->db, sql);
|
|
}
|
|
|
|
int
|
|
set_system_default_repo_id (SeafileSession *session, const char *repo_id)
|
|
{
|
|
char sql[256];
|
|
snprintf (sql, sizeof(sql),
|
|
"INSERT INTO SystemInfo (info_key, info_value) VALUES ('default_repo_id', '%s')",
|
|
repo_id);
|
|
return seaf_db_query (session->db, sql);
|
|
}
|
|
|
|
static int
|
|
del_system_default_repo_id (SeafileSession *session)
|
|
{
|
|
const char *sql = "DELETE FROM SystemInfo WHERE info_key='default_repo_id'";
|
|
return seaf_db_query (session->db, sql);
|
|
}
|
|
|
|
#define DEFAULT_TEMPLATE_DIR "library-template"
|
|
|
|
static void
|
|
copy_template_files_recursive (SeafileSession *session,
|
|
const char *repo_id,
|
|
const char *repo_dir_path,
|
|
const char *dir_path)
|
|
{
|
|
GDir *dir;
|
|
const char *name;
|
|
char *sub_path, *repo_sub_path;
|
|
SeafStat st;
|
|
GError *error = NULL;
|
|
int rc;
|
|
|
|
dir = g_dir_open (dir_path, 0, &error);
|
|
if (!dir) {
|
|
seaf_warning ("Failed to open template dir %s: %s.\n",
|
|
dir_path, error->message);
|
|
return;
|
|
}
|
|
|
|
while ((name = g_dir_read_name(dir)) != NULL) {
|
|
sub_path = g_build_filename (dir_path, name, NULL);
|
|
if (seaf_stat (sub_path, &st) < 0) {
|
|
seaf_warning ("Failed to stat %s: %s.\n", sub_path, strerror(errno));
|
|
g_free (sub_path);
|
|
continue;
|
|
}
|
|
|
|
if (S_ISREG(st.st_mode)) {
|
|
rc = seaf_repo_manager_post_file (session->repo_mgr,
|
|
repo_id,
|
|
sub_path,
|
|
repo_dir_path,
|
|
name,
|
|
"System",
|
|
NULL);
|
|
if (rc < 0)
|
|
seaf_warning ("Failed to add template file %s.\n", sub_path);
|
|
} else if (S_ISDIR(st.st_mode)) {
|
|
rc = seaf_repo_manager_post_dir (session->repo_mgr,
|
|
repo_id,
|
|
repo_dir_path,
|
|
name,
|
|
"System",
|
|
NULL);
|
|
if (rc < 0) {
|
|
seaf_warning ("Failed to add template dir %s.\n", sub_path);
|
|
g_free (sub_path);
|
|
continue;
|
|
}
|
|
|
|
repo_sub_path = g_build_path ("/", repo_dir_path, name, NULL);
|
|
copy_template_files_recursive (session, repo_id,
|
|
repo_sub_path, sub_path);
|
|
g_free (repo_sub_path);
|
|
}
|
|
g_free (sub_path);
|
|
}
|
|
g_dir_close (dir);
|
|
}
|
|
|
|
static void *
|
|
create_system_default_repo (void *data)
|
|
{
|
|
SeafileSession *session = data;
|
|
char *repo_id;
|
|
char *template_path;
|
|
|
|
/* If default repo is not set or doesn't exist, create a new one. */
|
|
repo_id = get_system_default_repo_id (session);
|
|
if (repo_id != NULL) {
|
|
SeafRepo *repo;
|
|
repo = seaf_repo_manager_get_repo (session->repo_mgr, repo_id);
|
|
if (!repo) {
|
|
seaf_warning ("Failed to get system default repo. Create a new one.\n");
|
|
del_system_default_repo_id (session);
|
|
seaf_repo_manager_del_repo (session->repo_mgr, repo_id, NULL);
|
|
g_free (repo_id);
|
|
} else {
|
|
seaf_repo_unref (repo);
|
|
g_free (repo_id);
|
|
return data;
|
|
}
|
|
}
|
|
|
|
repo_id = seaf_repo_manager_create_new_repo (session->repo_mgr,
|
|
"My Library Template",
|
|
"Template for creating 'My Library' for users",
|
|
"System",
|
|
NULL, NULL);
|
|
if (!repo_id) {
|
|
seaf_warning ("Failed to create system default repo.\n");
|
|
return data;
|
|
}
|
|
|
|
set_system_default_repo_id (session, repo_id);
|
|
|
|
template_path = g_build_filename (session->seaf_dir, DEFAULT_TEMPLATE_DIR, NULL);
|
|
copy_template_files_recursive (session, repo_id, "/", template_path);
|
|
|
|
g_free (repo_id);
|
|
g_free (template_path);
|
|
return data;
|
|
}
|
|
|
|
void
|
|
schedule_create_system_default_repo (SeafileSession *session)
|
|
{
|
|
int db_type = seaf_db_type (session->db);
|
|
char *sql;
|
|
|
|
if (db_type == SEAF_DB_TYPE_MYSQL)
|
|
sql = "CREATE TABLE IF NOT EXISTS SystemInfo "
|
|
"(id BIGINT NOT NULL PRIMARY KEY AUTO_INCREMENT, "
|
|
"info_key VARCHAR(256), info_value VARCHAR(1024))";
|
|
else
|
|
sql = "CREATE TABLE IF NOT EXISTS SystemInfo( "
|
|
"info_key VARCHAR(256), info_value VARCHAR(1024))";
|
|
|
|
if ((session->create_tables || db_type != SEAF_DB_TYPE_MYSQL)
|
|
&& seaf_db_query (session->db, sql) < 0)
|
|
return;
|
|
|
|
ccnet_job_manager_schedule_job (session->job_mgr,
|
|
create_system_default_repo,
|
|
NULL, session);
|
|
}
|