2022-09-19 20:41:30 +00:00
|
|
|
/* snac - A simple, minimalistic ActivityPub instance */
|
|
|
|
/* copyright (c) 2022 grunfink - MIT license */
|
|
|
|
|
|
|
|
#include "xs.h"
|
|
|
|
#include "xs_io.h"
|
|
|
|
#include "xs_json.h"
|
2022-09-20 07:39:28 +00:00
|
|
|
#include "xs_openssl.h"
|
2022-09-19 20:41:30 +00:00
|
|
|
|
|
|
|
#include "snac.h"
|
|
|
|
|
2022-09-19 21:33:11 +00:00
|
|
|
#include <glob.h>
|
|
|
|
|
|
|
|
|
2022-09-19 20:41:30 +00:00
|
|
|
int srv_open(char *basedir)
|
|
|
|
/* opens a server */
|
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
xs *cfg_file = NULL;
|
|
|
|
FILE *f;
|
2022-09-20 07:39:28 +00:00
|
|
|
d_char *error = NULL;
|
2022-09-19 20:41:30 +00:00
|
|
|
|
|
|
|
srv_basedir = xs_str_new(basedir);
|
|
|
|
|
2022-09-20 07:39:28 +00:00
|
|
|
if (xs_endswith(srv_basedir, "/"))
|
|
|
|
srv_basedir = xs_crop(srv_basedir, 0, -1);
|
|
|
|
|
2022-09-19 20:41:30 +00:00
|
|
|
cfg_file = xs_fmt("%s/server.json", basedir);
|
|
|
|
|
|
|
|
if ((f = fopen(cfg_file, "r")) == NULL)
|
2022-09-20 07:39:28 +00:00
|
|
|
error = xs_fmt("error opening '%s'", cfg_file);
|
2022-09-19 20:41:30 +00:00
|
|
|
else {
|
|
|
|
xs *cfg_data;
|
|
|
|
|
|
|
|
/* read full config file */
|
|
|
|
cfg_data = xs_readall(f);
|
|
|
|
|
|
|
|
/* parse */
|
|
|
|
srv_config = xs_json_loads(cfg_data);
|
|
|
|
|
|
|
|
if (srv_config == NULL)
|
2022-09-20 07:39:28 +00:00
|
|
|
error = xs_fmt("cannot parse '%s'", cfg_file);
|
2022-09-19 20:41:30 +00:00
|
|
|
else {
|
|
|
|
char *host;
|
|
|
|
char *prefix;
|
|
|
|
char *dbglvl;
|
|
|
|
|
|
|
|
host = xs_dict_get(srv_config, "host");
|
|
|
|
prefix = xs_dict_get(srv_config, "prefix");
|
|
|
|
dbglvl = xs_dict_get(srv_config, "dbglevel");
|
|
|
|
|
|
|
|
if (host == NULL || prefix == NULL)
|
2022-09-20 07:39:28 +00:00
|
|
|
error = xs_str_new("cannot get server data");
|
2022-09-19 20:41:30 +00:00
|
|
|
else {
|
|
|
|
srv_baseurl = xs_fmt("https://%s%s", host, prefix);
|
|
|
|
|
|
|
|
dbglevel = (int) xs_number_get(dbglvl);
|
|
|
|
|
|
|
|
if ((dbglvl = getenv("DEBUG")) != NULL) {
|
|
|
|
dbglevel = atoi(dbglvl);
|
2022-09-20 07:39:28 +00:00
|
|
|
error = xs_fmt("DEBUG level set to %d from environment", dbglevel);
|
2022-09-19 20:41:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
ret = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-09-20 07:39:28 +00:00
|
|
|
if (ret == 0 && error != NULL)
|
|
|
|
srv_log(error);
|
|
|
|
|
2022-09-19 20:41:30 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-09-19 21:33:11 +00:00
|
|
|
void user_free(snac *snac)
|
2022-09-19 20:41:30 +00:00
|
|
|
/* frees a user snac */
|
|
|
|
{
|
|
|
|
free(snac->uid);
|
|
|
|
free(snac->basedir);
|
|
|
|
free(snac->config);
|
|
|
|
free(snac->key);
|
|
|
|
free(snac->actor);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-09-19 21:03:18 +00:00
|
|
|
int user_open(snac *snac, char *uid)
|
2022-09-19 20:41:30 +00:00
|
|
|
/* opens a user */
|
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
memset(snac, '\0', sizeof(struct _snac));
|
|
|
|
|
|
|
|
if (validate_uid(uid)) {
|
|
|
|
xs *cfg_file;
|
|
|
|
FILE *f;
|
|
|
|
|
|
|
|
snac->uid = xs_str_new(uid);
|
|
|
|
|
|
|
|
snac->basedir = xs_fmt("%s/user/%s", srv_basedir, uid);
|
|
|
|
|
|
|
|
cfg_file = xs_fmt("%s/user.json", snac->basedir);
|
|
|
|
|
|
|
|
if ((f = fopen(cfg_file, "r")) != NULL) {
|
|
|
|
xs *cfg_data;
|
|
|
|
|
|
|
|
/* read full config file */
|
|
|
|
cfg_data = xs_readall(f);
|
|
|
|
fclose(f);
|
|
|
|
|
|
|
|
if ((snac->config = xs_json_loads(cfg_data)) != NULL) {
|
|
|
|
xs *key_file = xs_fmt("%s/key.json", snac->basedir);
|
|
|
|
|
|
|
|
if ((f = fopen(key_file, "r")) != NULL) {
|
|
|
|
xs *key_data;
|
|
|
|
|
|
|
|
key_data = xs_readall(f);
|
|
|
|
fclose(f);
|
|
|
|
|
|
|
|
if ((snac->key = xs_json_loads(key_data)) != NULL) {
|
|
|
|
snac->actor = xs_fmt("%s/%s", srv_baseurl, uid);
|
|
|
|
ret = 1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
srv_log(xs_fmt("cannot parse '%s'", key_file));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
srv_log(xs_fmt("error opening '%s'", key_file));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
srv_log(xs_fmt("cannot parse '%s'", cfg_file));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
srv_log(xs_fmt("error opening '%s'", cfg_file));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
srv_log(xs_fmt("invalid user '%s'", uid));
|
|
|
|
|
|
|
|
if (!ret)
|
2022-09-19 21:33:11 +00:00
|
|
|
user_free(snac);
|
2022-09-19 20:41:30 +00:00
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-09-19 21:33:11 +00:00
|
|
|
d_char *user_list(void)
|
|
|
|
/* returns the list of user ids */
|
|
|
|
{
|
|
|
|
d_char *list;
|
|
|
|
xs *spec;
|
|
|
|
glob_t globbuf;
|
|
|
|
|
|
|
|
globbuf.gl_offs = 1;
|
|
|
|
|
|
|
|
list = xs_list_new();
|
2022-09-20 07:39:28 +00:00
|
|
|
spec = xs_fmt("%s/user/" "*", srv_basedir);
|
2022-09-19 21:33:11 +00:00
|
|
|
|
|
|
|
if (glob(spec, 0, NULL, &globbuf) == 0) {
|
|
|
|
int n;
|
|
|
|
char *p;
|
|
|
|
|
|
|
|
for (n = 0; (p = globbuf.gl_pathv[n]) != NULL; n++) {
|
|
|
|
if ((p = strrchr(p, '/')) != NULL)
|
|
|
|
list = xs_list_append(list, p + 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
globfree(&globbuf);
|
|
|
|
|
|
|
|
return list;
|
|
|
|
}
|
|
|
|
|
2022-09-20 07:39:28 +00:00
|
|
|
|
|
|
|
float mtime(char *fn)
|
|
|
|
/* returns the mtime of a file or directory, or 0.0 */
|
|
|
|
{
|
|
|
|
struct stat st;
|
|
|
|
float r = 0.0;
|
|
|
|
|
|
|
|
if (stat(fn, &st) != -1)
|
|
|
|
r = (float)st.st_mtim.tv_sec;
|
|
|
|
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
d_char *_follower_fn(snac *snac, char *actor)
|
|
|
|
{
|
|
|
|
xs *md5 = xs_md5_hex(actor, strlen(actor));
|
|
|
|
return xs_fmt("%s/followers/%s.json", snac->basedir, md5);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int follower_add(snac *snac, char *actor, char *msg)
|
|
|
|
/* adds a follower */
|
|
|
|
{
|
|
|
|
int ret = 201; /* created */
|
|
|
|
xs *fn = _follower_fn(snac, actor);
|
|
|
|
FILE *f;
|
|
|
|
|
|
|
|
if ((f = fopen(fn, "w")) != NULL) {
|
|
|
|
xs *j = xs_json_dumps_pp(msg, 4);
|
|
|
|
|
|
|
|
fwrite(j, 1, strlen(j), f);
|
|
|
|
fclose(f);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
ret = 500;
|
|
|
|
|
|
|
|
snac_debug(snac, 2, xs_fmt("follower_add %s %s", actor, fn));
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int follower_del(snac *snac, char *actor)
|
|
|
|
/* deletes a follower */
|
|
|
|
{
|
|
|
|
xs *fn = _follower_fn(snac, actor);
|
|
|
|
|
|
|
|
unlink(fn);
|
|
|
|
|
|
|
|
snac_debug(snac, 2, xs_fmt("follower_del %s %s", actor, fn));
|
|
|
|
|
|
|
|
return 200;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int follower_check(snac *snac, char *actor)
|
|
|
|
/* checks if someone is a follower */
|
|
|
|
{
|
|
|
|
xs *fn = _follower_fn(snac, actor);
|
|
|
|
|
|
|
|
return !!(mtime(fn) != 0.0);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
d_char *follower_list(snac *snac)
|
|
|
|
/* returns the list of followers */
|
|
|
|
{
|
|
|
|
d_char *list;
|
|
|
|
xs *spec;
|
|
|
|
glob_t globbuf;
|
|
|
|
|
|
|
|
list = xs_list_new();
|
|
|
|
spec = xs_fmt("%s/followers/" "*.json", snac->basedir);
|
|
|
|
|
|
|
|
if (glob(spec, 0, NULL, &globbuf) == 0) {
|
|
|
|
int n;
|
2022-09-20 08:49:24 +00:00
|
|
|
char *fn;
|
2022-09-20 07:39:28 +00:00
|
|
|
|
2022-09-20 08:49:24 +00:00
|
|
|
for (n = 0; (fn = globbuf.gl_pathv[n]) != NULL; n++) {
|
2022-09-20 07:39:28 +00:00
|
|
|
FILE *f;
|
|
|
|
|
2022-09-20 08:49:24 +00:00
|
|
|
if ((f = fopen(fn, "r")) != NULL) {
|
2022-09-20 07:48:13 +00:00
|
|
|
xs *j = xs_readall(f);
|
|
|
|
xs *o = xs_json_loads(j);
|
|
|
|
|
|
|
|
if (o != NULL)
|
|
|
|
list = xs_list_append(list, o);
|
|
|
|
|
|
|
|
fclose(f);
|
2022-09-20 07:39:28 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
globfree(&globbuf);
|
|
|
|
|
|
|
|
return list;
|
|
|
|
}
|
2022-09-20 08:02:00 +00:00
|
|
|
|
|
|
|
|
2022-09-20 08:49:24 +00:00
|
|
|
d_char *_timeline_find_fn(snac *snac, char *id)
|
2022-09-20 08:02:00 +00:00
|
|
|
/* returns the file name of a timeline entry by its id */
|
|
|
|
{
|
|
|
|
xs *md5 = xs_md5_hex(id, strlen(id));
|
|
|
|
xs *spec = xs_fmt("%s/timeline/" "*-%s.json", snac->basedir, md5);
|
|
|
|
glob_t globbuf;
|
|
|
|
d_char *fn = NULL;
|
|
|
|
|
|
|
|
if (glob(spec, 0, NULL, &globbuf) == 0 && globbuf.gl_matchc) {
|
|
|
|
/* get just the first file */
|
|
|
|
fn = xs_str_new(globbuf.gl_pathv[0]);
|
|
|
|
}
|
|
|
|
|
|
|
|
globfree(&globbuf);
|
|
|
|
|
|
|
|
return fn;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-09-20 08:49:24 +00:00
|
|
|
d_char *timeline_find(snac *snac, char *id)
|
|
|
|
/* gets a message from the timeline by id */
|
2022-09-20 08:02:00 +00:00
|
|
|
{
|
2022-09-20 08:49:24 +00:00
|
|
|
xs *fn = _timeline_find_fn(snac, id);
|
2022-09-20 08:02:00 +00:00
|
|
|
xs *msg = NULL;
|
|
|
|
|
|
|
|
if (fn != NULL) {
|
|
|
|
FILE *f;
|
|
|
|
|
|
|
|
if ((f = fopen(fn, "r")) != NULL) {
|
|
|
|
xs *j = xs_readall(f);
|
|
|
|
|
|
|
|
msg = xs_json_loads(j);
|
|
|
|
fclose(f);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return msg;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void timeline_del(snac *snac, char *id)
|
|
|
|
/* deletes a message from the timeline */
|
|
|
|
{
|
2022-09-20 08:49:24 +00:00
|
|
|
xs *fn = _timeline_find_fn(snac, id);
|
2022-09-20 08:02:00 +00:00
|
|
|
|
|
|
|
if (fn != NULL) {
|
|
|
|
xs *lfn = NULL;
|
|
|
|
|
|
|
|
unlink(fn);
|
|
|
|
snac_debug(snac, 1, xs_fmt("timeline_del %s", id));
|
|
|
|
|
|
|
|
/* try to delete also from the local timeline */
|
|
|
|
lfn = xs_replace(fn, "/timeline/", "/local/");
|
|
|
|
|
|
|
|
if (unlink(lfn) != -1)
|
|
|
|
snac_debug(snac, 1, xs_fmt("timeline_del (local) %s", id));
|
|
|
|
}
|
|
|
|
}
|
2022-09-20 08:49:24 +00:00
|
|
|
|
|
|
|
|
|
|
|
d_char *timeline_get(snac *snac, char *fn)
|
|
|
|
/* gets a timeline entry by file name */
|
|
|
|
{
|
|
|
|
d_char *d = NULL;
|
|
|
|
FILE *f;
|
|
|
|
|
|
|
|
if ((f = fopen(fn, "r")) != NULL) {
|
|
|
|
xs *j = xs_readall(f);
|
|
|
|
|
|
|
|
d = xs_json_loads(j);
|
|
|
|
fclose(f);
|
|
|
|
}
|
|
|
|
|
|
|
|
return d;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
d_char *timeline_list(snac *snac)
|
|
|
|
/* returns a list of the timeline filenames */
|
|
|
|
{
|
|
|
|
d_char *list;
|
|
|
|
xs *spec = xs_fmt("%s/timeline/" "*.json", snac->basedir);
|
|
|
|
glob_t globbuf;
|
|
|
|
int max;
|
|
|
|
|
|
|
|
/* maximum number of items in the timeline */
|
|
|
|
max = xs_number_get(xs_dict_get(srv_config, "max_timeline_entries"));
|
|
|
|
|
|
|
|
list = xs_list_new();
|
|
|
|
|
|
|
|
/* get the list in reverse order */
|
|
|
|
if (glob(spec, 0, NULL, &globbuf) == 0) {
|
|
|
|
int n;
|
|
|
|
|
|
|
|
if (max > globbuf.gl_matchc)
|
|
|
|
max = globbuf.gl_matchc;
|
|
|
|
|
|
|
|
for (n = 0; n < max; n++) {
|
|
|
|
char *fn = globbuf.gl_pathv[globbuf.gl_matchc - n - 1];
|
|
|
|
|
|
|
|
list = xs_list_append(list, fn);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
globfree(&globbuf);
|
|
|
|
|
|
|
|
return list;
|
|
|
|
}
|