mirror of
https://codeberg.org/grunfink/snac2.git
synced 2024-12-26 01:03:37 +00:00
Merge pull request 'master' (#1) from grunfink/snac2:master into master
Reviewed-on: https://codeberg.org/louis77/snac2/pulls/1
This commit is contained in:
commit
84a767dd08
30 changed files with 2079 additions and 998 deletions
2
Makefile
2
Makefile
|
@ -36,7 +36,7 @@ uninstall:
|
|||
activitypub.o: activitypub.c xs.h xs_json.h xs_curl.h xs_mime.h \
|
||||
xs_openssl.h xs_regex.h xs_time.h xs_set.h xs_match.h snac.h
|
||||
data.o: data.c xs.h xs_hex.h xs_io.h xs_json.h xs_openssl.h xs_glob.h \
|
||||
xs_set.h xs_time.h xs_regex.h snac.h
|
||||
xs_set.h xs_time.h xs_regex.h xs_match.h snac.h
|
||||
format.o: format.c xs.h xs_regex.h xs_mime.h xs_html.h xs_json.h \
|
||||
xs_time.h snac.h
|
||||
html.o: html.c xs.h xs_io.h xs_json.h xs_regex.h xs_set.h xs_openssl.h \
|
||||
|
|
|
@ -38,7 +38,7 @@ uninstall:
|
|||
activitypub.o: activitypub.c xs.h xs_json.h xs_curl.h xs_mime.h \
|
||||
xs_openssl.h xs_regex.h xs_time.h xs_set.h xs_match.h snac.h
|
||||
data.o: data.c xs.h xs_hex.h xs_io.h xs_json.h xs_openssl.h xs_glob.h \
|
||||
xs_set.h xs_time.h xs_regex.h snac.h
|
||||
xs_set.h xs_time.h xs_regex.h xs_match.h snac.h
|
||||
format.o: format.c xs.h xs_regex.h xs_mime.h xs_html.h xs_json.h \
|
||||
xs_time.h snac.h
|
||||
html.o: html.c xs.h xs_io.h xs_json.h xs_regex.h xs_set.h xs_openssl.h \
|
||||
|
|
|
@ -58,7 +58,6 @@ Run `make` and then `make install` as root.
|
|||
|
||||
If you're compiling on NetBSD, you should use the specific provided Makefile and run `make -f Makefile.NetBSD` and then `make -f Makefile.NetBSD install` as root.
|
||||
|
||||
|
||||
From version 2.27, `snac` includes support for the Mastodon API; if you are not interested on it, you can compile it out by running
|
||||
|
||||
```sh
|
||||
|
@ -71,6 +70,12 @@ If your compilation process complains about undefined references to `shm_open()`
|
|||
make LDFLAGS=-lrt
|
||||
```
|
||||
|
||||
If it still gives compilation errors (because your system does not implement the shared memory functions), you can fix it with
|
||||
|
||||
```sh
|
||||
make CFLAGS=-DWITHOUT_SHM
|
||||
```
|
||||
|
||||
See the administrator manual on how to proceed from here.
|
||||
|
||||
## Testing via Docker
|
||||
|
|
|
@ -1,9 +1,29 @@
|
|||
# Release Notes
|
||||
|
||||
## 2.53
|
||||
|
||||
New user feature to search by post content (using regular expressions) or tag.
|
||||
|
||||
Added some (partial) support for `Event` object types.
|
||||
|
||||
Minor fixes: Allow unboosting your own posts (contributed by khm), CSS fixes for the Dillo browser (contributed by kvibber).
|
||||
|
||||
## 2.52
|
||||
|
||||
Posts that were liked or boosted can now be unliked and unboosted.
|
||||
|
||||
Outgoing message timeouts are no longer hardcoded and can be configured (see `snac(8)` for more information).
|
||||
|
||||
Fixed a bug that caused some incorrect unfollows under special conditions (with shared inboxes enabled and users from the same instance that follow each other, the internal message distributor was confused).
|
||||
|
||||
Mastodon API: Added support for lists.
|
||||
|
||||
Added a header to avoid over-zealous caching in some browsers (contributed by louis77).
|
||||
|
||||
Added support for running and federating inside hidden networks like Tor, I2P or Loki (contributed by iwojima).
|
||||
|
||||
Fixed an error processing polls coming from Pleroma instances.
|
||||
|
||||
## 2.51
|
||||
|
||||
Support for custom Emojis has been added; they are no longer hardcoded, but read from the `emojis.json` file at the server base directory. Also, they are no longer limited to string substitutions, but images as external URLs are also supported (see `snac(8)` for more information).
|
||||
|
|
24
TODO.md
24
TODO.md
|
@ -10,36 +10,34 @@ Mastodon API: fix whatever the fuck is making the official app and Megalodon to
|
|||
|
||||
Important: deleting a follower should do more that just delete the object, see https://codeberg.org/grunfink/snac2/issues/43#issuecomment-956721
|
||||
|
||||
Editing / Updating a post does not index newly added hashtags.
|
||||
|
||||
## Wishlist
|
||||
|
||||
Implement `Group`-like accounts (i.e. an actor that boosts to their followers all posts that mention it).
|
||||
Track 'Event' data types standardization; how to add plan-to-attend and similar activities (more info: https://event-federation.eu/)
|
||||
|
||||
Integrate "Ability to federate with hidden networks" see https://codeberg.org/grunfink/snac2/issues/93
|
||||
Implement "FEP-3b86: Activity Intents" https://codeberg.org/fediverse/fep/src/branch/main/fep/3b86/fep-3b86.md
|
||||
|
||||
Track "FEP-ef61: Portable Objects" https://codeberg.org/fediverse/fep/src/branch/main/fep/ef61/fep-ef61.md
|
||||
|
||||
Implement `Group`-like accounts (i.e. an actor that boosts to their followers all posts that mention it).
|
||||
|
||||
Integrate "Added handling for International Domain Names" PR https://codeberg.org/grunfink/snac2/pulls/104
|
||||
|
||||
Consider adding Mastodon import functionality (for following_accounts.csv and outbox.json).
|
||||
|
||||
Consider adding milter-like support to reject posts to mitigate spam.
|
||||
|
||||
Do something about Akkoma and Misskey's quoted replies (they use the `quoteUrl` field instead of `inReplyTo`).
|
||||
|
||||
Add more CSS classes according to https://comam.es/snac/grunfink/p/1705598619.090050
|
||||
|
||||
Add support for /share?text=tt&website=url (whatever it is, see https://mastodonshare.com/ for details).
|
||||
|
||||
Add support for /authorize_interaction (whatever it is).
|
||||
|
||||
Add a list of hashtags to drop.
|
||||
|
||||
Add domain/subdomain flexibility according to https://codeberg.org/grunfink/snac2/issues/3
|
||||
|
||||
The 'history' pages are just monthly HTML snapshots of the local timeline. This is ok and cheap and easy, but is problematic if you e.g. intentionally delete a post because it will remain there in the history forever. If you activate local timeline purging, purged entries will remain in the history as 'ghosts', which may or may not be what the user wants.
|
||||
|
||||
Implement bulleted lists. Mastodon is crap and won't show them, but other implementations (Friendica, Pleroma) will do.
|
||||
|
||||
User request: "will it be possible to click on a link and instead of opening the original instance, we'll be able only to see a list of the posts of this person here in comam?. Something like Mastodon does."
|
||||
|
||||
The actual storage system wastes too much disk space (lots of small files that really consume 4k of storage). Consider alternatives.
|
||||
|
||||
## Closed
|
||||
|
@ -311,3 +309,9 @@ Consider implementing the rejection of activities from recently-created accounts
|
|||
Consider discarding posts by content using string or regex to mitigate spam (2024-03-14T10:40:14+0100).
|
||||
|
||||
Post edits should preserve the image and the image description somewhat (2024-03-22T09:57:18+0100).
|
||||
|
||||
Integrate "Ability to federate with hidden networks" see https://codeberg.org/grunfink/snac2/issues/93
|
||||
|
||||
Consider adding milter-like support to reject posts to mitigate spam (discarded; 2024-04-20T22:46:35+0200).
|
||||
|
||||
Implement support for 'Event' data types. Example: https://fediversity.site/item/e9bdb383-eeb9-4d7d-b2f7-c6401267cae0 (2024-05-12T08:56:27+0200)
|
||||
|
|
424
activitypub.c
424
activitypub.c
|
@ -67,7 +67,7 @@ int activitypub_request(snac *user, const char *url, xs_dict **data)
|
|||
xs *response = NULL;
|
||||
xs *payload = NULL;
|
||||
int p_size;
|
||||
char *ctype;
|
||||
const char *ctype;
|
||||
|
||||
*data = NULL;
|
||||
|
||||
|
@ -154,20 +154,21 @@ int actor_request(snac *user, const char *actor, xs_dict **data)
|
|||
}
|
||||
|
||||
|
||||
char *get_atto(const xs_dict *msg)
|
||||
const char *get_atto(const xs_dict *msg)
|
||||
/* gets the attributedTo field (an actor) */
|
||||
{
|
||||
char *actor = xs_dict_get(msg, "attributedTo");
|
||||
const xs_val *actor = xs_dict_get(msg, "attributedTo");
|
||||
|
||||
/* if the actor is a list of objects (like on Peertube videos), pick the Person */
|
||||
if (xs_type(actor) == XSTYPE_LIST) {
|
||||
xs_list *p = actor;
|
||||
xs_dict *v;
|
||||
const xs_list *p = actor;
|
||||
int c = 0;
|
||||
const xs_dict *v;
|
||||
actor = NULL;
|
||||
|
||||
while (actor == NULL && xs_list_iter(&p, &v)) {
|
||||
while (actor == NULL && xs_list_next(p, &v, &c)) {
|
||||
if (xs_type(v) == XSTYPE_DICT) {
|
||||
char *type = xs_dict_get(v, "type");
|
||||
const char *type = xs_dict_get(v, "type");
|
||||
if (xs_type(type) == XSTYPE_STRING && strcmp(type, "Person") == 0) {
|
||||
actor = xs_dict_get(v, "id");
|
||||
|
||||
|
@ -186,12 +187,12 @@ xs_list *get_attachments(const xs_dict *msg)
|
|||
/* unify the garbage fire that are the attachments */
|
||||
{
|
||||
xs_list *l = xs_list_new();
|
||||
xs_list *p;
|
||||
const xs_list *p;
|
||||
|
||||
/* try first the attachments list */
|
||||
if (!xs_is_null(p = xs_dict_get(msg, "attachment"))) {
|
||||
xs *attach = NULL;
|
||||
xs_val *v;
|
||||
const xs_val *v;
|
||||
|
||||
/* ensure it's a list */
|
||||
if (xs_type(p) == XSTYPE_DICT) {
|
||||
|
@ -203,23 +204,24 @@ xs_list *get_attachments(const xs_dict *msg)
|
|||
|
||||
if (xs_type(attach) == XSTYPE_LIST) {
|
||||
/* does the message have an image? */
|
||||
if (xs_type(v = xs_dict_get(msg, "image")) == XSTYPE_DICT) {
|
||||
const xs_dict *d = xs_dict_get(msg, "image");
|
||||
if (xs_type(d) == XSTYPE_DICT) {
|
||||
/* add it to the attachment list */
|
||||
attach = xs_list_append(attach, v);
|
||||
attach = xs_list_append(attach, d);
|
||||
}
|
||||
}
|
||||
|
||||
/* now iterate the list */
|
||||
p = attach;
|
||||
while (xs_list_iter(&p, &v)) {
|
||||
char *type = xs_dict_get(v, "mediaType");
|
||||
int c = 0;
|
||||
while (xs_list_next(attach, &v, &c)) {
|
||||
const char *type = xs_dict_get(v, "mediaType");
|
||||
if (xs_is_null(type))
|
||||
type = xs_dict_get(v, "type");
|
||||
|
||||
if (xs_is_null(type))
|
||||
continue;
|
||||
|
||||
char *href = xs_dict_get(v, "url");
|
||||
const char *href = xs_dict_get(v, "url");
|
||||
if (xs_is_null(href))
|
||||
href = xs_dict_get(v, "href");
|
||||
if (xs_is_null(href))
|
||||
|
@ -233,7 +235,7 @@ xs_list *get_attachments(const xs_dict *msg)
|
|||
type = mt;
|
||||
}
|
||||
|
||||
char *name = xs_dict_get(v, "name");
|
||||
const char *name = xs_dict_get(v, "name");
|
||||
if (xs_is_null(name))
|
||||
name = xs_dict_get(msg, "name");
|
||||
if (xs_is_null(name))
|
||||
|
@ -252,29 +254,31 @@ xs_list *get_attachments(const xs_dict *msg)
|
|||
p = xs_dict_get(msg, "url");
|
||||
|
||||
if (xs_type(p) == XSTYPE_LIST) {
|
||||
char *href = NULL;
|
||||
char *type = NULL;
|
||||
xs_val *v;
|
||||
const char *href = NULL;
|
||||
const char *type = NULL;
|
||||
int c = 0;
|
||||
const xs_val *v;
|
||||
|
||||
while (href == NULL && xs_list_iter(&p, &v)) {
|
||||
while (href == NULL && xs_list_next(p, &v, &c)) {
|
||||
if (xs_type(v) == XSTYPE_DICT) {
|
||||
char *mtype = xs_dict_get(v, "type");
|
||||
const char *mtype = xs_dict_get(v, "type");
|
||||
|
||||
if (xs_type(mtype) == XSTYPE_STRING && strcmp(mtype, "Link") == 0) {
|
||||
mtype = xs_dict_get(v, "mediaType");
|
||||
xs_list *tag = xs_dict_get(v, "tag");
|
||||
const xs_list *tag = xs_dict_get(v, "tag");
|
||||
|
||||
if (xs_type(mtype) == XSTYPE_STRING &&
|
||||
strcmp(mtype, "application/x-mpegURL") == 0 &&
|
||||
xs_type(tag) == XSTYPE_LIST) {
|
||||
/* now iterate the tag list, looking for a video URL */
|
||||
xs_dict *d;
|
||||
const xs_dict *d;
|
||||
int c = 0;
|
||||
|
||||
while (href == NULL && xs_list_iter(&tag, &d)) {
|
||||
while (href == NULL && xs_list_next(tag, &d, &c)) {
|
||||
if (xs_type(d) == XSTYPE_DICT) {
|
||||
if (xs_type(mtype = xs_dict_get(d, "mediaType")) == XSTYPE_STRING &&
|
||||
xs_startswith(mtype, "video/")) {
|
||||
char *h = xs_dict_get(d, "href");
|
||||
const char *h = xs_dict_get(d, "href");
|
||||
|
||||
/* this is probably it */
|
||||
if (xs_type(h) == XSTYPE_STRING) {
|
||||
|
@ -303,7 +307,7 @@ xs_list *get_attachments(const xs_dict *msg)
|
|||
}
|
||||
|
||||
|
||||
int timeline_request(snac *snac, char **id, xs_str **wrk, int level)
|
||||
int timeline_request(snac *snac, const char **id, xs_str **wrk, int level)
|
||||
/* ensures that an entry and its ancestors are in the timeline */
|
||||
{
|
||||
int status = 0;
|
||||
|
@ -323,7 +327,7 @@ int timeline_request(snac *snac, char **id, xs_str **wrk, int level)
|
|||
status = activitypub_request(snac, *id, &msg);
|
||||
|
||||
if (valid_status(status)) {
|
||||
xs_dict *object = msg;
|
||||
const xs_dict *object = msg;
|
||||
const char *type = xs_dict_get(object, "type");
|
||||
|
||||
/* get the id again from the object, as it may be different */
|
||||
|
@ -355,102 +359,35 @@ int timeline_request(snac *snac, char **id, xs_str **wrk, int level)
|
|||
type = "(null)";
|
||||
}
|
||||
|
||||
if (xs_match(type, "Note|Page|Article|Video")) {
|
||||
const char *actor = get_atto(object);
|
||||
|
||||
if (content_check("filter_reject.txt", object))
|
||||
if (xs_match(type, POSTLIKE_OBJECT_TYPE)) {
|
||||
if (content_match("filter_reject.txt", object))
|
||||
snac_log(snac, xs_fmt("timeline_request rejected by content %s", nid));
|
||||
else {
|
||||
/* request (and drop) the actor for this entry */
|
||||
if (!xs_is_null(actor))
|
||||
actor_request(snac, actor, NULL);
|
||||
const char *actor = get_atto(object);
|
||||
|
||||
/* does it have an ancestor? */
|
||||
char *in_reply_to = xs_dict_get(object, "inReplyTo");
|
||||
|
||||
/* store */
|
||||
timeline_add(snac, nid, object);
|
||||
|
||||
/* recurse! */
|
||||
timeline_request(snac, &in_reply_to, NULL, level + 1);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
enqueue_request_replies(snac, *id);
|
||||
}
|
||||
|
||||
return status;
|
||||
}
|
||||
|
||||
|
||||
void timeline_request_replies(snac *user, const char *id)
|
||||
/* requests all replies of a message */
|
||||
/* FIXME: experimental -- needs more testing */
|
||||
{
|
||||
/* FIXME: TEMPORARILY DISABLED */
|
||||
/* Reason: I've found that many of the posts in the 'replies' Collection
|
||||
do not have an inReplyTo field (why??? aren't they 'replies'???).
|
||||
For this reason, these requested objects are not stored as children
|
||||
of the original post and they are shown as out-of-context, top level posts.
|
||||
This process is disabled until I find an elegant way of providing a parent
|
||||
for these 'stray' children. */
|
||||
return;
|
||||
|
||||
xs *msg = NULL;
|
||||
|
||||
if (!valid_status(object_get(id, &msg)))
|
||||
return;
|
||||
|
||||
/* does it have a replies collection? */
|
||||
const xs_dict *replies = xs_dict_get(msg, "replies");
|
||||
|
||||
if (!xs_is_null(replies)) {
|
||||
const char *type = xs_dict_get(replies, "type");
|
||||
const char *first = xs_dict_get(replies, "first");
|
||||
|
||||
if (!xs_is_null(type) && !xs_is_null(first) && strcmp(type, "Collection") == 0) {
|
||||
const char *next = xs_dict_get(first, "next");
|
||||
|
||||
if (!xs_is_null(next)) {
|
||||
xs *rpls = NULL;
|
||||
int status = activitypub_request(user, next, &rpls);
|
||||
|
||||
/* request the Collection of replies */
|
||||
if (valid_status(status)) {
|
||||
xs_list *items = xs_dict_get(rpls, "items");
|
||||
|
||||
if (xs_type(items) == XSTYPE_LIST) {
|
||||
xs_val *v;
|
||||
|
||||
/* request them all */
|
||||
while (xs_list_iter(&items, &v)) {
|
||||
if (xs_type(v) == XSTYPE_DICT) {
|
||||
/* not an id, but the object itself (!) */
|
||||
const char *c_id = xs_dict_get(v, "id");
|
||||
|
||||
if (!xs_is_null(id)) {
|
||||
snac_debug(user, 0, xs_fmt("embedded reply %s", c_id));
|
||||
|
||||
object_add(c_id, v);
|
||||
|
||||
/* get its own children */
|
||||
timeline_request_replies(user, v);
|
||||
}
|
||||
}
|
||||
else {
|
||||
snac_debug(user, 0, xs_fmt("request reply %s", v));
|
||||
timeline_request(user, &v, NULL, 0);
|
||||
if (!xs_is_null(actor)) {
|
||||
/* request (and drop) the actor for this entry */
|
||||
if (!valid_status(actor_request(snac, actor, NULL))) {
|
||||
/* failed? retry later */
|
||||
enqueue_actor_refresh(snac, actor, 60);
|
||||
}
|
||||
|
||||
/* does it have an ancestor? */
|
||||
const char *in_reply_to = xs_dict_get(object, "inReplyTo");
|
||||
|
||||
/* store */
|
||||
timeline_add(snac, nid, object);
|
||||
|
||||
/* recurse! */
|
||||
timeline_request(snac, &in_reply_to, NULL, level + 1);
|
||||
}
|
||||
}
|
||||
}
|
||||
else
|
||||
snac_debug(user, 0, xs_fmt("replies request error %s %d", next, status));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return status;
|
||||
}
|
||||
|
||||
|
||||
|
@ -476,7 +413,7 @@ int send_to_inbox(snac *snac, const xs_str *inbox, const xs_dict *msg,
|
|||
xs_val **payload, int *p_size, int timeout)
|
||||
/* sends a message to an Inbox */
|
||||
{
|
||||
char *seckey = xs_dict_get(snac->key, "secret");
|
||||
const char *seckey = xs_dict_get(snac->key, "secret");
|
||||
|
||||
return send_to_inbox_raw(snac->actor, seckey, inbox, msg, payload, p_size, timeout);
|
||||
}
|
||||
|
@ -486,7 +423,7 @@ xs_str *get_actor_inbox(const char *actor)
|
|||
/* gets an actor's inbox */
|
||||
{
|
||||
xs *data = NULL;
|
||||
char *v = NULL;
|
||||
const char *v = NULL;
|
||||
|
||||
if (valid_status(actor_request(NULL, actor, &data))) {
|
||||
/* try first endpoints/sharedInbox */
|
||||
|
@ -535,17 +472,17 @@ void post_message(snac *snac, const char *actor, const xs_dict *msg)
|
|||
xs_list *recipient_list(snac *snac, const xs_dict *msg, int expand_public)
|
||||
/* returns the list of recipients for a message */
|
||||
{
|
||||
char *to = xs_dict_get(msg, "to");
|
||||
char *cc = xs_dict_get(msg, "cc");
|
||||
const xs_val *to = xs_dict_get(msg, "to");
|
||||
const xs_val *cc = xs_dict_get(msg, "cc");
|
||||
xs_set rcpts;
|
||||
int n;
|
||||
|
||||
xs_set_init(&rcpts);
|
||||
|
||||
char *lists[] = { to, cc, NULL };
|
||||
const xs_list *lists[] = { to, cc, NULL };
|
||||
for (n = 0; lists[n]; n++) {
|
||||
char *l = lists[n];
|
||||
char *v;
|
||||
xs_list *l = (xs_list *)lists[n];
|
||||
const char *v;
|
||||
xs *tl = NULL;
|
||||
|
||||
/* if it's a string, create a list with only one element */
|
||||
|
@ -560,7 +497,7 @@ xs_list *recipient_list(snac *snac, const xs_dict *msg, int expand_public)
|
|||
if (expand_public && strcmp(v, public_address) == 0) {
|
||||
/* iterate the followers and add them */
|
||||
xs *fwers = follower_list(snac);
|
||||
char *actor;
|
||||
const char *actor;
|
||||
|
||||
char *p = fwers;
|
||||
while (xs_list_iter(&p, &actor))
|
||||
|
@ -667,13 +604,13 @@ int is_msg_for_me(snac *snac, const xs_dict *c_msg)
|
|||
|
||||
/* if it's a Follow, it must be explicitly for us */
|
||||
if (xs_match(type, "Follow")) {
|
||||
char *object = xs_dict_get(c_msg, "object");
|
||||
const char *object = xs_dict_get(c_msg, "object");
|
||||
return !xs_is_null(object) && strcmp(snac->actor, object) == 0;
|
||||
}
|
||||
|
||||
/* only accept Ping directed to us */
|
||||
if (xs_match(type, "Ping")) {
|
||||
char *dest = xs_dict_get(c_msg, "to");
|
||||
const char *dest = xs_dict_get(c_msg, "to");
|
||||
return !xs_is_null(dest) && strcmp(snac->actor, dest) == 0;
|
||||
}
|
||||
|
||||
|
@ -688,10 +625,10 @@ int is_msg_for_me(snac *snac, const xs_dict *c_msg)
|
|||
if (pub_msg && following_check(snac, actor))
|
||||
return 1;
|
||||
|
||||
xs_dict *msg = xs_dict_get(c_msg, "object");
|
||||
const xs_dict *msg = xs_dict_get(c_msg, "object");
|
||||
xs *rcpts = recipient_list(snac, msg, 0);
|
||||
xs_list *p = rcpts;
|
||||
xs_str *v;
|
||||
const xs_str *v;
|
||||
|
||||
xs *actor_followers = NULL;
|
||||
|
||||
|
@ -700,8 +637,9 @@ int is_msg_for_me(snac *snac, const xs_dict *c_msg)
|
|||
xs *actor_obj = NULL;
|
||||
|
||||
if (valid_status(object_get(actor, &actor_obj))) {
|
||||
if ((v = xs_dict_get(actor_obj, "followers")))
|
||||
actor_followers = xs_dup(v);
|
||||
const xs_val *fw = xs_dict_get(actor_obj, "followers");
|
||||
if (fw)
|
||||
actor_followers = xs_dup(fw);
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -724,13 +662,13 @@ int is_msg_for_me(snac *snac, const xs_dict *c_msg)
|
|||
}
|
||||
|
||||
/* accept if it's by someone we follow */
|
||||
char *atto = get_atto(msg);
|
||||
const char *atto = get_atto(msg);
|
||||
|
||||
if (pub_msg && !xs_is_null(atto) && following_check(snac, atto))
|
||||
return 3;
|
||||
|
||||
/* is this message a reply to another? */
|
||||
char *irt = xs_dict_get(msg, "inReplyTo");
|
||||
const char *irt = xs_dict_get(msg, "inReplyTo");
|
||||
if (!xs_is_null(irt)) {
|
||||
xs *r_msg = NULL;
|
||||
|
||||
|
@ -755,7 +693,7 @@ xs_str *process_tags(snac *snac, const char *content, xs_list **tag)
|
|||
xs_list *tl = *tag;
|
||||
xs *split;
|
||||
xs_list *p;
|
||||
xs_val *v;
|
||||
const xs_val *v;
|
||||
int n = 0;
|
||||
|
||||
/* create a default server for incomplete mentions */
|
||||
|
@ -983,8 +921,8 @@ void notify(snac *snac, const char *type, const char *utype, const char *actor,
|
|||
|
||||
/* telegram */
|
||||
|
||||
char *bot = xs_dict_get(snac->config, "telegram_bot");
|
||||
char *chat_id = xs_dict_get(snac->config, "telegram_chat_id");
|
||||
const char *bot = xs_dict_get(snac->config, "telegram_bot");
|
||||
const char *chat_id = xs_dict_get(snac->config, "telegram_chat_id");
|
||||
|
||||
if (!xs_is_null(bot) && !xs_is_null(chat_id) && *bot && *chat_id)
|
||||
enqueue_telegram(body, bot, chat_id);
|
||||
|
@ -997,8 +935,8 @@ void notify(snac *snac, const char *type, const char *utype, const char *actor,
|
|||
objid = actor;
|
||||
|
||||
/* ntfy */
|
||||
char *ntfy_server = xs_dict_get(snac->config, "ntfy_server");
|
||||
char *ntfy_token = xs_dict_get(snac->config, "ntfy_token");
|
||||
const char *ntfy_server = xs_dict_get(snac->config, "ntfy_server");
|
||||
const char *ntfy_token = xs_dict_get(snac->config, "ntfy_token");
|
||||
|
||||
if (!xs_is_null(ntfy_server) && *ntfy_server)
|
||||
enqueue_ntfy(body, ntfy_server, ntfy_token);
|
||||
|
@ -1084,7 +1022,7 @@ xs_dict *msg_base(snac *snac, const char *type, const char *id,
|
|||
}
|
||||
|
||||
|
||||
xs_dict *msg_collection(snac *snac, char *id)
|
||||
xs_dict *msg_collection(snac *snac, const char *id)
|
||||
/* creates an empty OrderedCollection message */
|
||||
{
|
||||
xs_dict *msg = msg_base(snac, "OrderedCollection", id, NULL, NULL, NULL);
|
||||
|
@ -1098,7 +1036,7 @@ xs_dict *msg_collection(snac *snac, char *id)
|
|||
}
|
||||
|
||||
|
||||
xs_dict *msg_accept(snac *snac, char *object, char *to)
|
||||
xs_dict *msg_accept(snac *snac, const xs_val *object, const char *to)
|
||||
/* creates an Accept message (as a response to a Follow) */
|
||||
{
|
||||
xs_dict *msg = msg_base(snac, "Accept", "@dummy", snac->actor, NULL, object);
|
||||
|
@ -1109,12 +1047,12 @@ xs_dict *msg_accept(snac *snac, char *object, char *to)
|
|||
}
|
||||
|
||||
|
||||
xs_dict *msg_update(snac *snac, xs_dict *object)
|
||||
xs_dict *msg_update(snac *snac, const xs_dict *object)
|
||||
/* creates an Update message */
|
||||
{
|
||||
xs_dict *msg = msg_base(snac, "Update", "@object", snac->actor, "@now", object);
|
||||
|
||||
char *type = xs_dict_get(object, "type");
|
||||
const char *type = xs_dict_get(object, "type");
|
||||
|
||||
if (strcmp(type, "Note") == 0) {
|
||||
msg = xs_dict_append(msg, "to", xs_dict_get(object, "to"));
|
||||
|
@ -1137,7 +1075,7 @@ xs_dict *msg_update(snac *snac, xs_dict *object)
|
|||
}
|
||||
|
||||
|
||||
xs_dict *msg_admiration(snac *snac, char *object, char *type)
|
||||
xs_dict *msg_admiration(snac *snac, const char *object, const char *type)
|
||||
/* creates a Like or Announce message */
|
||||
{
|
||||
xs *a_msg = NULL;
|
||||
|
@ -1168,7 +1106,7 @@ xs_dict *msg_admiration(snac *snac, char *object, char *type)
|
|||
}
|
||||
|
||||
|
||||
xs_dict *msg_repulsion(snac *user, char *id, char *type)
|
||||
xs_dict *msg_repulsion(snac *user, const char *id, const char *type)
|
||||
/* creates an Undo + admiration message */
|
||||
{
|
||||
xs *a_msg = NULL;
|
||||
|
@ -1206,7 +1144,7 @@ xs_dict *msg_actor(snac *snac)
|
|||
xs *kid = NULL;
|
||||
xs *f_bio = NULL;
|
||||
xs_dict *msg = msg_base(snac, "Person", snac->actor, NULL, NULL, NULL);
|
||||
char *p;
|
||||
const char *p;
|
||||
int n;
|
||||
|
||||
/* change the @context (is this really necessary?) */
|
||||
|
@ -1264,11 +1202,11 @@ xs_dict *msg_actor(snac *snac)
|
|||
}
|
||||
|
||||
/* add the metadata as attachments of PropertyValue */
|
||||
xs_dict *metadata = xs_dict_get(snac->config, "metadata");
|
||||
const xs_dict *metadata = xs_dict_get(snac->config, "metadata");
|
||||
if (xs_type(metadata) == XSTYPE_DICT) {
|
||||
xs *attach = xs_list_new();
|
||||
xs_str *k;
|
||||
xs_str *v;
|
||||
const xs_str *k;
|
||||
const xs_str *v;
|
||||
|
||||
int c = 0;
|
||||
while (xs_dict_next(metadata, &k, &v, &c)) {
|
||||
|
@ -1277,7 +1215,7 @@ xs_dict *msg_actor(snac *snac)
|
|||
xs *k2 = encode_html(k);
|
||||
xs *v2 = NULL;
|
||||
|
||||
if (xs_startswith(v, "https:")) {
|
||||
if (xs_startswith(v, "https:/") || xs_startswith(v, "http:/")) {
|
||||
xs *t = encode_html(v);
|
||||
v2 = xs_fmt("<a href=\"%s\" rel=\"me\">%s</a>", t, t);
|
||||
}
|
||||
|
@ -1310,7 +1248,7 @@ xs_dict *msg_create(snac *snac, const xs_dict *object)
|
|||
/* creates a 'Create' message */
|
||||
{
|
||||
xs_dict *msg = msg_base(snac, "Create", "@wrapper", snac->actor, NULL, object);
|
||||
xs_val *v;
|
||||
const xs_val *v;
|
||||
|
||||
if ((v = get_atto(object)))
|
||||
msg = xs_dict_append(msg, "attributedTo", v);
|
||||
|
@ -1327,7 +1265,7 @@ xs_dict *msg_create(snac *snac, const xs_dict *object)
|
|||
}
|
||||
|
||||
|
||||
xs_dict *msg_undo(snac *snac, char *object)
|
||||
xs_dict *msg_undo(snac *snac, const xs_val *object)
|
||||
/* creates an 'Undo' message */
|
||||
{
|
||||
xs_dict *msg = msg_base(snac, "Undo", "@object", snac->actor, "@now", object);
|
||||
|
@ -1340,7 +1278,7 @@ xs_dict *msg_undo(snac *snac, char *object)
|
|||
}
|
||||
|
||||
|
||||
xs_dict *msg_delete(snac *snac, char *id)
|
||||
xs_dict *msg_delete(snac *snac, const char *id)
|
||||
/* creates a 'Delete' + 'Tombstone' for a local entry */
|
||||
{
|
||||
xs *tomb = xs_dict_new();
|
||||
|
@ -1369,7 +1307,7 @@ xs_dict *msg_follow(snac *snac, const char *q)
|
|||
|
||||
xs *url_or_uid = xs_strip_i(xs_str_new(q));
|
||||
|
||||
if (xs_startswith(url_or_uid, "https:/"))
|
||||
if (xs_startswith(url_or_uid, "https:/") || xs_startswith(url_or_uid, "http:/"))
|
||||
actor = xs_dup(url_or_uid);
|
||||
else
|
||||
if (!valid_status(webfinger_request(url_or_uid, &actor, NULL)) || actor == NULL) {
|
||||
|
@ -1382,7 +1320,7 @@ xs_dict *msg_follow(snac *snac, const char *q)
|
|||
|
||||
if (valid_status(status)) {
|
||||
/* check if the actor is an alias */
|
||||
char *r_actor = xs_dict_get(actor_o, "id");
|
||||
const char *r_actor = xs_dict_get(actor_o, "id");
|
||||
|
||||
if (r_actor && strcmp(actor, r_actor) != 0) {
|
||||
snac_log(snac, xs_fmt("actor to follow is an alias %s -> %s", actor, r_actor));
|
||||
|
@ -1398,7 +1336,7 @@ xs_dict *msg_follow(snac *snac, const char *q)
|
|||
|
||||
|
||||
xs_dict *msg_note(snac *snac, const xs_str *content, const xs_val *rcpts,
|
||||
xs_str *in_reply_to, xs_list *attach, int priv)
|
||||
const xs_str *in_reply_to, const xs_list *attach, int priv)
|
||||
/* creates a 'Note' message */
|
||||
{
|
||||
xs *ntid = tid(0);
|
||||
|
@ -1413,7 +1351,7 @@ xs_dict *msg_note(snac *snac, const xs_str *content, const xs_val *rcpts,
|
|||
xs *atls = xs_list_new();
|
||||
xs_dict *msg = msg_base(snac, "Note", id, NULL, "@now", NULL);
|
||||
xs_list *p;
|
||||
xs_val *v;
|
||||
const xs_val *v;
|
||||
|
||||
if (rcpts == NULL)
|
||||
to = xs_list_new();
|
||||
|
@ -1438,7 +1376,7 @@ xs_dict *msg_note(snac *snac, const xs_str *content, const xs_val *rcpts,
|
|||
|
||||
if (valid_status(object_get(in_reply_to, &p_msg))) {
|
||||
/* add this author as recipient */
|
||||
char *a, *v;
|
||||
const char *a, *v;
|
||||
|
||||
if ((a = get_atto(p_msg)) && xs_list_in(to, a) == -1)
|
||||
to = xs_list_append(to, a);
|
||||
|
@ -1449,7 +1387,7 @@ xs_dict *msg_note(snac *snac, const xs_str *content, const xs_val *rcpts,
|
|||
xs *actor_o = NULL;
|
||||
|
||||
if (xs_list_len(l) > 3 && valid_status(object_get(a, &actor_o))) {
|
||||
char *uname = xs_dict_get(actor_o, "preferredUsername");
|
||||
const char *uname = xs_dict_get(actor_o, "preferredUsername");
|
||||
|
||||
if (!xs_is_null(uname) && *uname) {
|
||||
xs *handle = xs_fmt("@%s@%s", uname, xs_list_get(l, 2));
|
||||
|
@ -1488,7 +1426,8 @@ xs_dict *msg_note(snac *snac, const xs_str *content, const xs_val *rcpts,
|
|||
|
||||
/* create the attachment list, if there are any */
|
||||
if (!xs_is_null(attach)) {
|
||||
while (xs_list_iter(&attach, &v)) {
|
||||
int c = 0;
|
||||
while (xs_list_next(attach, &v, &c)) {
|
||||
xs *d = xs_dict_new();
|
||||
const char *url = xs_list_get(v, 0);
|
||||
const char *alt = xs_list_get(v, 1);
|
||||
|
@ -1511,7 +1450,7 @@ xs_dict *msg_note(snac *snac, const xs_str *content, const xs_val *rcpts,
|
|||
p = tag;
|
||||
while (xs_list_iter(&p, &v)) {
|
||||
if (xs_type(v) == XSTYPE_DICT) {
|
||||
char *t;
|
||||
const char *t;
|
||||
|
||||
if (!xs_is_null(t = xs_dict_get(v, "type")) && strcmp(t, "Mention") == 0) {
|
||||
if (!xs_is_null(t = xs_dict_get(v, "href")))
|
||||
|
@ -1589,7 +1528,7 @@ xs_dict *msg_question(snac *user, const char *content, xs_list *attach,
|
|||
|
||||
xs *o = xs_list_new();
|
||||
xs_list *p = (xs_list *)opts;
|
||||
xs_str *v;
|
||||
const xs_str *v;
|
||||
xs *replies = xs_json_loads("{\"type\":\"Collection\",\"totalItems\":0}");
|
||||
|
||||
xs_set_init(&seen);
|
||||
|
@ -1635,9 +1574,9 @@ int update_question(snac *user, const char *id)
|
|||
xs *msg = NULL;
|
||||
xs *rcnt = xs_dict_new();
|
||||
xs *lopts = xs_list_new();
|
||||
xs_list *opts;
|
||||
const xs_list *opts;
|
||||
xs_list *p;
|
||||
xs_val *v;
|
||||
const xs_val *v;
|
||||
|
||||
/* get the object */
|
||||
if (!valid_status(object_get(id, &msg)))
|
||||
|
@ -1653,8 +1592,8 @@ int update_question(snac *user, const char *id)
|
|||
return -3;
|
||||
|
||||
/* fill the initial count */
|
||||
p = opts;
|
||||
while (xs_list_iter(&p, &v)) {
|
||||
int c = 0;
|
||||
while (xs_list_next(opts, &v, &c)) {
|
||||
const char *name = xs_dict_get(v, "name");
|
||||
if (name) {
|
||||
lopts = xs_list_append(lopts, name);
|
||||
|
@ -1760,13 +1699,13 @@ int update_question(snac *user, const char *id)
|
|||
|
||||
/** queues **/
|
||||
|
||||
int process_input_message(snac *snac, xs_dict *msg, xs_dict *req)
|
||||
int process_input_message(snac *snac, const xs_dict *msg, const xs_dict *req)
|
||||
/* processes an ActivityPub message from the input queue */
|
||||
/* return values: -1, fatal error; 0, transient error, retry;
|
||||
1, processed and done; 2, propagate to users (only when no user is set) */
|
||||
{
|
||||
char *actor = xs_dict_get(msg, "actor");
|
||||
char *type = xs_dict_get(msg, "type");
|
||||
const char *actor = xs_dict_get(msg, "actor");
|
||||
const char *type = xs_dict_get(msg, "type");
|
||||
xs *actor_o = NULL;
|
||||
int a_status;
|
||||
int do_notify = 0;
|
||||
|
@ -1786,7 +1725,7 @@ int process_input_message(snac *snac, xs_dict *msg, xs_dict *req)
|
|||
return -1;
|
||||
}
|
||||
|
||||
char *object, *utype;
|
||||
const char *object, *utype;
|
||||
|
||||
object = xs_dict_get(msg, "object");
|
||||
if (object != NULL && xs_type(object) == XSTYPE_DICT)
|
||||
|
@ -1809,7 +1748,7 @@ int process_input_message(snac *snac, xs_dict *msg, xs_dict *req)
|
|||
}
|
||||
|
||||
/* also discard if the object to be deleted is not here */
|
||||
char *obj_id = object;
|
||||
const char *obj_id = object;
|
||||
if (xs_type(obj_id) == XSTYPE_DICT)
|
||||
obj_id = xs_dict_get(obj_id, "id");
|
||||
|
||||
|
@ -1881,7 +1820,7 @@ int process_input_message(snac *snac, xs_dict *msg, xs_dict *req)
|
|||
int min_account_age = xs_number_get(xs_dict_get(srv_config, "min_account_age"));
|
||||
|
||||
if (min_account_age > 0) {
|
||||
char *actor_date = xs_dict_get(actor_o, "published");
|
||||
const char *actor_date = xs_dict_get(actor_o, "published");
|
||||
if (!xs_is_null(actor_date)) {
|
||||
time_t actor_t = xs_parse_iso_date(actor_date, 0);
|
||||
|
||||
|
@ -1941,16 +1880,39 @@ int process_input_message(snac *snac, xs_dict *msg, xs_dict *req)
|
|||
}
|
||||
else
|
||||
if (strcmp(type, "Undo") == 0) { /** **/
|
||||
const char *id = xs_dict_get(object, "object");
|
||||
|
||||
if (xs_type(object) != XSTYPE_DICT)
|
||||
utype = "Follow";
|
||||
|
||||
if (strcmp(utype, "Follow") == 0) { /** **/
|
||||
if (valid_status(follower_del(snac, actor))) {
|
||||
snac_log(snac, xs_fmt("no longer following us %s", actor));
|
||||
do_notify = 1;
|
||||
if (id && strcmp(id, snac->actor) != 0)
|
||||
snac_debug(snac, 1, xs_fmt("Undo + Follow from %s not for us (%s)", actor, id));
|
||||
else {
|
||||
if (valid_status(follower_del(snac, actor))) {
|
||||
snac_log(snac, xs_fmt("no longer following us %s", actor));
|
||||
do_notify = 1;
|
||||
}
|
||||
else
|
||||
snac_log(snac, xs_fmt("error deleting follower %s", actor));
|
||||
}
|
||||
else
|
||||
snac_log(snac, xs_fmt("error deleting follower %s", actor));
|
||||
}
|
||||
else
|
||||
if (strcmp(utype, "Like") == 0) { /** **/
|
||||
int status = object_unadmire(id, actor, 1);
|
||||
|
||||
snac_log(snac, xs_fmt("Unlike for %s %d", id, status));
|
||||
}
|
||||
else
|
||||
if (strcmp(utype, "Announce") == 0) { /** **/
|
||||
int status = 200;
|
||||
|
||||
/* commented out: if a followed user boosts something that
|
||||
is requested and then unboosts, the post remains here,
|
||||
but with no apparent reason, and that is confusing */
|
||||
//status = object_unadmire(id, actor, 0);
|
||||
|
||||
snac_log(snac, xs_fmt("Unboost for %s %d", id, status));
|
||||
}
|
||||
else
|
||||
snac_debug(snac, 1, xs_fmt("ignored 'Undo' for object type '%s'", utype));
|
||||
|
@ -1963,19 +1925,29 @@ int process_input_message(snac *snac, xs_dict *msg, xs_dict *req)
|
|||
}
|
||||
|
||||
if (xs_match(utype, "Note|Article")) { /** **/
|
||||
char *id = xs_dict_get(object, "id");
|
||||
char *in_reply_to = xs_dict_get(object, "inReplyTo");
|
||||
const char *id = xs_dict_get(object, "id");
|
||||
const char *in_reply_to = xs_dict_get(object, "inReplyTo");
|
||||
const char *atto = get_atto(object);
|
||||
xs *wrk = NULL;
|
||||
|
||||
if (xs_is_null(id))
|
||||
snac_log(snac, xs_fmt("malformed message: no 'id' field"));
|
||||
else
|
||||
if (xs_is_null(atto))
|
||||
snac_log(snac, xs_fmt("malformed message: no 'attributedTo' field"));
|
||||
else
|
||||
if (!xs_is_null(in_reply_to) && is_hidden(snac, in_reply_to)) {
|
||||
snac_debug(snac, 0, xs_fmt("dropped reply %s to hidden post %s", id, in_reply_to));
|
||||
}
|
||||
else {
|
||||
if (content_check("filter_reject.txt", object)) {
|
||||
if (content_match("filter_reject.txt", object)) {
|
||||
snac_log(snac, xs_fmt("rejected by content %s", id));
|
||||
return 1;
|
||||
}
|
||||
|
||||
if (strcmp(actor, atto) != 0)
|
||||
snac_log(snac, xs_fmt("SUSPICIOUS: actor != atto (%s != %s)", actor, atto));
|
||||
|
||||
timeline_request(snac, &in_reply_to, &wrk, 0);
|
||||
|
||||
if (timeline_add(snac, id, object)) {
|
||||
|
@ -1992,14 +1964,14 @@ int process_input_message(snac *snac, xs_dict *msg, xs_dict *req)
|
|||
}
|
||||
else
|
||||
if (strcmp(utype, "Question") == 0) { /** **/
|
||||
char *id = xs_dict_get(object, "id");
|
||||
const char *id = xs_dict_get(object, "id");
|
||||
|
||||
if (timeline_add(snac, id, object))
|
||||
snac_log(snac, xs_fmt("new 'Question' %s %s", actor, id));
|
||||
}
|
||||
else
|
||||
if (strcmp(utype, "Video") == 0) { /** **/
|
||||
char *id = xs_dict_get(object, "id");
|
||||
const char *id = xs_dict_get(object, "id");
|
||||
|
||||
if (timeline_add(snac, id, object))
|
||||
snac_log(snac, xs_fmt("new 'Video' %s %s", actor, id));
|
||||
|
@ -2080,6 +2052,9 @@ int process_input_message(snac *snac, xs_dict *msg, xs_dict *req)
|
|||
snac_log(snac, xs_fmt("repeated 'Announce' from %s to %s",
|
||||
actor, object));
|
||||
|
||||
/* distribute the post with the actor as 'proxy' */
|
||||
list_distribute(snac, actor, a_msg);
|
||||
|
||||
do_notify = 1;
|
||||
}
|
||||
else
|
||||
|
@ -2172,7 +2147,7 @@ int process_input_message(snac *snac, xs_dict *msg, xs_dict *req)
|
|||
}
|
||||
|
||||
|
||||
int send_email(char *msg)
|
||||
int send_email(const char *msg)
|
||||
/* invoke sendmail with email headers and body in msg */
|
||||
{
|
||||
FILE *f;
|
||||
|
@ -2204,18 +2179,18 @@ int send_email(char *msg)
|
|||
void process_user_queue_item(snac *snac, xs_dict *q_item)
|
||||
/* processes an item from the user queue */
|
||||
{
|
||||
char *type;
|
||||
const char *type;
|
||||
int queue_retry_max = xs_number_get(xs_dict_get(srv_config, "queue_retry_max"));
|
||||
|
||||
if ((type = xs_dict_get(q_item, "type")) == NULL)
|
||||
type = "output";
|
||||
|
||||
if (strcmp(type, "message") == 0) {
|
||||
xs_dict *msg = xs_dict_get(q_item, "message");
|
||||
const xs_dict *msg = xs_dict_get(q_item, "message");
|
||||
xs *rcpts = recipient_list(snac, msg, 1);
|
||||
xs_set inboxes;
|
||||
xs_list *p;
|
||||
xs_str *actor;
|
||||
const xs_str *actor;
|
||||
|
||||
xs_set_init(&inboxes);
|
||||
|
||||
|
@ -2237,7 +2212,7 @@ void process_user_queue_item(snac *snac, xs_dict *q_item)
|
|||
if (is_msg_public(msg)) {
|
||||
if (xs_type(xs_dict_get(srv_config, "disable_inbox_collection")) != XSTYPE_TRUE) {
|
||||
xs *shibx = inbox_list();
|
||||
xs_str *inbox;
|
||||
const xs_str *inbox;
|
||||
|
||||
p = shibx;
|
||||
while (xs_list_iter(&p, &inbox)) {
|
||||
|
@ -2252,8 +2227,8 @@ void process_user_queue_item(snac *snac, xs_dict *q_item)
|
|||
else
|
||||
if (strcmp(type, "input") == 0) {
|
||||
/* process the message */
|
||||
xs_dict *msg = xs_dict_get(q_item, "message");
|
||||
xs_dict *req = xs_dict_get(q_item, "req");
|
||||
const xs_dict *msg = xs_dict_get(q_item, "message");
|
||||
const xs_dict *req = xs_dict_get(q_item, "req");
|
||||
int retries = xs_number_get(xs_dict_get(q_item, "retries"));
|
||||
|
||||
if (xs_is_null(msg))
|
||||
|
@ -2280,11 +2255,20 @@ void process_user_queue_item(snac *snac, xs_dict *q_item)
|
|||
update_question(snac, id);
|
||||
}
|
||||
else
|
||||
if (strcmp(type, "request_replies") == 0) {
|
||||
if (strcmp(type, "object_request") == 0) {
|
||||
const char *id = xs_dict_get(q_item, "message");
|
||||
|
||||
if (!xs_is_null(id))
|
||||
timeline_request_replies(snac, id);
|
||||
if (!xs_is_null(id)) {
|
||||
int status;
|
||||
xs *data = NULL;
|
||||
|
||||
status = activitypub_request(snac, id, &data);
|
||||
|
||||
if (valid_status(status))
|
||||
object_add_ow(id, data);
|
||||
|
||||
snac_debug(snac, 1, xs_fmt("object_request %s %d", id, status));
|
||||
}
|
||||
}
|
||||
else
|
||||
if (strcmp(type, "verify_links") == 0) {
|
||||
|
@ -2320,7 +2304,7 @@ int process_user_queue(snac *snac)
|
|||
xs *list = user_queue(snac);
|
||||
|
||||
xs_list *p = list;
|
||||
xs_str *fn;
|
||||
const xs_str *fn;
|
||||
|
||||
while (xs_list_iter(&p, &fn)) {
|
||||
xs *q_item = dequeue(fn);
|
||||
|
@ -2339,19 +2323,20 @@ int process_user_queue(snac *snac)
|
|||
void process_queue_item(xs_dict *q_item)
|
||||
/* processes an item from the global queue */
|
||||
{
|
||||
char *type = xs_dict_get(q_item, "type");
|
||||
const char *type = xs_dict_get(q_item, "type");
|
||||
int queue_retry_max = xs_number_get(xs_dict_get(srv_config, "queue_retry_max"));
|
||||
|
||||
if (strcmp(type, "output") == 0) {
|
||||
int status;
|
||||
xs_str *inbox = xs_dict_get(q_item, "inbox");
|
||||
xs_str *keyid = xs_dict_get(q_item, "keyid");
|
||||
xs_str *seckey = xs_dict_get(q_item, "seckey");
|
||||
xs_dict *msg = xs_dict_get(q_item, "message");
|
||||
const xs_str *inbox = xs_dict_get(q_item, "inbox");
|
||||
const xs_str *keyid = xs_dict_get(q_item, "keyid");
|
||||
const xs_str *seckey = xs_dict_get(q_item, "seckey");
|
||||
const xs_dict *msg = xs_dict_get(q_item, "message");
|
||||
int retries = xs_number_get(xs_dict_get(q_item, "retries"));
|
||||
int p_status = xs_number_get(xs_dict_get(q_item, "p_status"));
|
||||
xs *payload = NULL;
|
||||
int p_size = 0;
|
||||
int timeout = 0;
|
||||
|
||||
if (xs_is_null(inbox) || xs_is_null(msg) || xs_is_null(keyid) || xs_is_null(seckey)) {
|
||||
srv_log(xs_fmt("output message error: missing fields"));
|
||||
|
@ -2364,8 +2349,15 @@ void process_queue_item(xs_dict *q_item)
|
|||
}
|
||||
|
||||
/* deliver (if previous error status was a timeout, try now longer) */
|
||||
status = send_to_inbox_raw(keyid, seckey, inbox, msg,
|
||||
&payload, &p_size, p_status == 599 ? 8 : 6);
|
||||
if (p_status == 599)
|
||||
timeout = xs_number_get(xs_dict_get_def(srv_config, "queue_timeout_2", "8"));
|
||||
else
|
||||
timeout = xs_number_get(xs_dict_get_def(srv_config, "queue_timeout", "6"));
|
||||
|
||||
if (timeout == 0)
|
||||
timeout = 6;
|
||||
|
||||
status = send_to_inbox_raw(keyid, seckey, inbox, msg, &payload, &p_size, timeout);
|
||||
|
||||
if (payload) {
|
||||
if (p_size > 64) {
|
||||
|
@ -2411,7 +2403,7 @@ void process_queue_item(xs_dict *q_item)
|
|||
else
|
||||
if (strcmp(type, "email") == 0) {
|
||||
/* send this email */
|
||||
xs_str *msg = xs_dict_get(q_item, "message");
|
||||
const xs_str *msg = xs_dict_get(q_item, "message");
|
||||
int retries = xs_number_get(xs_dict_get(q_item, "retries"));
|
||||
|
||||
if (!send_email(msg))
|
||||
|
@ -2433,8 +2425,8 @@ void process_queue_item(xs_dict *q_item)
|
|||
else
|
||||
if (strcmp(type, "telegram") == 0) {
|
||||
/* send this via telegram */
|
||||
char *bot = xs_dict_get(q_item, "bot");
|
||||
char *msg = xs_dict_get(q_item, "message");
|
||||
const char *bot = xs_dict_get(q_item, "bot");
|
||||
const char *msg = xs_dict_get(q_item, "message");
|
||||
xs *chat_id = xs_dup(xs_dict_get(q_item, "chat_id"));
|
||||
int status = 0;
|
||||
|
||||
|
@ -2457,9 +2449,9 @@ void process_queue_item(xs_dict *q_item)
|
|||
else
|
||||
if (strcmp(type, "ntfy") == 0) {
|
||||
/* send this via ntfy */
|
||||
char *ntfy_server = xs_dict_get(q_item, "ntfy_server");
|
||||
char *msg = xs_dict_get(q_item, "message");
|
||||
char *ntfy_token = xs_dict_get(q_item, "ntfy_token");
|
||||
const char *ntfy_server = xs_dict_get(q_item, "ntfy_server");
|
||||
const char *msg = xs_dict_get(q_item, "message");
|
||||
const char *ntfy_token = xs_dict_get(q_item, "ntfy_token");
|
||||
int status = 0;
|
||||
|
||||
xs *url = xs_fmt("%s", ntfy_server);
|
||||
|
@ -2488,8 +2480,8 @@ void process_queue_item(xs_dict *q_item)
|
|||
}
|
||||
else
|
||||
if (strcmp(type, "input") == 0) {
|
||||
xs_dict *msg = xs_dict_get(q_item, "message");
|
||||
xs_dict *req = xs_dict_get(q_item, "req");
|
||||
const xs_dict *msg = xs_dict_get(q_item, "message");
|
||||
const xs_dict *req = xs_dict_get(q_item, "req");
|
||||
int retries = xs_number_get(xs_dict_get(q_item, "retries"));
|
||||
|
||||
/* do some instance-level checks */
|
||||
|
@ -2497,8 +2489,6 @@ void process_queue_item(xs_dict *q_item)
|
|||
|
||||
if (r == 0) {
|
||||
/* transient error? retry */
|
||||
int queue_retry_max = xs_number_get(xs_dict_get(srv_config, "queue_retry_max"));
|
||||
|
||||
if (retries > queue_retry_max)
|
||||
srv_log(xs_fmt("shared input giving up"));
|
||||
else {
|
||||
|
@ -2510,7 +2500,7 @@ void process_queue_item(xs_dict *q_item)
|
|||
else
|
||||
if (r == 2) {
|
||||
/* redistribute the input message to all users */
|
||||
char *ntid = xs_dict_get(q_item, "ntid");
|
||||
const char *ntid = xs_dict_get(q_item, "ntid");
|
||||
xs *tmpfn = xs_fmt("%s/tmp/%s.json", srv_basedir, ntid);
|
||||
FILE *f;
|
||||
|
||||
|
@ -2521,7 +2511,7 @@ void process_queue_item(xs_dict *q_item)
|
|||
|
||||
xs *users = user_list();
|
||||
xs_list *p = users;
|
||||
char *v;
|
||||
const char *v;
|
||||
int cnt = 0;
|
||||
|
||||
while (xs_list_iter(&p, &v)) {
|
||||
|
@ -2564,7 +2554,7 @@ int process_queue(void)
|
|||
xs *list = queue();
|
||||
|
||||
xs_list *p = list;
|
||||
xs_str *fn;
|
||||
const xs_str *fn;
|
||||
|
||||
while (xs_list_iter(&p, &fn)) {
|
||||
xs *q_item = dequeue(fn);
|
||||
|
@ -2585,7 +2575,7 @@ int activitypub_get_handler(const xs_dict *req, const char *q_path,
|
|||
char **body, int *b_size, char **ctype)
|
||||
{
|
||||
int status = 200;
|
||||
char *accept = xs_dict_get(req, "accept");
|
||||
const char *accept = xs_dict_get(req, "accept");
|
||||
snac snac;
|
||||
xs *msg = NULL;
|
||||
|
||||
|
@ -2597,7 +2587,8 @@ int activitypub_get_handler(const xs_dict *req, const char *q_path,
|
|||
return 0;
|
||||
|
||||
xs *l = xs_split_n(q_path, "/", 2);
|
||||
char *uid, *p_path;
|
||||
const char *uid;
|
||||
const char *p_path;
|
||||
|
||||
uid = xs_list_get(l, 1);
|
||||
if (!user_open(&snac, uid)) {
|
||||
|
@ -2615,7 +2606,7 @@ int activitypub_get_handler(const xs_dict *req, const char *q_path,
|
|||
msg = msg_actor(&snac);
|
||||
*ctype = "application/ld+json; profile=\"https://www.w3.org/ns/activitystreams\"";
|
||||
|
||||
char *ua = xs_dict_get(req, "user-agent");
|
||||
const char *ua = xs_dict_get(req, "user-agent");
|
||||
|
||||
snac_debug(&snac, 0, xs_fmt("serving actor [%s]", ua ? ua : "No UA"));
|
||||
}
|
||||
|
@ -2625,15 +2616,16 @@ int activitypub_get_handler(const xs_dict *req, const char *q_path,
|
|||
xs *elems = timeline_simple_list(&snac, "public", 0, 20);
|
||||
xs *list = xs_list_new();
|
||||
msg = msg_collection(&snac, id);
|
||||
char *p, *v;
|
||||
char *p;
|
||||
const char *v;
|
||||
|
||||
p = elems;
|
||||
while (xs_list_iter(&p, &v)) {
|
||||
xs *i = NULL;
|
||||
|
||||
if (valid_status(object_get_by_md5(v, &i))) {
|
||||
char *type = xs_dict_get(i, "type");
|
||||
char *id = xs_dict_get(i, "id");
|
||||
const char *type = xs_dict_get(i, "type");
|
||||
const char *id = xs_dict_get(i, "id");
|
||||
|
||||
if (type && id && strcmp(type, "Note") == 0 && xs_startswith(id, snac.actor)) {
|
||||
xs *c_msg = msg_create(&snac, i);
|
||||
|
@ -2686,9 +2678,9 @@ int activitypub_post_handler(const xs_dict *req, const char *q_path,
|
|||
(void)b_size;
|
||||
|
||||
int status = 202; /* accepted */
|
||||
char *i_ctype = xs_dict_get(req, "content-type");
|
||||
const char *i_ctype = xs_dict_get(req, "content-type");
|
||||
snac snac;
|
||||
char *v;
|
||||
const char *v;
|
||||
|
||||
if (i_ctype == NULL) {
|
||||
*body = xs_str_new("no content-type");
|
||||
|
|
15
doc/snac.8
15
doc/snac.8
|
@ -143,6 +143,14 @@ times the sending will be retried.
|
|||
The number of minutes to wait before the failed posting of a message is
|
||||
retried. This is not linear, but multipled by the number of retries
|
||||
already done.
|
||||
.It Ic queue_timeout
|
||||
The maximum number of seconds to wait when sending a message from the queue.
|
||||
.It Ic queue_timeout_2
|
||||
The maximum number of seconds to wait when sending a message from the queue
|
||||
to those servers that went timeout in the previous retry. If you want to
|
||||
give slow servers a chance to receive your messages, you can increase this
|
||||
value (but also take into account that processing the queue will take longer
|
||||
while waiting for these molasses to respond).
|
||||
.It Ic max_timeline_entries
|
||||
This is the maximum timeline entries shown in the web interface.
|
||||
.It Ic timeline_purge_days
|
||||
|
@ -209,6 +217,13 @@ with a large number of users.
|
|||
If this numeric value (in seconds) is set, any activity coming from an account
|
||||
that was created more recently than that will be rejected. This may be used
|
||||
to mitigate spam from automatically created accounts.
|
||||
.It Ic protocol
|
||||
This string value contains the protocol (schema) to be used in URLs. If not
|
||||
set, it defaults to "https". If you run
|
||||
.Nm
|
||||
as part of a hidden network like Tor or I2P that doesn't have a TLS /
|
||||
Certificate infrastructure, you need to set it to "http". Don't change it
|
||||
unless you know what you are doing.
|
||||
.El
|
||||
.Pp
|
||||
You must restart the server to make effective these changes.
|
||||
|
|
19
format.c
19
format.c
|
@ -82,7 +82,8 @@ static xs_str *format_line(const char *line, xs_list **attach)
|
|||
/* formats a line */
|
||||
{
|
||||
xs_str *s = xs_str_new(NULL);
|
||||
char *p, *v;
|
||||
char *p;
|
||||
const char *v;
|
||||
|
||||
/* split by markup */
|
||||
xs *sm = xs_regex_split(line,
|
||||
|
@ -155,7 +156,8 @@ xs_str *not_really_markdown(const char *content, xs_list **attach, xs_list **tag
|
|||
int in_pre = 0;
|
||||
int in_blq = 0;
|
||||
xs *list;
|
||||
char *p, *v;
|
||||
char *p;
|
||||
const char *v;
|
||||
|
||||
/* work by lines */
|
||||
list = xs_split(content, "\n");
|
||||
|
@ -234,14 +236,14 @@ xs_str *not_really_markdown(const char *content, xs_list **attach, xs_list **tag
|
|||
/* traditional emoticons */
|
||||
xs *d = emojis();
|
||||
int c = 0;
|
||||
char *k, *v;
|
||||
const char *k, *v;
|
||||
|
||||
while (xs_dict_next(d, &k, &v, &c)) {
|
||||
const char *t = NULL;
|
||||
|
||||
/* is it an URL to an image? */
|
||||
if (xs_startswith(v, "https:/" "/") && xs_startswith((t = xs_mime_by_ext(v)), "image/")) {
|
||||
if (tag) {
|
||||
if (tag && xs_str_in(s, k) != -1) {
|
||||
/* add the emoji to the tag list */
|
||||
xs *e = xs_dict_new();
|
||||
xs *i = xs_dict_new();
|
||||
|
@ -280,7 +282,8 @@ xs_str *sanitize(const char *content)
|
|||
xs_str *s = xs_str_new(NULL);
|
||||
xs *sl;
|
||||
int n = 0;
|
||||
char *p, *v;
|
||||
char *p;
|
||||
const char *v;
|
||||
|
||||
sl = xs_regex_split(content, "</?[^>]+>");
|
||||
|
||||
|
@ -311,9 +314,9 @@ xs_str *sanitize(const char *content)
|
|||
|
||||
s = xs_str_cat(s, s2);
|
||||
} else {
|
||||
/* else? just show it with encoded code.. that's it. */
|
||||
xs *el = encode_html(v);
|
||||
s = xs_str_cat(s, el);
|
||||
/* treat end of divs as paragraph breaks */
|
||||
if (strcmp(v, "</div>"))
|
||||
s = xs_str_cat(s, "<p>");
|
||||
}
|
||||
}
|
||||
else {
|
||||
|
|
36
http.c
36
http.c
|
@ -12,7 +12,7 @@
|
|||
|
||||
xs_dict *http_signed_request_raw(const char *keyid, const char *seckey,
|
||||
const char *method, const char *url,
|
||||
xs_dict *headers,
|
||||
const xs_dict *headers,
|
||||
const char *body, int b_size,
|
||||
int *status, xs_str **payload, int *p_size,
|
||||
int timeout)
|
||||
|
@ -24,15 +24,16 @@ xs_dict *http_signed_request_raw(const char *keyid, const char *seckey,
|
|||
xs *s64 = NULL;
|
||||
xs *signature = NULL;
|
||||
xs *hdrs = NULL;
|
||||
char *host;
|
||||
char *target;
|
||||
char *k, *v;
|
||||
const char *host;
|
||||
const char *target;
|
||||
const char *k, *v;
|
||||
xs_dict *response;
|
||||
|
||||
date = xs_str_utctime(0, "%a, %d %b %Y %H:%M:%S GMT");
|
||||
|
||||
{
|
||||
xs *s = xs_replace_n(url, "https:/" "/", "", 1);
|
||||
xs *s1 = xs_replace_n(url, "http:/" "/", "", 1);
|
||||
xs *s = xs_replace_n(s1, "https:/" "/", "", 1);
|
||||
l1 = xs_split_n(s, "/", 1);
|
||||
}
|
||||
|
||||
|
@ -105,13 +106,13 @@ xs_dict *http_signed_request_raw(const char *keyid, const char *seckey,
|
|||
|
||||
|
||||
xs_dict *http_signed_request(snac *snac, const char *method, const char *url,
|
||||
xs_dict *headers,
|
||||
const xs_dict *headers,
|
||||
const char *body, int b_size,
|
||||
int *status, xs_str **payload, int *p_size,
|
||||
int timeout)
|
||||
/* does a signed HTTP request */
|
||||
{
|
||||
char *seckey = xs_dict_get(snac->key, "secret");
|
||||
const char *seckey = xs_dict_get(snac->key, "secret");
|
||||
xs_dict *response;
|
||||
|
||||
response = http_signed_request_raw(snac->actor, seckey, method, url,
|
||||
|
@ -121,17 +122,18 @@ xs_dict *http_signed_request(snac *snac, const char *method, const char *url,
|
|||
}
|
||||
|
||||
|
||||
int check_signature(xs_dict *req, xs_str **err)
|
||||
int check_signature(const xs_dict *req, xs_str **err)
|
||||
/* check the signature */
|
||||
{
|
||||
char *sig_hdr = xs_dict_get(req, "signature");
|
||||
const char *sig_hdr = xs_dict_get(req, "signature");
|
||||
xs *keyId = NULL;
|
||||
xs *headers = NULL;
|
||||
xs *signature = NULL;
|
||||
xs *created = NULL;
|
||||
xs *expires = NULL;
|
||||
char *pubkey;
|
||||
char *p;
|
||||
const char *pubkey;
|
||||
const char *k;
|
||||
|
||||
if (xs_is_null(sig_hdr)) {
|
||||
*err = xs_fmt("missing 'signature' header");
|
||||
|
@ -141,10 +143,10 @@ int check_signature(xs_dict *req, xs_str **err)
|
|||
{
|
||||
/* extract the values */
|
||||
xs *l = xs_split(sig_hdr, ",");
|
||||
xs_list *p = l;
|
||||
xs_val *v;
|
||||
int c = 0;
|
||||
const xs_val *v;
|
||||
|
||||
while (xs_list_iter(&p, &v)) {
|
||||
while (xs_list_next(l, &v, &c)) {
|
||||
xs *kv = xs_split_n(v, "=", 1);
|
||||
|
||||
if (xs_list_len(kv) != 2)
|
||||
|
@ -191,8 +193,8 @@ int check_signature(xs_dict *req, xs_str **err)
|
|||
return 0;
|
||||
}
|
||||
|
||||
if ((p = xs_dict_get(actor, "publicKey")) == NULL ||
|
||||
((pubkey = xs_dict_get(p, "publicKeyPem")) == NULL)) {
|
||||
if ((k = xs_dict_get(actor, "publicKey")) == NULL ||
|
||||
((pubkey = xs_dict_get(k, "publicKeyPem")) == NULL)) {
|
||||
*err = xs_fmt("cannot get pubkey from %s", keyId);
|
||||
return 0;
|
||||
}
|
||||
|
@ -203,11 +205,11 @@ int check_signature(xs_dict *req, xs_str **err)
|
|||
{
|
||||
xs *l = xs_split(headers, " ");
|
||||
xs_list *p;
|
||||
xs_val *v;
|
||||
const xs_val *v;
|
||||
|
||||
p = l;
|
||||
while (xs_list_iter(&p, &v)) {
|
||||
char *hc;
|
||||
const char *hc;
|
||||
xs *ss = NULL;
|
||||
|
||||
if (*sig_str != '\0')
|
||||
|
|
66
httpd.c
66
httpd.c
|
@ -75,7 +75,7 @@ xs_str *nodeinfo_2_0(void)
|
|||
int n_posts = 0;
|
||||
xs *users = user_list();
|
||||
xs_list *p = users;
|
||||
char *v;
|
||||
const char *v;
|
||||
double now = (double)time(NULL);
|
||||
|
||||
while (xs_list_iter(&p, &v)) {
|
||||
|
@ -125,10 +125,10 @@ static xs_str *greeting_html(void)
|
|||
|
||||
/* does it have a %userlist% mark? */
|
||||
if (xs_str_in(s, "%userlist%") != -1) {
|
||||
char *host = xs_dict_get(srv_config, "host");
|
||||
const char *host = xs_dict_get(srv_config, "host");
|
||||
xs *list = user_list();
|
||||
xs_list *p = list;
|
||||
xs_str *uid;
|
||||
const xs_str *uid;
|
||||
|
||||
xs_html *ul = xs_html_tag("ul",
|
||||
xs_html_attr("class", "snac-user-list"));
|
||||
|
@ -169,18 +169,16 @@ int server_get_handler(xs_dict *req, const char *q_path,
|
|||
{
|
||||
int status = 0;
|
||||
|
||||
(void)req;
|
||||
|
||||
/* is it the server root? */
|
||||
if (*q_path == '\0') {
|
||||
xs_dict *q_vars = xs_dict_get(req, "q_vars");
|
||||
char *t = NULL;
|
||||
const xs_dict *q_vars = xs_dict_get(req, "q_vars");
|
||||
const char *t = NULL;
|
||||
|
||||
if (xs_type(q_vars) == XSTYPE_DICT && (t = xs_dict_get(q_vars, "t"))) {
|
||||
/** search by tag **/
|
||||
int skip = 0;
|
||||
int show = xs_number_get(xs_dict_get(srv_config, "max_timeline_entries"));
|
||||
char *v;
|
||||
const char *v;
|
||||
|
||||
if ((v = xs_dict_get(q_vars, "skip")) != NULL)
|
||||
skip = atoi(v);
|
||||
|
@ -195,13 +193,25 @@ int server_get_handler(xs_dict *req, const char *q_path,
|
|||
more = 1;
|
||||
}
|
||||
|
||||
*body = html_timeline(NULL, tl, 0, skip, show, more, t, NULL, 0);
|
||||
const char *accept = xs_dict_get(req, "accept");
|
||||
if (!xs_is_null(accept) && strcmp(accept, "application/rss+xml") == 0) {
|
||||
xs *link = xs_fmt("%s/?t=%s", srv_baseurl, t);
|
||||
|
||||
*body = timeline_to_rss(NULL, tl, link, link, link);
|
||||
*ctype = "application/rss+xml; charset=utf-8";
|
||||
}
|
||||
else {
|
||||
xs *page = xs_fmt("?t=%s", t);
|
||||
xs *title = xs_fmt(L("Search results for tag #%s"), t);
|
||||
*body = html_timeline(NULL, tl, 0, skip, show, more, title, page, 0);
|
||||
}
|
||||
}
|
||||
else
|
||||
if (xs_type(xs_dict_get(srv_config, "show_instance_timeline")) == XSTYPE_TRUE) {
|
||||
/** instance timeline **/
|
||||
xs *tl = timeline_instance_list(0, 30);
|
||||
*body = html_timeline(NULL, tl, 0, 0, 0, 0, NULL, NULL, 0);
|
||||
*body = html_timeline(NULL, tl, 0, 0, 0, 0,
|
||||
L("Recent posts by users in this instance"), NULL, 0);
|
||||
}
|
||||
else
|
||||
*body = greeting_html();
|
||||
|
@ -258,7 +268,7 @@ void httpd_connection(FILE *f)
|
|||
/* the connection processor */
|
||||
{
|
||||
xs *req;
|
||||
char *method;
|
||||
const char *method;
|
||||
int status = 0;
|
||||
xs_str *body = NULL;
|
||||
int b_size = 0;
|
||||
|
@ -268,7 +278,7 @@ void httpd_connection(FILE *f)
|
|||
xs *payload = NULL;
|
||||
xs *etag = NULL;
|
||||
int p_size = 0;
|
||||
char *p;
|
||||
const char *p;
|
||||
int fcgi_id;
|
||||
|
||||
if (p_state->use_fcgi)
|
||||
|
@ -360,7 +370,7 @@ void httpd_connection(FILE *f)
|
|||
#ifndef NO_MASTODON_API
|
||||
if (status == 0)
|
||||
status = mastoapi_delete_handler(req, q_path,
|
||||
&body, &b_size, &ctype);
|
||||
payload, p_size, &body, &b_size, &ctype);
|
||||
#endif
|
||||
}
|
||||
|
||||
|
@ -401,9 +411,9 @@ void httpd_connection(FILE *f)
|
|||
headers = xs_dict_append(headers, "etag", etag);
|
||||
|
||||
/* if there are any additional headers, add them */
|
||||
xs_dict *more_headers = xs_dict_get(srv_config, "http_headers");
|
||||
const xs_dict *more_headers = xs_dict_get(srv_config, "http_headers");
|
||||
if (xs_type(more_headers) == XSTYPE_DICT) {
|
||||
char *k, *v;
|
||||
const char *k, *v;
|
||||
int c = 0;
|
||||
while (xs_dict_next(more_headers, &k, &v, &c))
|
||||
headers = xs_dict_set(headers, k, v);
|
||||
|
@ -580,7 +590,8 @@ static void *background_thread(void *arg)
|
|||
|
||||
{
|
||||
xs *list = user_list();
|
||||
char *p, *uid;
|
||||
char *p;
|
||||
const char *uid;
|
||||
|
||||
/* process queues for all users */
|
||||
p = list;
|
||||
|
@ -654,6 +665,13 @@ srv_state *srv_state_op(xs_str **fname, int op)
|
|||
|
||||
switch (op) {
|
||||
case 0: /* open for writing */
|
||||
|
||||
#ifdef WITHOUT_SHM
|
||||
|
||||
errno = ENOTSUP;
|
||||
|
||||
#else
|
||||
|
||||
if ((fd = shm_open(*fname, O_CREAT | O_RDWR, 0666)) != -1) {
|
||||
ftruncate(fd, sizeof(*ss));
|
||||
|
||||
|
@ -664,6 +682,8 @@ srv_state *srv_state_op(xs_str **fname, int op)
|
|||
close(fd);
|
||||
}
|
||||
|
||||
#endif
|
||||
|
||||
if (ss == NULL) {
|
||||
/* shared memory error: just create a plain structure */
|
||||
srv_log(xs_fmt("warning: shm object error (%s)", strerror(errno)));
|
||||
|
@ -677,6 +697,13 @@ srv_state *srv_state_op(xs_str **fname, int op)
|
|||
break;
|
||||
|
||||
case 1: /* open for reading */
|
||||
|
||||
#ifdef WITHOUT_SHM
|
||||
|
||||
errno = ENOTSUP;
|
||||
|
||||
#else
|
||||
|
||||
if ((fd = shm_open(*fname, O_RDONLY, 0666)) != -1) {
|
||||
if ((ss = mmap(0, sizeof(*ss), PROT_READ, MAP_SHARED, fd, 0)) == MAP_FAILED)
|
||||
ss = NULL;
|
||||
|
@ -684,6 +711,8 @@ srv_state *srv_state_op(xs_str **fname, int op)
|
|||
close(fd);
|
||||
}
|
||||
|
||||
#endif
|
||||
|
||||
if (ss == NULL) {
|
||||
/* shared memory error */
|
||||
srv_log(xs_fmt("error: shm object error (%s) server not running?", strerror(errno)));
|
||||
|
@ -701,9 +730,14 @@ srv_state *srv_state_op(xs_str **fname, int op)
|
|||
break;
|
||||
|
||||
case 2: /* unlink */
|
||||
|
||||
#ifndef WITHOUT_SHM
|
||||
|
||||
if (*fname)
|
||||
shm_unlink(*fname);
|
||||
|
||||
#endif
|
||||
|
||||
break;
|
||||
}
|
||||
|
||||
|
|
26
main.c
26
main.c
|
@ -44,6 +44,7 @@ int usage(void)
|
|||
printf("limit {basedir} {uid} {actor} Limits an actor (drops their announces)\n");
|
||||
printf("unlimit {basedir} {uid} {actor} Unlimits an actor\n");
|
||||
printf("verify_links {basedir} {uid} Verifies a user's links (in the metadata)\n");
|
||||
printf("search {basedir} {uid} {regex} Searches posts by content\n");
|
||||
|
||||
return 1;
|
||||
}
|
||||
|
@ -314,7 +315,7 @@ int main(int argc, char *argv[])
|
|||
xs *msg = msg_follow(&snac, url);
|
||||
|
||||
if (msg != NULL) {
|
||||
char *actor = xs_dict_get(msg, "object");
|
||||
const char *actor = xs_dict_get(msg, "object");
|
||||
|
||||
following_add(&snac, actor, msg);
|
||||
|
||||
|
@ -374,6 +375,23 @@ int main(int argc, char *argv[])
|
|||
return 0;
|
||||
}
|
||||
|
||||
if (strcmp(cmd, "search") == 0) { /** **/
|
||||
int to;
|
||||
|
||||
/* 'url' contains the regex */
|
||||
xs *r = content_search(&snac, url, 1, 0, XS_ALL, 10, &to);
|
||||
|
||||
int c = 0;
|
||||
const char *v;
|
||||
|
||||
/* print results as standalone links */
|
||||
while (xs_list_next(r, &v, &c)) {
|
||||
printf("%s/admin/p/%s\n", snac.actor, v);
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
if (strcmp(cmd, "ping") == 0) { /** **/
|
||||
xs *actor_o = NULL;
|
||||
|
||||
|
@ -458,6 +476,12 @@ int main(int argc, char *argv[])
|
|||
return 0;
|
||||
}
|
||||
|
||||
if (strcmp(cmd, "request2") == 0) { /** **/
|
||||
enqueue_object_request(&snac, url, 2);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
if (strcmp(cmd, "actor") == 0) { /** **/
|
||||
int status;
|
||||
xs *data = NULL;
|
||||
|
|
495
mastoapi.c
495
mastoapi.c
|
@ -156,7 +156,7 @@ const char *login_page = ""
|
|||
"</head>\n"
|
||||
"<body><h1>%s OAuth identify</h1>\n"
|
||||
"<div style=\"background-color: red; color: white\">%s</div>\n"
|
||||
"<form method=\"post\" action=\"https:/" "/%s/%s\">\n"
|
||||
"<form method=\"post\" action=\"%s:/" "/%s/%s\">\n"
|
||||
"<p>Login: <input type=\"text\" name=\"login\"></p>\n"
|
||||
"<p>Password: <input type=\"password\" name=\"passwd\"></p>\n"
|
||||
"<input type=\"hidden\" name=\"redir\" value=\"%s\">\n"
|
||||
|
@ -175,7 +175,7 @@ int oauth_get_handler(const xs_dict *req, const char *q_path,
|
|||
return 0;
|
||||
|
||||
int status = 404;
|
||||
xs_dict *msg = xs_dict_get(req, "q_vars");
|
||||
const xs_dict *msg = xs_dict_get(req, "q_vars");
|
||||
xs *cmd = xs_replace_n(q_path, "/oauth", "", 1);
|
||||
|
||||
srv_debug(1, xs_fmt("oauth_get_handler %s", q_path));
|
||||
|
@ -193,11 +193,12 @@ int oauth_get_handler(const xs_dict *req, const char *q_path,
|
|||
|
||||
if (app != NULL) {
|
||||
const char *host = xs_dict_get(srv_config, "host");
|
||||
const char *proto = xs_dict_get_def(srv_config, "protocol", "https");
|
||||
|
||||
if (xs_is_null(state))
|
||||
state = "";
|
||||
|
||||
*body = xs_fmt(login_page, host, host, "", host, "oauth/x-snac-login",
|
||||
*body = xs_fmt(login_page, host, host, "", proto, host, "oauth/x-snac-login",
|
||||
ruri, cid, state, USER_AGENT);
|
||||
*ctype = "text/html";
|
||||
status = 200;
|
||||
|
@ -213,8 +214,9 @@ int oauth_get_handler(const xs_dict *req, const char *q_path,
|
|||
else
|
||||
if (strcmp(cmd, "/x-snac-get-token") == 0) { /** **/
|
||||
const char *host = xs_dict_get(srv_config, "host");
|
||||
const char *proto = xs_dict_get_def(srv_config, "protocol", "https");
|
||||
|
||||
*body = xs_fmt(login_page, host, host, "", host, "oauth/x-snac-get-token",
|
||||
*body = xs_fmt(login_page, host, host, "", proto, host, "oauth/x-snac-get-token",
|
||||
"", "", "", USER_AGENT);
|
||||
*ctype = "text/html";
|
||||
status = 200;
|
||||
|
@ -237,7 +239,7 @@ int oauth_post_handler(const xs_dict *req, const char *q_path,
|
|||
|
||||
int status = 404;
|
||||
|
||||
char *i_ctype = xs_dict_get(req, "content-type");
|
||||
const char *i_ctype = xs_dict_get(req, "content-type");
|
||||
xs *args = NULL;
|
||||
|
||||
if (i_ctype && xs_startswith(i_ctype, "application/json")) {
|
||||
|
@ -265,11 +267,11 @@ int oauth_post_handler(const xs_dict *req, const char *q_path,
|
|||
const char *redir = xs_dict_get(args, "redir");
|
||||
const char *cid = xs_dict_get(args, "cid");
|
||||
const char *state = xs_dict_get(args, "state");
|
||||
|
||||
const char *host = xs_dict_get(srv_config, "host");
|
||||
const char *host = xs_dict_get(srv_config, "host");
|
||||
const char *proto = xs_dict_get_def(srv_config, "protocol", "https");
|
||||
|
||||
/* by default, generate another login form with an error */
|
||||
*body = xs_fmt(login_page, host, host, "LOGIN INCORRECT", host, "oauth/x-snac-login",
|
||||
*body = xs_fmt(login_page, host, host, "LOGIN INCORRECT", proto, host, "oauth/x-snac-login",
|
||||
redir, cid, state, USER_AGENT);
|
||||
*ctype = "text/html";
|
||||
status = 200;
|
||||
|
@ -289,7 +291,11 @@ int oauth_post_handler(const xs_dict *req, const char *q_path,
|
|||
*body = xs_dup(code);
|
||||
}
|
||||
else {
|
||||
*body = xs_fmt("%s?code=%s", redir, code);
|
||||
if (xs_str_in(redir, "?") != -1)
|
||||
*body = xs_fmt("%s&code=%s", redir, code);
|
||||
else
|
||||
*body = xs_fmt("%s?code=%s", redir, code);
|
||||
|
||||
status = 303;
|
||||
}
|
||||
|
||||
|
@ -446,11 +452,11 @@ int oauth_post_handler(const xs_dict *req, const char *q_path,
|
|||
if (strcmp(cmd, "/x-snac-get-token") == 0) { /** **/
|
||||
const char *login = xs_dict_get(args, "login");
|
||||
const char *passwd = xs_dict_get(args, "passwd");
|
||||
|
||||
const char *host = xs_dict_get(srv_config, "host");
|
||||
const char *host = xs_dict_get(srv_config, "host");
|
||||
const char *proto = xs_dict_get_def(srv_config, "protocol", "https");
|
||||
|
||||
/* by default, generate another login form with an error */
|
||||
*body = xs_fmt(login_page, host, host, "LOGIN INCORRECT", host, "oauth/x-snac-get-token",
|
||||
*body = xs_fmt(login_page, host, host, "LOGIN INCORRECT", proto, host, "oauth/x-snac-get-token",
|
||||
"", "", "", USER_AGENT);
|
||||
*ctype = "text/html";
|
||||
status = 200;
|
||||
|
@ -544,6 +550,9 @@ xs_dict *mastoapi_account(const xs_dict *actor)
|
|||
acct = xs_dict_append(acct, "created_at", date);
|
||||
}
|
||||
|
||||
xs *last_status_at = xs_str_utctime(0, "%Y-%m-%d");
|
||||
acct = xs_dict_append(acct, "last_status_at", last_status_at);
|
||||
|
||||
const char *note = xs_dict_get(actor, "summary");
|
||||
if (xs_is_null(note))
|
||||
note = "";
|
||||
|
@ -559,10 +568,10 @@ xs_dict *mastoapi_account(const xs_dict *actor)
|
|||
acct = xs_dict_append(acct, "uri", id);
|
||||
|
||||
xs *avatar = NULL;
|
||||
xs_dict *av = xs_dict_get(actor, "icon");
|
||||
const xs_dict *av = xs_dict_get(actor, "icon");
|
||||
|
||||
if (xs_type(av) == XSTYPE_DICT) {
|
||||
char *url = xs_dict_get(av, "url");
|
||||
const char *url = xs_dict_get(av, "url");
|
||||
|
||||
if (url != NULL)
|
||||
avatar = xs_dup(url);
|
||||
|
@ -575,7 +584,7 @@ xs_dict *mastoapi_account(const xs_dict *actor)
|
|||
acct = xs_dict_append(acct, "avatar_static", avatar);
|
||||
|
||||
xs *header = NULL;
|
||||
xs_dict *hd = xs_dict_get(actor, "image");
|
||||
const xs_dict *hd = xs_dict_get(actor, "image");
|
||||
|
||||
if (xs_type(hd) == XSTYPE_DICT)
|
||||
header = xs_dup(xs_dict_get(hd, "url"));
|
||||
|
@ -587,12 +596,13 @@ xs_dict *mastoapi_account(const xs_dict *actor)
|
|||
acct = xs_dict_append(acct, "header_static", header);
|
||||
|
||||
/* emojis */
|
||||
xs_list *p;
|
||||
const xs_list *p;
|
||||
if (!xs_is_null(p = xs_dict_get(actor, "tag"))) {
|
||||
xs *eml = xs_list_new();
|
||||
xs_dict *v;
|
||||
const xs_dict *v;
|
||||
int c = 0;
|
||||
|
||||
while (xs_list_iter(&p, &v)) {
|
||||
while (xs_list_next(p, &v, &c)) {
|
||||
const char *type = xs_dict_get(v, "type");
|
||||
|
||||
if (!xs_is_null(type) && strcmp(type, "Emoji") == 0) {
|
||||
|
@ -627,11 +637,11 @@ xs_dict *mastoapi_account(const xs_dict *actor)
|
|||
|
||||
xs *fields = xs_list_new();
|
||||
p = xs_dict_get(actor, "attachment");
|
||||
xs_dict *v;
|
||||
const xs_dict *v;
|
||||
|
||||
/* dict of validated links */
|
||||
xs_dict *val_links = NULL;
|
||||
xs_dict *metadata = xs_stock(XSTYPE_DICT);
|
||||
const xs_dict *metadata = xs_stock(XSTYPE_DICT);
|
||||
snac user = {0};
|
||||
|
||||
if (xs_startswith(id, srv_baseurl)) {
|
||||
|
@ -645,19 +655,20 @@ xs_dict *mastoapi_account(const xs_dict *actor)
|
|||
if (xs_is_null(val_links))
|
||||
val_links = xs_stock(XSTYPE_DICT);
|
||||
|
||||
while (xs_list_iter(&p, &v)) {
|
||||
char *type = xs_dict_get(v, "type");
|
||||
char *name = xs_dict_get(v, "name");
|
||||
char *value = xs_dict_get(v, "value");
|
||||
int c = 0;
|
||||
while (xs_list_next(p, &v, &c)) {
|
||||
const char *type = xs_dict_get(v, "type");
|
||||
const char *name = xs_dict_get(v, "name");
|
||||
const char *value = xs_dict_get(v, "value");
|
||||
|
||||
if (!xs_is_null(type) && !xs_is_null(name) &&
|
||||
!xs_is_null(value) && strcmp(type, "PropertyValue") == 0) {
|
||||
xs *val_date = NULL;
|
||||
|
||||
char *url = xs_dict_get(metadata, name);
|
||||
const char *url = xs_dict_get(metadata, name);
|
||||
|
||||
if (!xs_is_null(url) && xs_startswith(url, "https:/" "/")) {
|
||||
xs_number *verified_time = xs_dict_get(val_links, url);
|
||||
const xs_number *verified_time = xs_dict_get(val_links, url);
|
||||
if (xs_type(verified_time) == XSTYPE_NUMBER) {
|
||||
time_t t = xs_number_get(verified_time);
|
||||
|
||||
|
@ -686,7 +697,7 @@ xs_dict *mastoapi_account(const xs_dict *actor)
|
|||
}
|
||||
|
||||
|
||||
xs_str *mastoapi_date(char *date)
|
||||
xs_str *mastoapi_date(const char *date)
|
||||
/* converts an ISO 8601 date to whatever format Mastodon uses */
|
||||
{
|
||||
xs_str *s = xs_crop_i(xs_dup(date), 0, 19);
|
||||
|
@ -701,16 +712,29 @@ xs_dict *mastoapi_poll(snac *snac, const xs_dict *msg)
|
|||
{
|
||||
xs_dict *poll = xs_dict_new();
|
||||
xs *mid = mastoapi_id(msg);
|
||||
xs_list *opts = NULL;
|
||||
xs_val *v;
|
||||
const xs_list *opts = NULL;
|
||||
const xs_val *v;
|
||||
int num_votes = 0;
|
||||
xs *options = xs_list_new();
|
||||
|
||||
poll = xs_dict_append(poll, "id", mid);
|
||||
xs *fd = mastoapi_date(xs_dict_get(msg, "endTime"));
|
||||
const char *date = xs_dict_get(msg, "endTime");
|
||||
if (date == NULL)
|
||||
date = xs_dict_get(msg, "closed");
|
||||
if (date == NULL)
|
||||
return NULL;
|
||||
|
||||
xs *fd = mastoapi_date(date);
|
||||
poll = xs_dict_append(poll, "expires_at", fd);
|
||||
|
||||
date = xs_dict_get(msg, "closed");
|
||||
time_t t = 0;
|
||||
|
||||
if (date != NULL)
|
||||
t = xs_parse_iso_date(date, 0);
|
||||
|
||||
poll = xs_dict_append(poll, "expired",
|
||||
xs_dict_get(msg, "closed") != NULL ? xs_stock(XSTYPE_TRUE) : xs_stock(XSTYPE_FALSE));
|
||||
t < time(NULL) ? xs_stock(XSTYPE_FALSE) : xs_stock(XSTYPE_TRUE));
|
||||
|
||||
if ((opts = xs_dict_get(msg, "oneOf")) != NULL)
|
||||
poll = xs_dict_append(poll, "multiple", xs_stock(XSTYPE_FALSE));
|
||||
|
@ -719,7 +743,8 @@ xs_dict *mastoapi_poll(snac *snac, const xs_dict *msg)
|
|||
poll = xs_dict_append(poll, "multiple", xs_stock(XSTYPE_TRUE));
|
||||
}
|
||||
|
||||
while (xs_list_iter(&opts, &v)) {
|
||||
int c = 0;
|
||||
while (xs_list_next(opts, &v, &c)) {
|
||||
const char *title = xs_dict_get(v, "name");
|
||||
const char *replies = xs_dict_get(v, "replies");
|
||||
|
||||
|
@ -772,7 +797,7 @@ xs_dict *mastoapi_status(snac *snac, const xs_dict *msg)
|
|||
|
||||
xs *idx = NULL;
|
||||
xs *ixc = NULL;
|
||||
char *tmp;
|
||||
const char *tmp;
|
||||
xs *mid = mastoapi_id(msg);
|
||||
|
||||
xs_dict *st = xs_dict_new();
|
||||
|
@ -824,14 +849,14 @@ xs_dict *mastoapi_status(snac *snac, const xs_dict *msg)
|
|||
|
||||
{
|
||||
xs_list *p = attach;
|
||||
xs_dict *v;
|
||||
const xs_dict *v;
|
||||
|
||||
xs *matt = xs_list_new();
|
||||
|
||||
while (xs_list_iter(&p, &v)) {
|
||||
char *type = xs_dict_get(v, "type");
|
||||
char *href = xs_dict_get(v, "href");
|
||||
char *name = xs_dict_get(v, "name");
|
||||
const char *type = xs_dict_get(v, "type");
|
||||
const char *href = xs_dict_get(v, "href");
|
||||
const char *name = xs_dict_get(v, "name");
|
||||
|
||||
if (xs_match(type, "image/*|video/*|Image|Video")) { /* */
|
||||
xs *matteid = xs_fmt("%s_%d", id, xs_list_len(matt));
|
||||
|
@ -857,7 +882,7 @@ xs_dict *mastoapi_status(snac *snac, const xs_dict *msg)
|
|||
xs *ml = xs_list_new();
|
||||
xs *htl = xs_list_new();
|
||||
xs *eml = xs_list_new();
|
||||
xs_list *tag = xs_dict_get(msg, "tag");
|
||||
const xs_list *tag = xs_dict_get(msg, "tag");
|
||||
int n = 0;
|
||||
|
||||
xs *tag_list = NULL;
|
||||
|
@ -873,9 +898,10 @@ xs_dict *mastoapi_status(snac *snac, const xs_dict *msg)
|
|||
tag_list = xs_list_new();
|
||||
|
||||
tag = tag_list;
|
||||
xs_dict *v;
|
||||
const xs_dict *v;
|
||||
|
||||
while (xs_list_iter(&tag, &v)) {
|
||||
int c = 0;
|
||||
while (xs_list_next(tag, &v, &c)) {
|
||||
const char *type = xs_dict_get(v, "type");
|
||||
|
||||
if (xs_is_null(type))
|
||||
|
@ -984,7 +1010,7 @@ xs_dict *mastoapi_status(snac *snac, const xs_dict *msg)
|
|||
xs *irt_mid = mastoapi_id(irto);
|
||||
st = xs_dict_set(st, "in_reply_to_id", irt_mid);
|
||||
|
||||
char *at = NULL;
|
||||
const char *at = NULL;
|
||||
if (!xs_is_null(at = get_atto(irto))) {
|
||||
xs *at_md5 = xs_md5_hex(at, strlen(at));
|
||||
st = xs_dict_set(st, "in_reply_to_account_id", at_md5);
|
||||
|
@ -994,7 +1020,10 @@ xs_dict *mastoapi_status(snac *snac, const xs_dict *msg)
|
|||
|
||||
st = xs_dict_append(st, "reblog", xs_stock(XSTYPE_NULL));
|
||||
st = xs_dict_append(st, "card", xs_stock(XSTYPE_NULL));
|
||||
st = xs_dict_append(st, "language", xs_stock(XSTYPE_NULL));
|
||||
st = xs_dict_append(st, "language", "en");
|
||||
|
||||
st = xs_dict_append(st, "filtered", xs_stock(XSTYPE_LIST));
|
||||
st = xs_dict_append(st, "muted", xs_stock(XSTYPE_FALSE));
|
||||
|
||||
tmp = xs_dict_get(msg, "sourceContent");
|
||||
if (xs_is_null(tmp))
|
||||
|
@ -1093,7 +1122,7 @@ int process_auth_token(snac *snac, const xs_dict *req)
|
|||
/* processes an authorization token, if there is one */
|
||||
{
|
||||
int logged_in = 0;
|
||||
char *v;
|
||||
const char *v;
|
||||
|
||||
/* if there is an authorization field, try to validate it */
|
||||
if (!xs_is_null(v = xs_dict_get(req, "authorization")) && xs_startswith(v, "Bearer ")) {
|
||||
|
@ -1131,7 +1160,7 @@ int mastoapi_get_handler(const xs_dict *req, const char *q_path,
|
|||
return 0;
|
||||
|
||||
int status = 404;
|
||||
xs_dict *args = xs_dict_get(req, "q_vars");
|
||||
const xs_dict *args = xs_dict_get(req, "q_vars");
|
||||
xs *cmd = xs_replace_n(q_path, "/api", "", 1);
|
||||
|
||||
snac snac1 = {0};
|
||||
|
@ -1157,7 +1186,7 @@ int mastoapi_get_handler(const xs_dict *req, const char *q_path,
|
|||
acct = xs_dict_append(acct, "source", src);
|
||||
|
||||
xs *avatar = NULL;
|
||||
char *av = xs_dict_get(snac1.config, "avatar");
|
||||
const char *av = xs_dict_get(snac1.config, "avatar");
|
||||
|
||||
if (xs_is_null(av) || *av == '\0')
|
||||
avatar = xs_fmt("%s/susie.png", srv_baseurl);
|
||||
|
@ -1168,7 +1197,7 @@ int mastoapi_get_handler(const xs_dict *req, const char *q_path,
|
|||
acct = xs_dict_append(acct, "avatar_static", avatar);
|
||||
|
||||
xs *header = NULL;
|
||||
char *hd = xs_dict_get(snac1.config, "header");
|
||||
const char *hd = xs_dict_get(snac1.config, "header");
|
||||
|
||||
if (!xs_is_null(hd))
|
||||
header = xs_dup(hd);
|
||||
|
@ -1178,11 +1207,11 @@ int mastoapi_get_handler(const xs_dict *req, const char *q_path,
|
|||
acct = xs_dict_append(acct, "header", header);
|
||||
acct = xs_dict_append(acct, "header_static", header);
|
||||
|
||||
xs_dict *metadata = xs_dict_get(snac1.config, "metadata");
|
||||
const xs_dict *metadata = xs_dict_get(snac1.config, "metadata");
|
||||
if (xs_type(metadata) == XSTYPE_DICT) {
|
||||
xs *fields = xs_list_new();
|
||||
xs_str *k;
|
||||
xs_str *v;
|
||||
const xs_str *k;
|
||||
const xs_str *v;
|
||||
|
||||
xs_dict *val_links = snac1.links;
|
||||
if (xs_is_null(val_links))
|
||||
|
@ -1192,7 +1221,7 @@ int mastoapi_get_handler(const xs_dict *req, const char *q_path,
|
|||
while (xs_dict_next(metadata, &k, &v, &c)) {
|
||||
xs *val_date = NULL;
|
||||
|
||||
xs_number *verified_time = xs_dict_get(val_links, v);
|
||||
const xs_number *verified_time = xs_dict_get(val_links, v);
|
||||
if (xs_type(verified_time) == XSTYPE_NUMBER) {
|
||||
time_t t = xs_number_get(verified_time);
|
||||
|
||||
|
@ -1258,13 +1287,13 @@ int mastoapi_get_handler(const xs_dict *req, const char *q_path,
|
|||
else
|
||||
if (strcmp(cmd, "/v1/accounts/lookup") == 0) { /** **/
|
||||
/* lookup an account */
|
||||
char *acct = xs_dict_get(args, "acct");
|
||||
const char *acct = xs_dict_get(args, "acct");
|
||||
|
||||
if (!xs_is_null(acct)) {
|
||||
xs *s = xs_strip_chars_i(xs_dup(acct), "@");
|
||||
xs *l = xs_split_n(s, "@", 1);
|
||||
char *uid = xs_list_get(l, 0);
|
||||
char *host = xs_list_get(l, 1);
|
||||
const char *uid = xs_list_get(l, 0);
|
||||
const char *host = xs_list_get(l, 1);
|
||||
|
||||
if (uid && (!host || strcmp(host, xs_dict_get(srv_config, "host")) == 0)) {
|
||||
snac user;
|
||||
|
@ -1305,7 +1334,7 @@ int mastoapi_get_handler(const xs_dict *req, const char *q_path,
|
|||
xs *wers = follower_list(&snac1);
|
||||
xs *ulst = user_list();
|
||||
xs_list *p;
|
||||
xs_str *v;
|
||||
const xs_str *v;
|
||||
xs_set seen;
|
||||
|
||||
xs_set_init(&seen);
|
||||
|
@ -1381,7 +1410,7 @@ int mastoapi_get_handler(const xs_dict *req, const char *q_path,
|
|||
/* the public list of posts of a user */
|
||||
xs *timeline = timeline_simple_list(&snac2, "public", 0, 256);
|
||||
xs_list *p = timeline;
|
||||
xs_str *v;
|
||||
const xs_str *v;
|
||||
|
||||
out = xs_list_new();
|
||||
|
||||
|
@ -1446,7 +1475,7 @@ int mastoapi_get_handler(const xs_dict *req, const char *q_path,
|
|||
|
||||
xs *out = xs_list_new();
|
||||
xs_list *p = timeline;
|
||||
xs_str *v;
|
||||
const xs_str *v;
|
||||
|
||||
while (xs_list_iter(&p, &v) && cnt < limit) {
|
||||
xs *msg = NULL;
|
||||
|
@ -1479,7 +1508,7 @@ int mastoapi_get_handler(const xs_dict *req, const char *q_path,
|
|||
/* discard non-Notes */
|
||||
const char *id = xs_dict_get(msg, "id");
|
||||
const char *type = xs_dict_get(msg, "type");
|
||||
if (!xs_match(type, "Note|Question|Page|Article|Video"))
|
||||
if (!xs_match(type, POSTLIKE_OBJECT_TYPE))
|
||||
continue;
|
||||
|
||||
const char *from = NULL;
|
||||
|
@ -1550,7 +1579,7 @@ int mastoapi_get_handler(const xs_dict *req, const char *q_path,
|
|||
xs *timeline = timeline_instance_list(0, limit);
|
||||
xs *out = xs_list_new();
|
||||
xs_list *p = timeline;
|
||||
xs_str *md5;
|
||||
const xs_str *md5;
|
||||
|
||||
snac *user = NULL;
|
||||
if (logged_in)
|
||||
|
@ -1599,12 +1628,12 @@ int mastoapi_get_handler(const xs_dict *req, const char *q_path,
|
|||
|
||||
/* get the tag */
|
||||
xs *l = xs_split(cmd, "/");
|
||||
char *tag = xs_list_get(l, -1);
|
||||
const char *tag = xs_list_get(l, -1);
|
||||
|
||||
xs *timeline = tag_search(tag, 0, limit);
|
||||
xs *out = xs_list_new();
|
||||
xs_list *p = timeline;
|
||||
xs_str *md5;
|
||||
const xs_str *md5;
|
||||
|
||||
while (xs_list_iter(&p, &md5) && cnt < limit) {
|
||||
xs *msg = NULL;
|
||||
|
@ -1635,6 +1664,77 @@ int mastoapi_get_handler(const xs_dict *req, const char *q_path,
|
|||
status = 200;
|
||||
}
|
||||
else
|
||||
if (xs_startswith(cmd, "/v1/timelines/list/")) { /** **/
|
||||
/* get the list id */
|
||||
if (logged_in) {
|
||||
xs *l = xs_split(cmd, "/");
|
||||
const char *list = xs_list_get(l, -1);
|
||||
|
||||
xs *timeline = list_timeline(&snac1, list, 0, 2048);
|
||||
xs *out = xs_list_new();
|
||||
int c = 0;
|
||||
const char *md5;
|
||||
|
||||
while (xs_list_next(timeline, &md5, &c)) {
|
||||
xs *msg = NULL;
|
||||
|
||||
/* get the entry */
|
||||
if (!valid_status(timeline_get_by_md5(&snac1, md5, &msg)))
|
||||
continue;
|
||||
|
||||
/* discard non-Notes */
|
||||
const char *id = xs_dict_get(msg, "id");
|
||||
const char *type = xs_dict_get(msg, "type");
|
||||
if (!xs_match(type, POSTLIKE_OBJECT_TYPE))
|
||||
continue;
|
||||
|
||||
const char *from = NULL;
|
||||
if (strcmp(type, "Page") == 0)
|
||||
from = xs_dict_get(msg, "audience");
|
||||
|
||||
if (from == NULL)
|
||||
from = get_atto(msg);
|
||||
|
||||
if (from == NULL)
|
||||
continue;
|
||||
|
||||
/* is this message from a person we don't follow? */
|
||||
if (strcmp(from, snac1.actor) && !following_check(&snac1, from)) {
|
||||
/* discard if it was not boosted */
|
||||
xs *idx = object_announces(id);
|
||||
|
||||
if (xs_list_len(idx) == 0)
|
||||
continue;
|
||||
}
|
||||
|
||||
/* discard notes from muted morons */
|
||||
if (is_muted(&snac1, from))
|
||||
continue;
|
||||
|
||||
/* discard hidden notes */
|
||||
if (is_hidden(&snac1, id))
|
||||
continue;
|
||||
|
||||
/* if it has a name and it's not a Page or a Video,
|
||||
it's a poll vote, so discard it */
|
||||
if (!xs_is_null(xs_dict_get(msg, "name")) && !xs_match(type, "Page|Video"))
|
||||
continue;
|
||||
|
||||
/* convert the Note into a Mastodon status */
|
||||
xs *st = mastoapi_status(&snac1, msg);
|
||||
|
||||
if (st != NULL)
|
||||
out = xs_list_append(out, st);
|
||||
}
|
||||
|
||||
*body = xs_json_dumps(out, 4);
|
||||
*ctype = "application/json";
|
||||
status = 200;
|
||||
}
|
||||
else
|
||||
status = 421;
|
||||
}
|
||||
else
|
||||
if (strcmp(cmd, "/v1/conversations") == 0) { /** **/
|
||||
/* TBD */
|
||||
*body = xs_dup("[]");
|
||||
|
@ -1647,8 +1747,8 @@ int mastoapi_get_handler(const xs_dict *req, const char *q_path,
|
|||
xs *l = notify_list(&snac1, 0, 64);
|
||||
xs *out = xs_list_new();
|
||||
xs_list *p = l;
|
||||
xs_dict *v;
|
||||
xs_list *excl = xs_dict_get(args, "exclude_types[]");
|
||||
const xs_dict *v;
|
||||
const xs_list *excl = xs_dict_get(args, "exclude_types[]");
|
||||
|
||||
while (xs_list_iter(&p, &v)) {
|
||||
xs *noti = notify_get(&snac1, v);
|
||||
|
@ -1753,11 +1853,88 @@ int mastoapi_get_handler(const xs_dict *req, const char *q_path,
|
|||
status = 200;
|
||||
}
|
||||
else
|
||||
if (strcmp(cmd, "/v1/lists") == 0) { /** **/
|
||||
/* snac does not support lists */
|
||||
*body = xs_dup("[]");
|
||||
*ctype = "application/json";
|
||||
status = 200;
|
||||
if (strcmp(cmd, "/v1/lists") == 0) { /** list of lists **/
|
||||
if (logged_in) {
|
||||
xs *lol = list_maint(&snac1, NULL, 0);
|
||||
xs *l = xs_list_new();
|
||||
int c = 0;
|
||||
const xs_list *li;
|
||||
|
||||
while (xs_list_next(lol, &li, &c)) {
|
||||
xs *d = xs_dict_new();
|
||||
|
||||
d = xs_dict_append(d, "id", xs_list_get(li, 0));
|
||||
d = xs_dict_append(d, "title", xs_list_get(li, 1));
|
||||
d = xs_dict_append(d, "replies_policy", "list");
|
||||
d = xs_dict_append(d, "exclusive", xs_stock(XSTYPE_FALSE));
|
||||
|
||||
l = xs_list_append(l, d);
|
||||
}
|
||||
|
||||
*body = xs_json_dumps(l, 4);
|
||||
*ctype = "application/json";
|
||||
status = 200;
|
||||
}
|
||||
}
|
||||
else
|
||||
if (xs_startswith(cmd, "/v1/lists/")) { /** list information **/
|
||||
if (logged_in) {
|
||||
xs *l = xs_split(cmd, "/");
|
||||
const char *p = xs_list_get(l, -1);
|
||||
|
||||
if (p) {
|
||||
if (strcmp(p, "accounts") == 0) {
|
||||
p = xs_list_get(l, -2);
|
||||
|
||||
if (p && xs_is_hex(p)) {
|
||||
xs *actors = list_content(&snac1, p, NULL, 0);
|
||||
xs *out = xs_list_new();
|
||||
int c = 0;
|
||||
const char *v;
|
||||
|
||||
while (xs_list_next(actors, &v, &c)) {
|
||||
xs *actor = NULL;
|
||||
|
||||
if (valid_status(object_get_by_md5(v, &actor))) {
|
||||
xs *acct = mastoapi_account(actor);
|
||||
out = xs_list_append(out, acct);
|
||||
}
|
||||
}
|
||||
|
||||
*body = xs_json_dumps(out, 4);
|
||||
*ctype = "application/json";
|
||||
status = 200;
|
||||
}
|
||||
}
|
||||
else
|
||||
if (xs_is_hex(p)) {
|
||||
xs *out = xs_list_new();
|
||||
xs *lol = list_maint(&snac1, NULL, 0);
|
||||
int c = 0;
|
||||
const xs_list *v;
|
||||
|
||||
while (xs_list_next(lol, &v, &c)) {
|
||||
const char *id = xs_list_get(v, 0);
|
||||
|
||||
if (id && strcmp(id, p) == 0) {
|
||||
xs *d = xs_dict_new();
|
||||
|
||||
d = xs_dict_append(d, "id", p);
|
||||
d = xs_dict_append(d, "title", xs_list_get(v, 1));
|
||||
d = xs_dict_append(d, "replies_policy", "list");
|
||||
d = xs_dict_append(d, "exclusive", xs_stock(XSTYPE_FALSE));
|
||||
|
||||
out = xs_list_append(out, d);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
*body = xs_json_dumps(out, 4);
|
||||
*ctype = "application/json";
|
||||
status = 200;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
else
|
||||
if (strcmp(cmd, "/v1/scheduled_statuses") == 0) { /** **/
|
||||
|
@ -1928,7 +2105,7 @@ int mastoapi_get_handler(const xs_dict *req, const char *q_path,
|
|||
xs *anc = xs_list_new();
|
||||
xs *des = xs_list_new();
|
||||
xs_list *p;
|
||||
xs_str *v;
|
||||
const xs_str *v;
|
||||
char pid[64];
|
||||
|
||||
/* build the [grand]parent list, moving up */
|
||||
|
@ -1982,7 +2159,7 @@ int mastoapi_get_handler(const xs_dict *req, const char *q_path,
|
|||
l = object_likes(xs_dict_get(msg, "id"));
|
||||
|
||||
xs_list *p = l;
|
||||
xs_str *v;
|
||||
const xs_str *v;
|
||||
|
||||
while (xs_list_iter(&p, &v)) {
|
||||
xs *actor2 = NULL;
|
||||
|
@ -2052,8 +2229,8 @@ int mastoapi_get_handler(const xs_dict *req, const char *q_path,
|
|||
/* reply something only for offset 0; otherwise,
|
||||
apps like Tusky keep asking again and again */
|
||||
|
||||
if (!xs_is_null(q) && !xs_is_null(type)) {
|
||||
if (strcmp(type, "accounts") == 0) {
|
||||
if (!xs_is_null(q)) {
|
||||
if (xs_is_null(type) || strcmp(type, "accounts") == 0) {
|
||||
/* do a webfinger query */
|
||||
char *actor = NULL;
|
||||
char *user = NULL;
|
||||
|
@ -2068,8 +2245,8 @@ int mastoapi_get_handler(const xs_dict *req, const char *q_path,
|
|||
}
|
||||
}
|
||||
}
|
||||
else
|
||||
if (strcmp(type, "hashtags") == 0) {
|
||||
|
||||
if (xs_is_null(type) || strcmp(type, "hashtags") == 0) {
|
||||
/* search this tag */
|
||||
xs *tl = tag_search((char *)q, 0, 1);
|
||||
|
||||
|
@ -2084,6 +2261,26 @@ int mastoapi_get_handler(const xs_dict *req, const char *q_path,
|
|||
htl = xs_list_append(htl, d);
|
||||
}
|
||||
}
|
||||
|
||||
if (xs_is_null(type) || strcmp(type, "statuses") == 0) {
|
||||
int to = 0;
|
||||
int cnt = 40;
|
||||
xs *tl = content_search(&snac1, q, 1, 0, cnt, 0, &to);
|
||||
int c = 0;
|
||||
const char *v;
|
||||
|
||||
while (xs_list_next(tl, &v, &c) && --cnt) {
|
||||
xs *post = NULL;
|
||||
|
||||
if (!valid_status(timeline_get_by_md5(&snac1, v, &post)))
|
||||
continue;
|
||||
|
||||
xs *s = mastoapi_status(&snac1, post);
|
||||
|
||||
if (!xs_is_null(s))
|
||||
stl = xs_list_append(stl, s);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -2119,11 +2316,9 @@ int mastoapi_post_handler(const xs_dict *req, const char *q_path,
|
|||
if (!xs_startswith(q_path, "/api/v1/") && !xs_startswith(q_path, "/api/v2/"))
|
||||
return 0;
|
||||
|
||||
srv_debug(1, xs_fmt("mastoapi_post_handler %s", q_path));
|
||||
|
||||
int status = 404;
|
||||
xs *args = NULL;
|
||||
char *i_ctype = xs_dict_get(req, "content-type");
|
||||
const char *i_ctype = xs_dict_get(req, "content-type");
|
||||
|
||||
if (i_ctype && xs_startswith(i_ctype, "application/json")) {
|
||||
if (!xs_is_null(payload))
|
||||
|
@ -2238,7 +2433,7 @@ int mastoapi_post_handler(const xs_dict *req, const char *q_path,
|
|||
}
|
||||
|
||||
xs_list *p = mi;
|
||||
xs_str *v;
|
||||
const xs_str *v;
|
||||
|
||||
while (xs_list_iter(&p, &v)) {
|
||||
xs *l = xs_list_new();
|
||||
|
@ -2296,7 +2491,7 @@ int mastoapi_post_handler(const xs_dict *req, const char *q_path,
|
|||
mid = MID_TO_MD5(mid);
|
||||
|
||||
if (valid_status(timeline_get_by_md5(&snac, mid, &msg))) {
|
||||
char *id = xs_dict_get(msg, "id");
|
||||
const char *id = xs_dict_get(msg, "id");
|
||||
|
||||
if (op == NULL) {
|
||||
/* no operation (?) */
|
||||
|
@ -2402,7 +2597,7 @@ int mastoapi_post_handler(const xs_dict *req, const char *q_path,
|
|||
if (strcmp(cmd, "/v1/push/subscription") == 0) { /** **/
|
||||
/* I don't know what I'm doing */
|
||||
if (logged_in) {
|
||||
char *v;
|
||||
const char *v;
|
||||
|
||||
xs *wpush = xs_dict_new();
|
||||
|
||||
|
@ -2574,7 +2769,7 @@ int mastoapi_post_handler(const xs_dict *req, const char *q_path,
|
|||
const char *id = xs_dict_get(msg, "id");
|
||||
const char *atto = get_atto(msg);
|
||||
|
||||
xs_list *opts = xs_dict_get(msg, "oneOf");
|
||||
const xs_list *opts = xs_dict_get(msg, "oneOf");
|
||||
if (opts == NULL)
|
||||
opts = xs_dict_get(msg, "anyOf");
|
||||
|
||||
|
@ -2582,15 +2777,16 @@ int mastoapi_post_handler(const xs_dict *req, const char *q_path,
|
|||
}
|
||||
else
|
||||
if (strcmp(op, "votes") == 0) {
|
||||
xs_list *choices = xs_dict_get(args, "choices[]");
|
||||
const xs_list *choices = xs_dict_get(args, "choices[]");
|
||||
|
||||
if (xs_is_null(choices))
|
||||
choices = xs_dict_get(args, "choices");
|
||||
|
||||
if (xs_type(choices) == XSTYPE_LIST) {
|
||||
xs_str *v;
|
||||
const xs_str *v;
|
||||
|
||||
while (xs_list_iter(&choices, &v)) {
|
||||
int c = 0;
|
||||
while (xs_list_next(choices, &v, &c)) {
|
||||
int io = atoi(v);
|
||||
const xs_dict *o = xs_list_get(opts, io);
|
||||
|
||||
|
@ -2621,19 +2817,73 @@ int mastoapi_post_handler(const xs_dict *req, const char *q_path,
|
|||
else
|
||||
status = 401;
|
||||
}
|
||||
else
|
||||
if (strcmp(cmd, "/v1/lists") == 0) {
|
||||
if (logged_in) {
|
||||
const char *title = xs_dict_get(args, "title");
|
||||
|
||||
if (xs_type(title) == XSTYPE_STRING) {
|
||||
/* add the list */
|
||||
xs *out = xs_dict_new();
|
||||
|
||||
if (xs_type(list_maint(&snac, title, 1)) == XSTYPE_TRUE) {
|
||||
out = xs_dict_append(out, "title", title);
|
||||
out = xs_dict_append(out, "replies_policy", xs_dict_get_def(args, "replies_policy", "list"));
|
||||
out = xs_dict_append(out, "exclusive", xs_stock(XSTYPE_FALSE));
|
||||
|
||||
status = 200;
|
||||
}
|
||||
else {
|
||||
out = xs_dict_append(out, "error", "cannot create list");
|
||||
status = 422;
|
||||
}
|
||||
|
||||
*body = xs_json_dumps(out, 4);
|
||||
*ctype = "application/json";
|
||||
}
|
||||
else
|
||||
status = 422;
|
||||
}
|
||||
}
|
||||
if (xs_startswith(cmd, "/v1/lists/")) { /** list maintenance **/
|
||||
if (logged_in) {
|
||||
xs *l = xs_split(cmd, "/");
|
||||
const char *op = xs_list_get(l, -1);
|
||||
const char *id = xs_list_get(l, -2);
|
||||
|
||||
if (op && id && xs_is_hex(id)) {
|
||||
if (strcmp(op, "accounts") == 0) {
|
||||
const xs_list *accts = xs_dict_get(args, "account_ids[]");
|
||||
int c = 0;
|
||||
const char *v;
|
||||
|
||||
while (xs_list_next(accts, &v, &c)) {
|
||||
list_content(&snac, id, v, 1);
|
||||
}
|
||||
|
||||
status = 200;
|
||||
}
|
||||
}
|
||||
}
|
||||
else
|
||||
status = 422;
|
||||
}
|
||||
|
||||
/* user cleanup */
|
||||
if (logged_in)
|
||||
user_free(&snac);
|
||||
|
||||
srv_debug(1, xs_fmt("mastoapi_post_handler %s %d", q_path, status));
|
||||
|
||||
return status;
|
||||
}
|
||||
|
||||
|
||||
int mastoapi_delete_handler(const xs_dict *req, const char *q_path,
|
||||
char **body, int *b_size, char **ctype) {
|
||||
|
||||
(void)req;
|
||||
const char *payload, int p_size,
|
||||
char **body, int *b_size, char **ctype)
|
||||
{
|
||||
(void)p_size;
|
||||
(void)body;
|
||||
(void)b_size;
|
||||
(void)ctype;
|
||||
|
@ -2641,13 +2891,76 @@ int mastoapi_delete_handler(const xs_dict *req, const char *q_path,
|
|||
if (!xs_startswith(q_path, "/api/v1/") && !xs_startswith(q_path, "/api/v2/"))
|
||||
return 0;
|
||||
|
||||
srv_debug(1, xs_fmt("mastoapi_delete_handler %s", q_path));
|
||||
int status = 404;
|
||||
xs *args = NULL;
|
||||
const char *i_ctype = xs_dict_get(req, "content-type");
|
||||
|
||||
if (i_ctype && xs_startswith(i_ctype, "application/json")) {
|
||||
if (!xs_is_null(payload))
|
||||
args = xs_json_loads(payload);
|
||||
}
|
||||
else if (i_ctype && xs_startswith(i_ctype, "application/x-www-form-urlencoded"))
|
||||
{
|
||||
// Some apps send form data instead of json so we should cater for those
|
||||
if (!xs_is_null(payload)) {
|
||||
xs *upl = xs_url_dec(payload);
|
||||
args = xs_url_vars(upl);
|
||||
}
|
||||
}
|
||||
else
|
||||
args = xs_dup(xs_dict_get(req, "p_vars"));
|
||||
|
||||
if (args == NULL)
|
||||
return 400;
|
||||
|
||||
snac snac = {0};
|
||||
int logged_in = process_auth_token(&snac, req);
|
||||
|
||||
xs *cmd = xs_replace_n(q_path, "/api", "", 1);
|
||||
|
||||
if (xs_startswith(cmd, "/v1/push/subscription") || xs_startswith(cmd, "/v2/push/subscription")) { /** **/
|
||||
// pretend we deleted it, since it doesn't exist anyway
|
||||
return 200;
|
||||
status = 200;
|
||||
}
|
||||
return 0;
|
||||
else
|
||||
if (xs_startswith(cmd, "/v1/lists/")) {
|
||||
if (logged_in) {
|
||||
xs *l = xs_split(cmd, "/");
|
||||
const char *p = xs_list_get(l, -1);
|
||||
|
||||
if (p) {
|
||||
if (strcmp(p, "accounts") == 0) {
|
||||
/* delete account from list */
|
||||
p = xs_list_get(l, -2);
|
||||
const xs_list *accts = xs_dict_get(args, "account_ids[]");
|
||||
int c = 0;
|
||||
const char *v;
|
||||
|
||||
while (xs_list_next(accts, &v, &c)) {
|
||||
list_content(&snac, p, v, 2);
|
||||
}
|
||||
}
|
||||
else {
|
||||
/* delete list */
|
||||
if (xs_is_hex(p)) {
|
||||
list_maint(&snac, p, 2);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
status = 200;
|
||||
}
|
||||
else
|
||||
status = 401;
|
||||
}
|
||||
|
||||
/* user cleanup */
|
||||
if (logged_in)
|
||||
user_free(&snac);
|
||||
|
||||
srv_debug(1, xs_fmt("mastoapi_delete_handler %s %d", q_path, status));
|
||||
|
||||
return status;
|
||||
}
|
||||
|
||||
|
||||
|
@ -2663,7 +2976,7 @@ int mastoapi_put_handler(const xs_dict *req, const char *q_path,
|
|||
|
||||
int status = 404;
|
||||
xs *args = NULL;
|
||||
char *i_ctype = xs_dict_get(req, "content-type");
|
||||
const char *i_ctype = xs_dict_get(req, "content-type");
|
||||
|
||||
if (i_ctype && xs_startswith(i_ctype, "application/json")) {
|
||||
if (!xs_is_null(payload))
|
||||
|
@ -2770,7 +3083,7 @@ void mastoapi_purge(void)
|
|||
xs *spec = xs_fmt("%s/app/" "*.json", srv_basedir);
|
||||
xs *files = xs_glob(spec, 1, 0);
|
||||
xs_list *p = files;
|
||||
xs_str *v;
|
||||
const xs_str *v;
|
||||
|
||||
time_t mt = time(NULL) - 3600;
|
||||
|
||||
|
|
69
snac.h
69
snac.h
|
@ -1,7 +1,7 @@
|
|||
/* snac - A simple, minimalistic ActivityPub instance */
|
||||
/* copyright (c) 2022 - 2024 grunfink et al. / MIT license */
|
||||
|
||||
#define VERSION "2.52-dev"
|
||||
#define VERSION "2.53"
|
||||
|
||||
#define USER_AGENT "snac/" VERSION
|
||||
|
||||
|
@ -29,6 +29,8 @@ extern int dbglevel;
|
|||
|
||||
#define L(s) (s)
|
||||
|
||||
#define POSTLIKE_OBJECT_TYPE "Note|Question|Page|Article|Video|Event"
|
||||
|
||||
int mkdirx(const char *pathname);
|
||||
|
||||
int valid_status(int status);
|
||||
|
@ -67,7 +69,7 @@ void snac_log(snac *user, xs_str *str);
|
|||
#define snac_debug(user, level, str) do { if (dbglevel >= (level)) \
|
||||
{ snac_log((user), (str)); } } while (0)
|
||||
|
||||
int srv_open(char *basedir, int auto_upgrade);
|
||||
int srv_open(const char *basedir, int auto_upgrade);
|
||||
void srv_free(void);
|
||||
|
||||
int user_open(snac *snac, const char *uid);
|
||||
|
@ -86,7 +88,7 @@ void srv_archive(const char *direction, const char *url, xs_dict *req,
|
|||
const char *body, int b_size);
|
||||
void srv_archive_error(const char *prefix, const xs_str *err,
|
||||
const xs_dict *req, const xs_val *data);
|
||||
void srv_archive_qitem(char *prefix, xs_dict *q_item);
|
||||
void srv_archive_qitem(const char *prefix, xs_dict *q_item);
|
||||
|
||||
double mtime_nl(const char *fn, int *n_link);
|
||||
#define mtime(fn) mtime_nl(fn, NULL)
|
||||
|
@ -137,13 +139,13 @@ double timeline_mtime(snac *snac);
|
|||
int timeline_touch(snac *snac);
|
||||
int timeline_here(snac *snac, const char *md5);
|
||||
int timeline_get_by_md5(snac *snac, const char *md5, xs_dict **msg);
|
||||
int timeline_del(snac *snac, char *id);
|
||||
int timeline_del(snac *snac, const char *id);
|
||||
xs_list *timeline_simple_list(snac *snac, const char *idx_name, int skip, int show);
|
||||
xs_list *timeline_list(snac *snac, const char *idx_name, int skip, int show);
|
||||
int timeline_add(snac *snac, const char *id, const xs_dict *o_msg);
|
||||
int timeline_admire(snac *snac, const char *id, const char *admirer, int like);
|
||||
|
||||
xs_list *timeline_top_level(snac *snac, xs_list *list);
|
||||
xs_list *timeline_top_level(snac *snac, const xs_list *list);
|
||||
xs_list *local_list(snac *snac, int max);
|
||||
xs_list *timeline_instance_list(int skip, int show);
|
||||
|
||||
|
@ -172,9 +174,14 @@ void hide(snac *snac, const char *id);
|
|||
int is_hidden(snac *snac, const char *id);
|
||||
|
||||
void tag_index(const char *id, const xs_dict *obj);
|
||||
xs_list *tag_search(char *tag, int skip, int show);
|
||||
xs_list *tag_search(const char *tag, int skip, int show);
|
||||
|
||||
int actor_add(const char *actor, xs_dict *msg);
|
||||
xs_val *list_maint(snac *user, const char *list, int op);
|
||||
xs_list *list_timeline(snac *user, const char *list, int skip, int show);
|
||||
xs_val *list_content(snac *user, const char *list_id, const char *actor_md5, int op);
|
||||
void list_distribute(snac *user, const char *who, const xs_dict *post);
|
||||
|
||||
int actor_add(const char *actor, const xs_dict *msg);
|
||||
int actor_get(const char *actor, xs_dict **data);
|
||||
int actor_get_refresh(snac *user, const char *actor, xs_dict **data);
|
||||
|
||||
|
@ -209,22 +216,27 @@ int is_instance_blocked(const char *instance);
|
|||
int instance_block(const char *instance);
|
||||
int instance_unblock(const char *instance);
|
||||
|
||||
int content_check(const char *file, const xs_dict *msg);
|
||||
int content_match(const char *file, const xs_dict *msg);
|
||||
xs_list *content_search(snac *user, const char *regex,
|
||||
int priv, int skip, int show, int max_secs, int *timeout);
|
||||
|
||||
void enqueue_input(snac *snac, const xs_dict *msg, const xs_dict *req, int retries);
|
||||
void enqueue_shared_input(const xs_dict *msg, const xs_dict *req, int retries);
|
||||
void enqueue_output_raw(const char *keyid, const char *seckey,
|
||||
xs_dict *msg, xs_str *inbox, int retries, int p_status);
|
||||
void enqueue_output(snac *snac, xs_dict *msg, xs_str *inbox, int retries, int p_status);
|
||||
void enqueue_output_by_actor(snac *snac, xs_dict *msg, const xs_str *actor, int retries);
|
||||
void enqueue_email(xs_str *msg, int retries);
|
||||
const xs_dict *msg, const xs_str *inbox,
|
||||
int retries, int p_status);
|
||||
void enqueue_output(snac *snac, const xs_dict *msg,
|
||||
const xs_str *inbox, int retries, int p_status);
|
||||
void enqueue_output_by_actor(snac *snac, const xs_dict *msg,
|
||||
const xs_str *actor, int retries);
|
||||
void enqueue_email(const xs_str *msg, int retries);
|
||||
void enqueue_telegram(const xs_str *msg, const char *bot, const char *chat_id);
|
||||
void enqueue_ntfy(const xs_str *msg, const char *ntfy_server, const char *ntfy_token);
|
||||
void enqueue_message(snac *snac, const xs_dict *msg);
|
||||
void enqueue_close_question(snac *user, const char *id, int end_secs);
|
||||
void enqueue_object_request(snac *user, const char *id, int forward_secs);
|
||||
void enqueue_verify_links(snac *user);
|
||||
void enqueue_actor_refresh(snac *user, const char *actor);
|
||||
void enqueue_request_replies(snac *user, const char *id);
|
||||
void enqueue_actor_refresh(snac *user, const char *actor, int forward_secs);
|
||||
int was_question_voted(snac *user, const char *id);
|
||||
|
||||
xs_list *user_queue(snac *snac);
|
||||
|
@ -237,16 +249,16 @@ void purge_all(void);
|
|||
|
||||
xs_dict *http_signed_request_raw(const char *keyid, const char *seckey,
|
||||
const char *method, const char *url,
|
||||
xs_dict *headers,
|
||||
const xs_dict *headers,
|
||||
const char *body, int b_size,
|
||||
int *status, xs_str **payload, int *p_size,
|
||||
int timeout);
|
||||
xs_dict *http_signed_request(snac *snac, const char *method, const char *url,
|
||||
xs_dict *headers,
|
||||
const xs_dict *headers,
|
||||
const char *body, int b_size,
|
||||
int *status, xs_str **payload, int *p_size,
|
||||
int timeout);
|
||||
int check_signature(xs_dict *req, xs_str **err);
|
||||
int check_signature(const xs_dict *req, xs_str **err);
|
||||
|
||||
srv_state *srv_state_op(xs_str **fname, int op);
|
||||
void httpd(void);
|
||||
|
@ -260,21 +272,21 @@ const char *default_avatar_base64(void);
|
|||
|
||||
xs_str *process_tags(snac *snac, const char *content, xs_list **tag);
|
||||
|
||||
char *get_atto(const xs_dict *msg);
|
||||
const char *get_atto(const xs_dict *msg);
|
||||
xs_list *get_attachments(const xs_dict *msg);
|
||||
|
||||
xs_dict *msg_admiration(snac *snac, char *object, char *type);
|
||||
xs_dict *msg_repulsion(snac *user, char *id, char *type);
|
||||
xs_dict *msg_admiration(snac *snac, const char *object, const char *type);
|
||||
xs_dict *msg_repulsion(snac *user, const char *id, const char *type);
|
||||
xs_dict *msg_create(snac *snac, const xs_dict *object);
|
||||
xs_dict *msg_follow(snac *snac, const char *actor);
|
||||
|
||||
xs_dict *msg_note(snac *snac, const xs_str *content, const xs_val *rcpts,
|
||||
xs_str *in_reply_to, xs_list *attach, int priv);
|
||||
const xs_str *in_reply_to, const xs_list *attach, int priv);
|
||||
|
||||
xs_dict *msg_undo(snac *snac, char *object);
|
||||
xs_dict *msg_delete(snac *snac, char *id);
|
||||
xs_dict *msg_undo(snac *snac, const xs_val *object);
|
||||
xs_dict *msg_delete(snac *snac, const char *id);
|
||||
xs_dict *msg_actor(snac *snac);
|
||||
xs_dict *msg_update(snac *snac, xs_dict *object);
|
||||
xs_dict *msg_update(snac *snac, const xs_dict *object);
|
||||
xs_dict *msg_ping(snac *user, const char *rcpt);
|
||||
xs_dict *msg_pong(snac *user, const char *rcpt, const char *object);
|
||||
xs_dict *msg_question(snac *user, const char *content, xs_list *attach,
|
||||
|
@ -282,7 +294,6 @@ xs_dict *msg_question(snac *user, const char *content, xs_list *attach,
|
|||
|
||||
int activitypub_request(snac *snac, const char *url, xs_dict **data);
|
||||
int actor_request(snac *user, const char *actor, xs_dict **data);
|
||||
void timeline_request_replies(snac *user, const char *id);
|
||||
int send_to_inbox_raw(const char *keyid, const char *seckey,
|
||||
const xs_str *inbox, const xs_dict *msg,
|
||||
xs_val **payload, int *p_size, int timeout);
|
||||
|
@ -312,13 +323,14 @@ xs_str *encode_html(const char *str);
|
|||
|
||||
xs_str *html_timeline(snac *user, const xs_list *list, int read_only,
|
||||
int skip, int show, int show_more,
|
||||
char *tag, char *page, int utl);
|
||||
char *title, char *page, int utl);
|
||||
|
||||
int html_get_handler(const xs_dict *req, const char *q_path,
|
||||
char **body, int *b_size, char **ctype, xs_str **etag);
|
||||
int html_post_handler(const xs_dict *req, const char *q_path,
|
||||
char *payload, int p_size,
|
||||
char **body, int *b_size, char **ctype);
|
||||
xs_str *timeline_to_rss(snac *user, const xs_list *timeline, char *title, char *link, char *desc);
|
||||
|
||||
int snac_init(const char *_basedir);
|
||||
int adduser(const char *uid);
|
||||
|
@ -337,11 +349,12 @@ int oauth_post_handler(const xs_dict *req, const char *q_path,
|
|||
char **body, int *b_size, char **ctype);
|
||||
int mastoapi_get_handler(const xs_dict *req, const char *q_path,
|
||||
char **body, int *b_size, char **ctype);
|
||||
int mastoapi_delete_handler(const xs_dict *req, const char *q_path,
|
||||
char **body, int *b_size, char **ctype);
|
||||
int mastoapi_post_handler(const xs_dict *req, const char *q_path,
|
||||
const char *payload, int p_size,
|
||||
char **body, int *b_size, char **ctype);
|
||||
int mastoapi_delete_handler(const xs_dict *req, const char *q_path,
|
||||
const char *payload, int p_size,
|
||||
char **body, int *b_size, char **ctype);
|
||||
int mastoapi_put_handler(const xs_dict *req, const char *q_path,
|
||||
const char *payload, int p_size,
|
||||
char **body, int *b_size, char **ctype);
|
||||
|
|
73
upgrade.c
73
upgrade.c
|
@ -18,7 +18,7 @@ int snac_upgrade(xs_str **error)
|
|||
double f = 0.0;
|
||||
|
||||
for (;;) {
|
||||
char *layout = xs_dict_get(srv_config, "layout");
|
||||
const char *layout = xs_dict_get(srv_config, "layout");
|
||||
double nf;
|
||||
|
||||
f = nf = xs_number_get(layout);
|
||||
|
@ -43,7 +43,8 @@ int snac_upgrade(xs_str **error)
|
|||
else
|
||||
if (f < 2.2) {
|
||||
xs *users = user_list();
|
||||
char *p, *v;
|
||||
char *p;
|
||||
const char *v;
|
||||
|
||||
p = users;
|
||||
while (xs_list_iter(&p, &v)) {
|
||||
|
@ -52,12 +53,13 @@ int snac_upgrade(xs_str **error)
|
|||
if (user_open(&snac, v)) {
|
||||
xs *spec = xs_fmt("%s/actors/" "*.json", snac.basedir);
|
||||
xs *list = xs_glob(spec, 0, 0);
|
||||
char *g, *fn;
|
||||
char *g;
|
||||
const char *fn;
|
||||
|
||||
g = list;
|
||||
while (xs_list_iter(&g, &fn)) {
|
||||
xs *l = xs_split(fn, "/");
|
||||
char *b = xs_list_get(l, -1);
|
||||
xs *l = xs_split(fn, "/");
|
||||
const char *b = xs_list_get(l, -1);
|
||||
xs *dir = xs_fmt("%s/object/%c%c", srv_basedir, b[0], b[1]);
|
||||
xs *nfn = xs_fmt("%s/%s", dir, b);
|
||||
|
||||
|
@ -77,14 +79,16 @@ int snac_upgrade(xs_str **error)
|
|||
else
|
||||
if (f < 2.3) {
|
||||
xs *users = user_list();
|
||||
char *p, *v;
|
||||
char *p;
|
||||
const char *v;
|
||||
|
||||
p = users;
|
||||
while (xs_list_iter(&p, &v)) {
|
||||
snac snac;
|
||||
|
||||
if (user_open(&snac, v)) {
|
||||
char *p, *v;
|
||||
char *p;
|
||||
const char *v;
|
||||
xs *dir = xs_fmt("%s/hidden", snac.basedir);
|
||||
|
||||
/* create the hidden directory */
|
||||
|
@ -109,7 +113,8 @@ int snac_upgrade(xs_str **error)
|
|||
else
|
||||
if (f < 2.4) {
|
||||
xs *users = user_list();
|
||||
char *p, *v;
|
||||
char *p;
|
||||
const char *v;
|
||||
|
||||
p = users;
|
||||
while (xs_list_iter(&p, &v)) {
|
||||
|
@ -132,7 +137,8 @@ int snac_upgrade(xs_str **error)
|
|||
if (f < 2.5) {
|
||||
/* upgrade followers */
|
||||
xs *users = user_list();
|
||||
char *p, *v;
|
||||
char *p;
|
||||
const char *v;
|
||||
|
||||
p = users;
|
||||
while (xs_list_iter(&p, &v)) {
|
||||
|
@ -141,7 +147,8 @@ int snac_upgrade(xs_str **error)
|
|||
if (user_open(&snac, v)) {
|
||||
xs *spec = xs_fmt("%s/followers/" "*.json", snac.basedir);
|
||||
xs *dir = xs_glob(spec, 0, 0);
|
||||
char *p, *v;
|
||||
char *p;
|
||||
const char *v;
|
||||
|
||||
p = dir;
|
||||
while (xs_list_iter(&p, &v)) {
|
||||
|
@ -152,12 +159,12 @@ int snac_upgrade(xs_str **error)
|
|||
xs *o = xs_json_loads(s);
|
||||
fclose(f);
|
||||
|
||||
char *type = xs_dict_get(o, "type");
|
||||
const char *type = xs_dict_get(o, "type");
|
||||
|
||||
if (!xs_is_null(type) && strcmp(type, "Follow") == 0) {
|
||||
unlink(v);
|
||||
|
||||
char *actor = xs_dict_get(o, "actor");
|
||||
const char *actor = xs_dict_get(o, "actor");
|
||||
|
||||
if (!xs_is_null(actor))
|
||||
follower_add(&snac, actor);
|
||||
|
@ -175,7 +182,8 @@ int snac_upgrade(xs_str **error)
|
|||
if (f < 2.6) {
|
||||
/* upgrade local/ to public/ */
|
||||
xs *users = user_list();
|
||||
char *p, *v;
|
||||
char *p;
|
||||
const char *v;
|
||||
|
||||
p = users;
|
||||
while (xs_list_iter(&p, &v)) {
|
||||
|
@ -184,7 +192,8 @@ int snac_upgrade(xs_str **error)
|
|||
if (user_open(&snac, v)) {
|
||||
xs *spec = xs_fmt("%s/local/" "*.json", snac.basedir);
|
||||
xs *dir = xs_glob(spec, 0, 0);
|
||||
char *p, *v;
|
||||
char *p;
|
||||
const char *v;
|
||||
|
||||
p = dir;
|
||||
while (xs_list_iter(&p, &v)) {
|
||||
|
@ -198,22 +207,29 @@ int snac_upgrade(xs_str **error)
|
|||
xs *meta = xs_dup(xs_dict_get(o, "_snac"));
|
||||
o = xs_dict_del(o, "_snac");
|
||||
|
||||
char *id = xs_dict_get(o, "id");
|
||||
const char *id = xs_dict_get(o, "id");
|
||||
|
||||
/* store object */
|
||||
object_add_ow(id, o);
|
||||
|
||||
/* if it's from us, add to public */
|
||||
if (xs_startswith(id, snac.actor)) {
|
||||
char *p, *v;
|
||||
const xs_list *p;
|
||||
const char *v;
|
||||
int c;
|
||||
|
||||
object_user_cache_add(&snac, id, "public");
|
||||
|
||||
p = xs_dict_get(meta, "announced_by");
|
||||
while (xs_list_iter(&p, &v))
|
||||
|
||||
c = 0;
|
||||
while (xs_list_next(p, &v, &c))
|
||||
object_admire(id, v, 0);
|
||||
|
||||
p = xs_dict_get(meta, "liked_by");
|
||||
while (xs_list_iter(&p, &v))
|
||||
|
||||
c = 0;
|
||||
while (xs_list_next(p, &v, &c))
|
||||
object_admire(id, v, 1);
|
||||
}
|
||||
|
||||
|
@ -234,7 +250,8 @@ int snac_upgrade(xs_str **error)
|
|||
if (f < 2.7) {
|
||||
/* upgrade timeline/ to private/ */
|
||||
xs *users = user_list();
|
||||
char *p, *v;
|
||||
char *p;
|
||||
const char *v;
|
||||
|
||||
p = users;
|
||||
while (xs_list_iter(&p, &v)) {
|
||||
|
@ -243,7 +260,8 @@ int snac_upgrade(xs_str **error)
|
|||
if (user_open(&snac, v)) {
|
||||
xs *spec = xs_fmt("%s/timeline/" "*.json", snac.basedir);
|
||||
xs *dir = xs_glob(spec, 0, 0);
|
||||
char *p, *v;
|
||||
char *p;
|
||||
const char *v;
|
||||
|
||||
p = dir;
|
||||
while (xs_list_iter(&p, &v)) {
|
||||
|
@ -257,21 +275,28 @@ int snac_upgrade(xs_str **error)
|
|||
xs *meta = xs_dup(xs_dict_get(o, "_snac"));
|
||||
o = xs_dict_del(o, "_snac");
|
||||
|
||||
char *id = xs_dict_get(o, "id");
|
||||
const char *id = xs_dict_get(o, "id");
|
||||
|
||||
/* store object */
|
||||
object_add_ow(id, o);
|
||||
|
||||
{
|
||||
char *p, *v;
|
||||
const xs_list *p;
|
||||
const char *v;
|
||||
int c = 0;
|
||||
|
||||
object_user_cache_add(&snac, id, "private");
|
||||
|
||||
p = xs_dict_get(meta, "announced_by");
|
||||
while (xs_list_iter(&p, &v))
|
||||
|
||||
c = 0;
|
||||
while (xs_list_next(p, &v, &c))
|
||||
object_admire(id, v, 0);
|
||||
|
||||
p = xs_dict_get(meta, "liked_by");
|
||||
while (xs_list_iter(&p, &v))
|
||||
|
||||
c = 0;
|
||||
while (xs_list_next(p, &v, &c))
|
||||
object_admire(id, v, 1);
|
||||
}
|
||||
|
||||
|
|
17
utils.c
17
utils.c
|
@ -25,6 +25,8 @@ static const char *default_srv_config = "{"
|
|||
"\"dbglevel\": 0,"
|
||||
"\"queue_retry_minutes\": 2,"
|
||||
"\"queue_retry_max\": 10,"
|
||||
"\"queue_timeout\": 6,"
|
||||
"\"queue_timeout_2\": 8,"
|
||||
"\"cssurls\": [\"\"],"
|
||||
"\"max_timeline_entries\": 50,"
|
||||
"\"timeline_purge_days\": 120,"
|
||||
|
@ -34,6 +36,7 @@ static const char *default_srv_config = "{"
|
|||
"\"admin_account\": \"\","
|
||||
"\"title\": \"\","
|
||||
"\"short_description\": \"\","
|
||||
"\"protocol\": \"https\","
|
||||
"\"fastcgi\": false"
|
||||
"}";
|
||||
|
||||
|
@ -46,7 +49,7 @@ static const char *default_css =
|
|||
".snac-top-user { text-align: center; padding-bottom: 2em }\n"
|
||||
".snac-top-user-name { font-size: 200% }\n"
|
||||
".snac-top-user-id { font-size: 150% }\n"
|
||||
".snac-avatar { float: left; height: 2.5em; padding: 0.25em }\n"
|
||||
".snac-avatar { float: left; height: 2.5em; width: 2.5em; padding: 0.25em }\n"
|
||||
".snac-author { font-size: 90%; text-decoration: none }\n"
|
||||
".snac-author-tag { font-size: 80% }\n"
|
||||
".snac-pubdate { color: #a0a0a0; font-size: 90% }\n"
|
||||
|
@ -355,7 +358,7 @@ void rm_rf(const char *dir)
|
|||
xs *d = xs_str_cat(xs_dup(dir), "/" "*");
|
||||
xs *l = xs_glob(d, 0, 0);
|
||||
xs_list *p = l;
|
||||
xs_str *v;
|
||||
const xs_str *v;
|
||||
|
||||
if (dbglevel >= 1)
|
||||
printf("Deleting directory %s\n", dir);
|
||||
|
@ -390,7 +393,7 @@ int deluser(snac *user)
|
|||
int ret = 0;
|
||||
xs *fwers = following_list(user);
|
||||
xs_list *p = fwers;
|
||||
xs_str *v;
|
||||
const xs_str *v;
|
||||
|
||||
while (xs_list_iter(&p, &v)) {
|
||||
xs *object = NULL;
|
||||
|
@ -415,8 +418,8 @@ int deluser(snac *user)
|
|||
void verify_links(snac *user)
|
||||
/* verifies a user's links */
|
||||
{
|
||||
xs_dict *p = xs_dict_get(user->config, "metadata");
|
||||
char *k, *v;
|
||||
const xs_dict *p = xs_dict_get(user->config, "metadata");
|
||||
const char *k, *v;
|
||||
int changed = 0;
|
||||
|
||||
xs *headers = xs_dict_new();
|
||||
|
@ -446,7 +449,7 @@ void verify_links(snac *user)
|
|||
xs *ls = xs_regex_select(payload, "< *(a|link) +[^>]+>");
|
||||
|
||||
xs_list *lp = ls;
|
||||
char *ll;
|
||||
const char *ll;
|
||||
int vfied = 0;
|
||||
|
||||
while (!vfied && xs_list_iter(&lp, &ll)) {
|
||||
|
@ -460,7 +463,7 @@ void verify_links(snac *user)
|
|||
xs *href = NULL;
|
||||
int is_rel_me = 0;
|
||||
xs_list *pr = r;
|
||||
char *ar;
|
||||
const char *ar;
|
||||
|
||||
while (xs_list_iter(&pr, &ar)) {
|
||||
xs *nq = xs_dup(ar);
|
||||
|
|
49
webfinger.c
49
webfinger.c
|
@ -16,12 +16,13 @@ int webfinger_request_signed(snac *snac, const char *qs, char **actor, char **us
|
|||
int p_size = 0;
|
||||
xs *headers = xs_dict_new();
|
||||
xs *l = NULL;
|
||||
xs_str *host = NULL;
|
||||
const char *host = NULL;
|
||||
xs *resource = NULL;
|
||||
|
||||
if (xs_startswith(qs, "https:/" "/")) {
|
||||
if (xs_startswith(qs, "https:/") || xs_startswith(qs, "http:/")) {
|
||||
/* actor query: pick the host */
|
||||
xs *s = xs_replace_n(qs, "https:/" "/", "", 1);
|
||||
xs *s1 = xs_replace_n(qs, "http:/" "/", "", 1);
|
||||
xs *s = xs_replace_n(s1, "https:/" "/", "", 1);
|
||||
|
||||
l = xs_split_n(s, "/", 1);
|
||||
|
||||
|
@ -69,7 +70,9 @@ int webfinger_request_signed(snac *snac, const char *qs, char **actor, char **us
|
|||
&payload, &p_size, &ctype);
|
||||
}
|
||||
else {
|
||||
xs *url = xs_fmt("https:/" "/%s/.well-known/webfinger?resource=%s", host, resource);
|
||||
const char *proto = xs_dict_get_def(srv_config, "protocol", "https");
|
||||
|
||||
xs *url = xs_fmt("%s:/" "/%s/.well-known/webfinger?resource=%s", proto, host, resource);
|
||||
|
||||
if (snac == NULL)
|
||||
xs_http_request("GET", url, headers, NULL, 0, &status, &payload, &p_size, 0);
|
||||
|
@ -84,22 +87,24 @@ int webfinger_request_signed(snac *snac, const char *qs, char **actor, char **us
|
|||
|
||||
if (obj) {
|
||||
if (user != NULL) {
|
||||
char *subject = xs_dict_get(obj, "subject");
|
||||
const char *subject = xs_dict_get(obj, "subject");
|
||||
|
||||
if (subject)
|
||||
*user = xs_replace_n(subject, "acct:", "", 1);
|
||||
}
|
||||
|
||||
if (actor != NULL) {
|
||||
char *list = xs_dict_get(obj, "links");
|
||||
char *v;
|
||||
const xs_list *list = xs_dict_get(obj, "links");
|
||||
int c = 0;
|
||||
const char *v;
|
||||
|
||||
while (xs_list_iter(&list, &v)) {
|
||||
while (xs_list_next(list, &v, &c)) {
|
||||
if (xs_type(v) == XSTYPE_DICT) {
|
||||
char *type = xs_dict_get(v, "type");
|
||||
const char *type = xs_dict_get(v, "type");
|
||||
|
||||
if (type && (strcmp(type, "application/activity+json") == 0 ||
|
||||
strcmp(type, "application/ld+json; profile=\"https://www.w3.org/ns/activitystreams\"") == 0)) {
|
||||
strcmp(type, "application/ld+json; profile=\"https:/"
|
||||
"/www.w3.org/ns/activitystreams\"") == 0)) {
|
||||
*actor = xs_dup(xs_dict_get(v, "href"));
|
||||
break;
|
||||
}
|
||||
|
@ -130,8 +135,8 @@ int webfinger_get_handler(xs_dict *req, char *q_path,
|
|||
if (strcmp(q_path, "/.well-known/webfinger") != 0)
|
||||
return 0;
|
||||
|
||||
char *q_vars = xs_dict_get(req, "q_vars");
|
||||
char *resource = xs_dict_get(q_vars, "resource");
|
||||
const char *q_vars = xs_dict_get(req, "q_vars");
|
||||
const char *resource = xs_dict_get(q_vars, "resource");
|
||||
|
||||
if (resource == NULL)
|
||||
return 400;
|
||||
|
@ -139,10 +144,10 @@ int webfinger_get_handler(xs_dict *req, char *q_path,
|
|||
snac snac;
|
||||
int found = 0;
|
||||
|
||||
if (xs_startswith(resource, "https:/" "/")) {
|
||||
if (xs_startswith(resource, "https:/") || xs_startswith(resource, "http:/")) {
|
||||
/* actor search: find a user with this actor */
|
||||
xs *l = xs_split(resource, "/");
|
||||
char *uid = xs_list_get(l, -1);
|
||||
const char *uid = xs_list_get(l, -1);
|
||||
|
||||
if (uid)
|
||||
found = user_open(&snac, uid);
|
||||
|
@ -160,8 +165,8 @@ int webfinger_get_handler(xs_dict *req, char *q_path,
|
|||
l = xs_split_n(an, "@", 1);
|
||||
|
||||
if (xs_list_len(l) == 2) {
|
||||
char *uid = xs_list_get(l, 0);
|
||||
char *host = xs_list_get(l, 1);
|
||||
const char *uid = xs_list_get(l, 0);
|
||||
const char *host = xs_list_get(l, 1);
|
||||
|
||||
if (strcmp(host, xs_dict_get(srv_config, "host")) == 0)
|
||||
found = user_open(&snac, uid);
|
||||
|
@ -185,13 +190,19 @@ int webfinger_get_handler(xs_dict *req, char *q_path,
|
|||
|
||||
links = xs_list_append(links, aaj);
|
||||
|
||||
/* duplicate with the ld+json type */
|
||||
aaj = xs_dict_set(aaj, "type", "application/ld+json; profile=\"https:/"
|
||||
"/www.w3.org/ns/activitystreams\"");
|
||||
|
||||
links = xs_list_append(links, aaj);
|
||||
|
||||
prof = xs_dict_append(prof, "rel", "http://webfinger.net/rel/profile-page");
|
||||
prof = xs_dict_append(prof, "type", "text/html");
|
||||
prof = xs_dict_append(prof, "href", snac.actor);
|
||||
|
||||
links = xs_list_append(links, prof);
|
||||
|
||||
char *avatar = xs_dict_get(snac.config, "avatar");
|
||||
const char *avatar = xs_dict_get(snac.config, "avatar");
|
||||
if (!xs_is_null(avatar) && *avatar) {
|
||||
xs *d = xs_dict_new();
|
||||
|
||||
|
@ -211,10 +222,12 @@ int webfinger_get_handler(xs_dict *req, char *q_path,
|
|||
|
||||
status = 200;
|
||||
*body = j;
|
||||
*ctype = "application/json";
|
||||
*ctype = "application/jrd+json";
|
||||
}
|
||||
else
|
||||
status = 404;
|
||||
|
||||
srv_debug(1, xs_fmt("webfinger_get_handler resource=%s %d", resource, status));
|
||||
|
||||
return status;
|
||||
}
|
||||
|
|
254
xs.h
254
xs.h
|
@ -21,8 +21,8 @@ typedef enum {
|
|||
XSTYPE_FALSE = 0x15, /* Boolean */
|
||||
XSTYPE_LIST = 0x1d, /* Sequence of LITEMs up to EOM (with size) */
|
||||
XSTYPE_LITEM = 0x1f, /* Element of a list (any type) */
|
||||
XSTYPE_DICT = 0x1c, /* Sequence of DITEMs up to EOM (with size) */
|
||||
XSTYPE_DITEM = 0x1e, /* Element of a dict (STRING key + any type) */
|
||||
XSTYPE_DICT = 0x1c, /* Sequence of KEYVALs up to EOM (with size) */
|
||||
XSTYPE_KEYVAL = 0x1e, /* key + value (STRING key + any type) */
|
||||
XSTYPE_EOM = 0x19, /* End of Multiple (LIST or DICT) */
|
||||
XSTYPE_DATA = 0x10 /* A block of anonymous data */
|
||||
} xstype;
|
||||
|
@ -32,6 +32,7 @@ typedef enum {
|
|||
typedef char xs_val;
|
||||
typedef char xs_str;
|
||||
typedef char xs_list;
|
||||
typedef char xs_keyval;
|
||||
typedef char xs_dict;
|
||||
typedef char xs_number;
|
||||
typedef char xs_data;
|
||||
|
@ -45,6 +46,10 @@ typedef char xs_data;
|
|||
/* not really all, just very much */
|
||||
#define XS_ALL 0xfffffff
|
||||
|
||||
#ifndef xs_countof
|
||||
#define xs_countof(a) (sizeof((a)) / sizeof((*a)))
|
||||
#endif
|
||||
|
||||
void *xs_free(void *ptr);
|
||||
void *_xs_realloc(void *ptr, size_t size, const char *file, int line, const char *func);
|
||||
#define xs_realloc(ptr, size) _xs_realloc(ptr, size, __FILE__, __LINE__, __FUNCTION__)
|
||||
|
@ -89,9 +94,10 @@ xs_list *xs_list_new(void);
|
|||
xs_list *xs_list_append_m(xs_list *list, const char *mem, int dsz);
|
||||
xs_list *_xs_list_append(xs_list *list, const xs_val *vals[]);
|
||||
#define xs_list_append(list, ...) _xs_list_append(list, (const xs_val *[]){ __VA_ARGS__, NULL })
|
||||
int xs_list_iter(xs_list **list, xs_val **value);
|
||||
int xs_list_iter(xs_list **list, const xs_val **value);
|
||||
int xs_list_next(const xs_list *list, const xs_val **value, int *ctxt);
|
||||
int xs_list_len(const xs_list *list);
|
||||
xs_val *xs_list_get(const xs_list *list, int num);
|
||||
const xs_val *xs_list_get(const xs_list *list, int num);
|
||||
xs_list *xs_list_del(xs_list *list, int num);
|
||||
xs_list *xs_list_insert(xs_list *list, int num, const xs_val *data);
|
||||
xs_list *xs_list_set(xs_list *list, int num, const xs_val *data);
|
||||
|
@ -104,17 +110,20 @@ xs_list *xs_split_n(const char *str, const char *sep, int times);
|
|||
#define xs_split(str, sep) xs_split_n(str, sep, XS_ALL)
|
||||
xs_list *xs_list_cat(xs_list *l1, const xs_list *l2);
|
||||
|
||||
int xs_keyval_size(const xs_str *key, const xs_val *value);
|
||||
xs_str *xs_keyval_key(const xs_keyval *keyval);
|
||||
xs_val *xs_keyval_value(const xs_keyval *keyval);
|
||||
xs_keyval *xs_keyval_make(xs_keyval *keyval, const xs_str *key, const xs_val *value);
|
||||
|
||||
xs_dict *xs_dict_new(void);
|
||||
xs_dict *xs_dict_append_m(xs_dict *dict, const xs_str *key, const xs_val *mem, int dsz);
|
||||
#define xs_dict_append(dict, key, data) xs_dict_append_m(dict, key, data, xs_size(data))
|
||||
xs_dict *xs_dict_prepend_m(xs_dict *dict, const xs_str *key, const xs_val *mem, int dsz);
|
||||
#define xs_dict_prepend(dict, key, data) xs_dict_prepend_m(dict, key, data, xs_size(data))
|
||||
int xs_dict_iter(xs_dict **dict, xs_str **key, xs_val **value);
|
||||
int xs_dict_next(const xs_dict *dict, xs_str **key, xs_val **value, int *ctxt);
|
||||
xs_val *xs_dict_get_def(const xs_dict *dict, const xs_str *key, const xs_val *def);
|
||||
xs_dict *xs_dict_append(xs_dict *dict, const xs_str *key, const xs_val *value);
|
||||
xs_dict *xs_dict_prepend(xs_dict *dict, const xs_str *key, const xs_val *value);
|
||||
int xs_dict_next(const xs_dict *dict, const xs_str **key, const xs_val **value, int *ctxt);
|
||||
const xs_val *xs_dict_get_def(const xs_dict *dict, const xs_str *key, const xs_val *def);
|
||||
#define xs_dict_get(dict, key) xs_dict_get_def(dict, key, NULL)
|
||||
xs_dict *xs_dict_del(xs_dict *dict, const xs_str *key);
|
||||
xs_dict *xs_dict_set(xs_dict *dict, const xs_str *key, const xs_val *data);
|
||||
xs_dict *xs_dict_gc(xs_dict *dict);
|
||||
|
||||
xs_val *xs_val_new(xstype t);
|
||||
xs_number *xs_number_new(double f);
|
||||
|
@ -242,7 +251,7 @@ xstype xs_type(const xs_val *data)
|
|||
case XSTYPE_LIST:
|
||||
case XSTYPE_LITEM:
|
||||
case XSTYPE_DICT:
|
||||
case XSTYPE_DITEM:
|
||||
case XSTYPE_KEYVAL:
|
||||
case XSTYPE_NUMBER:
|
||||
case XSTYPE_EOM:
|
||||
case XSTYPE_DATA:
|
||||
|
@ -260,7 +269,7 @@ xstype xs_type(const xs_val *data)
|
|||
void _xs_put_size(xs_val *ptr, int i)
|
||||
/* must match _XS_TYPE_SIZE */
|
||||
{
|
||||
memcpy(ptr, &i, sizeof(i));
|
||||
memcpy(ptr + 1, &i, sizeof(i));
|
||||
}
|
||||
|
||||
|
||||
|
@ -294,7 +303,7 @@ int xs_size(const xs_val *data)
|
|||
|
||||
break;
|
||||
|
||||
case XSTYPE_DITEM:
|
||||
case XSTYPE_KEYVAL:
|
||||
/* calculate the size of the key and the value */
|
||||
p = data + 1;
|
||||
p += xs_size(p);
|
||||
|
@ -378,7 +387,7 @@ xs_val *xs_expand(xs_val *data, int offset, int size)
|
|||
if (xs_type(data) == XSTYPE_LIST ||
|
||||
xs_type(data) == XSTYPE_DICT ||
|
||||
xs_type(data) == XSTYPE_DATA)
|
||||
_xs_put_size(data + 1, sz);
|
||||
_xs_put_size(data, sz);
|
||||
|
||||
return data;
|
||||
}
|
||||
|
@ -403,7 +412,7 @@ xs_val *xs_collapse(xs_val *data, int offset, int size)
|
|||
if (xs_type(data) == XSTYPE_LIST ||
|
||||
xs_type(data) == XSTYPE_DICT ||
|
||||
xs_type(data) == XSTYPE_DATA)
|
||||
_xs_put_size(data + 1, sz);
|
||||
_xs_put_size(data, sz);
|
||||
|
||||
return xs_realloc(data, _xs_blk_size(sz));
|
||||
}
|
||||
|
@ -664,10 +673,10 @@ xs_list *xs_list_new(void)
|
|||
{
|
||||
int sz = 1 + _XS_TYPE_SIZE + 1;
|
||||
xs_list *l = xs_realloc(NULL, sz);
|
||||
memset(l, '\0', sz);
|
||||
memset(l, XSTYPE_EOM, sz);
|
||||
|
||||
l[0] = XSTYPE_LIST;
|
||||
_xs_put_size(&l[1], sz);
|
||||
_xs_put_size(l, sz);
|
||||
|
||||
return l;
|
||||
}
|
||||
|
@ -717,7 +726,7 @@ xs_list *_xs_list_append(xs_list *list, const xs_val *vals[])
|
|||
}
|
||||
|
||||
|
||||
int xs_list_iter(xs_list **list, xs_val **value)
|
||||
int xs_list_iter(xs_list **list, const xs_val **value)
|
||||
/* iterates a list value */
|
||||
{
|
||||
int goon = 1;
|
||||
|
@ -748,23 +757,58 @@ int xs_list_iter(xs_list **list, xs_val **value)
|
|||
}
|
||||
|
||||
|
||||
int xs_list_next(const xs_list *list, const xs_val **value, int *ctxt)
|
||||
/* iterates a list, with context */
|
||||
{
|
||||
if (xs_type(list) != XSTYPE_LIST)
|
||||
return 0;
|
||||
|
||||
int goon = 1;
|
||||
|
||||
const char *p = list;
|
||||
|
||||
/* skip the start of the list */
|
||||
if (*ctxt == 0)
|
||||
*ctxt = 1 + _XS_TYPE_SIZE;
|
||||
|
||||
p += *ctxt;
|
||||
|
||||
/* an element? */
|
||||
if (xs_type(p) == XSTYPE_LITEM) {
|
||||
p++;
|
||||
|
||||
*value = p;
|
||||
|
||||
p += xs_size(*value);
|
||||
}
|
||||
else {
|
||||
/* end of list */
|
||||
goon = 0;
|
||||
}
|
||||
|
||||
/* update the context */
|
||||
*ctxt = p - list;
|
||||
|
||||
return goon;
|
||||
}
|
||||
|
||||
|
||||
int xs_list_len(const xs_list *list)
|
||||
/* returns the number of elements in the list */
|
||||
{
|
||||
XS_ASSERT_TYPE_NULL(list, XSTYPE_LIST);
|
||||
|
||||
int c = 0;
|
||||
xs_list *p = (xs_list *)list;
|
||||
xs_val *v;
|
||||
int c = 0, ct = 0;
|
||||
const xs_val *v;
|
||||
|
||||
while (xs_list_iter(&p, &v))
|
||||
while (xs_list_next(list, &v, &ct))
|
||||
c++;
|
||||
|
||||
return c;
|
||||
}
|
||||
|
||||
|
||||
xs_val *xs_list_get(const xs_list *list, int num)
|
||||
const xs_val *xs_list_get(const xs_list *list, int num)
|
||||
/* returns the element #num */
|
||||
{
|
||||
XS_ASSERT_TYPE(list, XSTYPE_LIST);
|
||||
|
@ -772,11 +816,10 @@ xs_val *xs_list_get(const xs_list *list, int num)
|
|||
if (num < 0)
|
||||
num = xs_list_len(list) + num;
|
||||
|
||||
int c = 0;
|
||||
xs_list *p = (xs_list *)list;
|
||||
xs_val *v;
|
||||
int c = 0, ct = 0;
|
||||
const xs_val *v;
|
||||
|
||||
while (xs_list_iter(&p, &v)) {
|
||||
while (xs_list_next(list, &v, &ct)) {
|
||||
if (c == num)
|
||||
return v;
|
||||
|
||||
|
@ -792,7 +835,7 @@ xs_list *xs_list_del(xs_list *list, int num)
|
|||
{
|
||||
XS_ASSERT_TYPE(list, XSTYPE_LIST);
|
||||
|
||||
xs_val *v;
|
||||
const xs_val *v;
|
||||
|
||||
if ((v = xs_list_get(list, num)) != NULL)
|
||||
list = xs_collapse(list, v - 1 - list, xs_size(v - 1));
|
||||
|
@ -806,7 +849,7 @@ xs_list *xs_list_insert(xs_list *list, int num, const xs_val *data)
|
|||
{
|
||||
XS_ASSERT_TYPE(list, XSTYPE_LIST);
|
||||
|
||||
xs_val *v;
|
||||
const xs_val *v;
|
||||
int offset;
|
||||
|
||||
if ((v = xs_list_get(list, num)) != NULL)
|
||||
|
@ -835,16 +878,16 @@ xs_list *xs_list_dequeue(xs_list *list, xs_val **data, int last)
|
|||
{
|
||||
XS_ASSERT_TYPE(list, XSTYPE_LIST);
|
||||
|
||||
xs_list *p = list;
|
||||
xs_val *v = NULL;
|
||||
int ct = 0;
|
||||
const xs_val *v = NULL;
|
||||
|
||||
if (!last) {
|
||||
/* get the first */
|
||||
xs_list_iter(&p, &v);
|
||||
xs_list_next(list, &v, &ct);
|
||||
}
|
||||
else {
|
||||
/* iterate to the end */
|
||||
while (xs_list_iter(&p, &v));
|
||||
while (xs_list_next(list, &v, &ct));
|
||||
}
|
||||
|
||||
if (v != NULL) {
|
||||
|
@ -864,11 +907,11 @@ int xs_list_in(const xs_list *list, const xs_val *val)
|
|||
XS_ASSERT_TYPE_NULL(list, XSTYPE_LIST);
|
||||
|
||||
int n = 0;
|
||||
xs_list *p = (xs_list *)list;
|
||||
xs_val *v;
|
||||
int ct = 0;
|
||||
const xs_val *v;
|
||||
int sz = xs_size(val);
|
||||
|
||||
while (xs_list_iter(&p, &v)) {
|
||||
while (xs_list_next(list, &v, &ct)) {
|
||||
if (sz == xs_size(v) && memcmp(val, v, sz) == 0)
|
||||
return n;
|
||||
|
||||
|
@ -885,13 +928,13 @@ xs_str *xs_join(const xs_list *list, const char *sep)
|
|||
XS_ASSERT_TYPE(list, XSTYPE_LIST);
|
||||
|
||||
xs_str *s = NULL;
|
||||
xs_list *p = (xs_list *)list;
|
||||
xs_val *v;
|
||||
const xs_val *v;
|
||||
int c = 0;
|
||||
int ct = 0;
|
||||
int offset = 0;
|
||||
int ssz = strlen(sep);
|
||||
|
||||
while (xs_list_iter(&p, &v)) {
|
||||
while (xs_list_next(list, &v, &ct)) {
|
||||
/* refuse to join non-string values */
|
||||
if (xs_type(v) == XSTYPE_STRING) {
|
||||
int sz;
|
||||
|
@ -961,6 +1004,40 @@ xs_list *xs_list_cat(xs_list *l1, const xs_list *l2)
|
|||
}
|
||||
|
||||
|
||||
/** keyvals **/
|
||||
|
||||
int xs_keyval_size(const xs_str *key, const xs_val *value)
|
||||
/* returns the needed size for a keyval */
|
||||
{
|
||||
return 1 + xs_size(key) + xs_size(value);
|
||||
}
|
||||
|
||||
|
||||
xs_str *xs_keyval_key(const xs_keyval *keyval)
|
||||
/* returns a pointer to the key of the keyval */
|
||||
{
|
||||
return (xs_str *)&keyval[1];
|
||||
}
|
||||
|
||||
|
||||
xs_val *xs_keyval_value(const xs_keyval *keyval)
|
||||
/* returns a pointer to the value of the keyval */
|
||||
{
|
||||
return (xs_val *)&keyval[1 + xs_size(xs_keyval_key(keyval))];
|
||||
}
|
||||
|
||||
|
||||
xs_keyval *xs_keyval_make(xs_keyval *keyval, const xs_str *key, const xs_val *value)
|
||||
/* builds a keyval into mem (should have enough size) */
|
||||
{
|
||||
keyval[0] = XSTYPE_KEYVAL;
|
||||
memcpy(xs_keyval_key(keyval), key, xs_size(key));
|
||||
memcpy(xs_keyval_value(keyval), value, xs_size(value));
|
||||
|
||||
return keyval;
|
||||
}
|
||||
|
||||
|
||||
/** dicts **/
|
||||
|
||||
xs_dict *xs_dict_new(void)
|
||||
|
@ -968,87 +1045,47 @@ xs_dict *xs_dict_new(void)
|
|||
{
|
||||
int sz = 1 + _XS_TYPE_SIZE + 1;
|
||||
xs_dict *d = xs_realloc(NULL, sz);
|
||||
memset(d, '\0', sz);
|
||||
memset(d, XSTYPE_EOM, sz);
|
||||
|
||||
d[0] = XSTYPE_DICT;
|
||||
_xs_put_size(&d[1], sz);
|
||||
_xs_put_size(d, sz);
|
||||
|
||||
return d;
|
||||
}
|
||||
|
||||
|
||||
xs_dict *_xs_dict_write_ditem(xs_dict *dict, int offset, const xs_str *key,
|
||||
const xs_val *data, int dsz)
|
||||
/* inserts a memory block into the dict */
|
||||
xs_dict *_xs_dict_write_keyval(xs_dict *dict, int offset, const xs_str *key, const xs_val *value)
|
||||
/* adds a new keyval to the dict */
|
||||
{
|
||||
XS_ASSERT_TYPE(dict, XSTYPE_DICT);
|
||||
XS_ASSERT_TYPE(key, XSTYPE_STRING);
|
||||
|
||||
if (data == NULL) {
|
||||
data = xs_stock(XSTYPE_NULL);
|
||||
dsz = xs_size(data);
|
||||
}
|
||||
if (value == NULL)
|
||||
value = xs_stock(XSTYPE_NULL);
|
||||
|
||||
int ksz = xs_size(key);
|
||||
dict = xs_expand(dict, offset, xs_keyval_size(key, value));
|
||||
|
||||
dict = xs_expand(dict, offset, 1 + ksz + dsz);
|
||||
|
||||
dict[offset] = XSTYPE_DITEM;
|
||||
memcpy(&dict[offset + 1], key, ksz);
|
||||
memcpy(&dict[offset + 1 + ksz], data, dsz);
|
||||
xs_keyval_make(&dict[offset], key, value);
|
||||
|
||||
return dict;
|
||||
}
|
||||
|
||||
|
||||
xs_dict *xs_dict_append_m(xs_dict *dict, const xs_str *key, const xs_val *mem, int dsz)
|
||||
xs_dict *xs_dict_append(xs_dict *dict, const xs_str *key, const xs_val *value)
|
||||
/* appends a memory block to the dict */
|
||||
{
|
||||
return _xs_dict_write_ditem(dict, xs_size(dict) - 1, key, mem, dsz);
|
||||
return _xs_dict_write_keyval(dict, xs_size(dict) - 1, key, value);
|
||||
}
|
||||
|
||||
|
||||
xs_dict *xs_dict_prepend_m(xs_dict *dict, const xs_str *key, const xs_val *mem, int dsz)
|
||||
xs_dict *xs_dict_prepend(xs_dict *dict, const xs_str *key, const xs_val *value)
|
||||
/* prepends a memory block to the dict */
|
||||
{
|
||||
return _xs_dict_write_ditem(dict, 4, key, mem, dsz);
|
||||
return _xs_dict_write_keyval(dict, 1 + _XS_TYPE_SIZE, key, value);
|
||||
}
|
||||
|
||||
|
||||
int xs_dict_iter(xs_dict **dict, xs_str **key, xs_val **value)
|
||||
/* iterates a dict value */
|
||||
{
|
||||
int goon = 1;
|
||||
|
||||
xs_val *p = *dict;
|
||||
|
||||
/* skip the start of the list */
|
||||
if (xs_type(p) == XSTYPE_DICT)
|
||||
p += 1 + _XS_TYPE_SIZE;
|
||||
|
||||
/* an element? */
|
||||
if (xs_type(p) == XSTYPE_DITEM) {
|
||||
p++;
|
||||
|
||||
*key = p;
|
||||
p += xs_size(*key);
|
||||
|
||||
*value = p;
|
||||
p += xs_size(*value);
|
||||
}
|
||||
else {
|
||||
/* end of list */
|
||||
goon = 0;
|
||||
}
|
||||
|
||||
/* store back the pointer */
|
||||
*dict = p;
|
||||
|
||||
return goon;
|
||||
}
|
||||
|
||||
|
||||
int xs_dict_next(const xs_dict *dict, xs_str **key, xs_val **value, int *ctxt)
|
||||
int xs_dict_next(const xs_dict *dict, const xs_str **key, const xs_val **value, int *ctxt)
|
||||
/* iterates a dict, with context */
|
||||
{
|
||||
if (xs_type(dict) != XSTYPE_DICT)
|
||||
|
@ -1065,7 +1102,7 @@ int xs_dict_next(const xs_dict *dict, xs_str **key, xs_val **value, int *ctxt)
|
|||
p += *ctxt;
|
||||
|
||||
/* an element? */
|
||||
if (xs_type(p) == XSTYPE_DITEM) {
|
||||
if (xs_type(p) == XSTYPE_KEYVAL) {
|
||||
p++;
|
||||
|
||||
*key = p;
|
||||
|
@ -1086,14 +1123,14 @@ int xs_dict_next(const xs_dict *dict, xs_str **key, xs_val **value, int *ctxt)
|
|||
}
|
||||
|
||||
|
||||
xs_val *xs_dict_get_def(const xs_dict *dict, const xs_str *key, const xs_val *def)
|
||||
const xs_val *xs_dict_get_def(const xs_dict *dict, const xs_str *key, const xs_val *def)
|
||||
/* returns the value directed by key, or the default value */
|
||||
{
|
||||
XS_ASSERT_TYPE(dict, XSTYPE_DICT);
|
||||
XS_ASSERT_TYPE(key, XSTYPE_STRING);
|
||||
|
||||
xs_str *k;
|
||||
xs_val *v;
|
||||
const xs_str *k;
|
||||
const xs_val *v;
|
||||
int c = 0;
|
||||
|
||||
while (xs_dict_next(dict, &k, &v, &c)) {
|
||||
|
@ -1111,14 +1148,14 @@ xs_dict *xs_dict_del(xs_dict *dict, const xs_str *key)
|
|||
XS_ASSERT_TYPE(dict, XSTYPE_DICT);
|
||||
XS_ASSERT_TYPE(key, XSTYPE_STRING);
|
||||
|
||||
xs_str *k;
|
||||
xs_val *v;
|
||||
const xs_str *k;
|
||||
const xs_val *v;
|
||||
int c = 0;
|
||||
|
||||
while (xs_dict_next(dict, &k, &v, &c)) {
|
||||
if (strcmp(k, key) == 0) {
|
||||
/* the address of the item is just behind the key */
|
||||
char *i = k - 1;
|
||||
char *i = (char *)k - 1;
|
||||
|
||||
dict = xs_collapse(dict, i - dict, xs_size(i));
|
||||
break;
|
||||
|
@ -1145,6 +1182,14 @@ xs_dict *xs_dict_set(xs_dict *dict, const xs_str *key, const xs_val *data)
|
|||
}
|
||||
|
||||
|
||||
xs_dict *xs_dict_gc(xs_dict *dict)
|
||||
/* collects garbage (leaked values) inside a dict */
|
||||
{
|
||||
/* this kind of dicts does not get garbage */
|
||||
return dict;
|
||||
}
|
||||
|
||||
|
||||
/** other values **/
|
||||
|
||||
xs_val *xs_val_new(xstype t)
|
||||
|
@ -1195,8 +1240,11 @@ double xs_number_get(const xs_number *v)
|
|||
{
|
||||
double f = 0.0;
|
||||
|
||||
if (v != NULL && v[0] == XSTYPE_NUMBER)
|
||||
if (xs_type(v) == XSTYPE_NUMBER)
|
||||
f = atof(&v[1]);
|
||||
else
|
||||
if (xs_type(v) == XSTYPE_STRING)
|
||||
f = atof(v);
|
||||
|
||||
return f;
|
||||
}
|
||||
|
@ -1207,7 +1255,7 @@ const char *xs_number_str(const xs_number *v)
|
|||
{
|
||||
const char *p = NULL;
|
||||
|
||||
if (v != NULL && v[0] == XSTYPE_NUMBER)
|
||||
if (xs_type(v) == XSTYPE_NUMBER)
|
||||
p = &v[1];
|
||||
|
||||
return p;
|
||||
|
@ -1227,7 +1275,7 @@ xs_data *xs_data_new(const void *data, int size)
|
|||
v = xs_realloc(NULL, _xs_blk_size(total_size));
|
||||
v[0] = XSTYPE_DATA;
|
||||
|
||||
_xs_put_size(v + 1, total_size);
|
||||
_xs_put_size(v, total_size);
|
||||
|
||||
memcpy(&v[1 + _XS_TYPE_SIZE], data, size);
|
||||
|
||||
|
|
|
@ -28,7 +28,7 @@ static size_t _header_callback(char *buffer, size_t size,
|
|||
if (xs_str_in(l, ": ") != -1) {
|
||||
xs *knv = xs_split_n(l, ": ", 1);
|
||||
|
||||
xs_tolower_i(xs_list_get(knv, 0));
|
||||
xs_tolower_i((xs_str *)xs_list_get(knv, 0));
|
||||
|
||||
headers = xs_dict_set(headers, xs_list_get(knv, 0), xs_list_get(knv, 1));
|
||||
}
|
||||
|
@ -93,8 +93,8 @@ xs_dict *xs_http_request(const char *method, const char *url,
|
|||
xs_dict *response;
|
||||
CURL *curl;
|
||||
struct curl_slist *list = NULL;
|
||||
xs_str *k;
|
||||
xs_val *v;
|
||||
const xs_str *k;
|
||||
const xs_val *v;
|
||||
long lstatus = 0;
|
||||
struct _payload_data pd;
|
||||
|
||||
|
|
|
@ -293,8 +293,8 @@ void xs_fcgi_response(FILE *f, int status, xs_dict *headers, xs_str *body, int b
|
|||
struct fcgi_record_header hdr = {0};
|
||||
struct fcgi_end_request ereq = {0};
|
||||
xs *out = xs_str_new(NULL);
|
||||
xs_str *k;
|
||||
xs_str *v;
|
||||
const xs_str *k;
|
||||
const xs_str *v;
|
||||
|
||||
/* no previous id? it's an error */
|
||||
if (fcgi_id == -1)
|
||||
|
|
34
xs_html.h
34
xs_html.h
|
@ -6,26 +6,26 @@
|
|||
|
||||
typedef struct xs_html xs_html;
|
||||
|
||||
xs_str *xs_html_encode(char *str);
|
||||
xs_str *xs_html_encode(const char *str);
|
||||
|
||||
xs_html *xs_html_attr(char *key, char *value);
|
||||
xs_html *xs_html_text(char *content);
|
||||
xs_html *xs_html_raw(char *content);
|
||||
xs_html *xs_html_attr(const char *key, const char *value);
|
||||
xs_html *xs_html_text(const char *content);
|
||||
xs_html *xs_html_raw(const char *content);
|
||||
|
||||
xs_html *_xs_html_add(xs_html *tag, xs_html *var[]);
|
||||
#define xs_html_add(tag, ...) _xs_html_add(tag, (xs_html *[]) { __VA_ARGS__, NULL })
|
||||
|
||||
xs_html *_xs_html_tag(char *tag, xs_html *var[]);
|
||||
xs_html *_xs_html_tag(const char *tag, xs_html *var[]);
|
||||
#define xs_html_tag(tag, ...) _xs_html_tag(tag, (xs_html *[]) { __VA_ARGS__, NULL })
|
||||
|
||||
xs_html *_xs_html_sctag(char *tag, xs_html *var[]);
|
||||
xs_html *_xs_html_sctag(const char *tag, xs_html *var[]);
|
||||
#define xs_html_sctag(tag, ...) _xs_html_sctag(tag, (xs_html *[]) { __VA_ARGS__, NULL })
|
||||
|
||||
xs_html *_xs_html_container(xs_html *var[]);
|
||||
#define xs_html_container(...) _xs_html_container((xs_html *[]) { __VA_ARGS__, NULL })
|
||||
|
||||
void xs_html_render_f(xs_html *h, FILE *f);
|
||||
xs_str *xs_html_render_s(xs_html *tag, char *prefix);
|
||||
xs_str *xs_html_render_s(xs_html *tag, const char *prefix);
|
||||
#define xs_html_render(tag) xs_html_render_s(tag, NULL)
|
||||
|
||||
|
||||
|
@ -47,16 +47,16 @@ struct xs_html {
|
|||
xs_html *next;
|
||||
};
|
||||
|
||||
xs_str *xs_html_encode(char *str)
|
||||
xs_str *xs_html_encode(const char *str)
|
||||
/* encodes str using HTML entities */
|
||||
{
|
||||
xs_str *s = xs_str_new(NULL);
|
||||
int o = 0;
|
||||
char *e = str + strlen(str);
|
||||
const char *e = str + strlen(str);
|
||||
|
||||
for (;;) {
|
||||
char *ec = "<>\"'&"; /* characters to escape */
|
||||
char *q = e;
|
||||
const char *q = e;
|
||||
int z;
|
||||
|
||||
/* find the nearest happening of a char */
|
||||
|
@ -90,7 +90,7 @@ xs_str *xs_html_encode(char *str)
|
|||
|
||||
#define XS_HTML_NEW() memset(xs_realloc(NULL, sizeof(xs_html)), '\0', sizeof(xs_html))
|
||||
|
||||
xs_html *xs_html_attr(char *key, char *value)
|
||||
xs_html *xs_html_attr(const char *key, const char *value)
|
||||
/* creates an HTML block with an attribute */
|
||||
{
|
||||
xs_html *a = XS_HTML_NEW();
|
||||
|
@ -108,7 +108,7 @@ xs_html *xs_html_attr(char *key, char *value)
|
|||
}
|
||||
|
||||
|
||||
xs_html *xs_html_text(char *content)
|
||||
xs_html *xs_html_text(const char *content)
|
||||
/* creates an HTML block of text, escaping it previously */
|
||||
{
|
||||
xs_html *a = XS_HTML_NEW();
|
||||
|
@ -120,7 +120,7 @@ xs_html *xs_html_text(char *content)
|
|||
}
|
||||
|
||||
|
||||
xs_html *xs_html_raw(char *content)
|
||||
xs_html *xs_html_raw(const char *content)
|
||||
/* creates an HTML block without escaping (for pre-formatted HTML, comments, etc) */
|
||||
{
|
||||
xs_html *a = XS_HTML_NEW();
|
||||
|
@ -152,7 +152,7 @@ xs_html *_xs_html_add(xs_html *tag, xs_html *var[])
|
|||
}
|
||||
|
||||
|
||||
static xs_html *_xs_html_tag_t(xs_html_type type, char *tag, xs_html *var[])
|
||||
static xs_html *_xs_html_tag_t(xs_html_type type, const char *tag, xs_html *var[])
|
||||
/* creates a tag with a variable list of attributes and subtags */
|
||||
{
|
||||
xs_html *a = XS_HTML_NEW();
|
||||
|
@ -169,13 +169,13 @@ static xs_html *_xs_html_tag_t(xs_html_type type, char *tag, xs_html *var[])
|
|||
}
|
||||
|
||||
|
||||
xs_html *_xs_html_tag(char *tag, xs_html *var[])
|
||||
xs_html *_xs_html_tag(const char *tag, xs_html *var[])
|
||||
{
|
||||
return _xs_html_tag_t(XS_HTML_TAG, tag, var);
|
||||
}
|
||||
|
||||
|
||||
xs_html *_xs_html_sctag(char *tag, xs_html *var[])
|
||||
xs_html *_xs_html_sctag(const char *tag, xs_html *var[])
|
||||
{
|
||||
return _xs_html_tag_t(XS_HTML_SCTAG, tag, var);
|
||||
}
|
||||
|
@ -239,7 +239,7 @@ void xs_html_render_f(xs_html *h, FILE *f)
|
|||
}
|
||||
|
||||
|
||||
xs_str *xs_html_render_s(xs_html *tag, char *prefix)
|
||||
xs_str *xs_html_render_s(xs_html *tag, const char *prefix)
|
||||
/* renders to a string */
|
||||
{
|
||||
xs_str *s = NULL;
|
||||
|
|
|
@ -16,7 +16,7 @@ xs_dict *xs_httpd_request(FILE *f, xs_str **payload, int *p_size)
|
|||
xs *q_vars = NULL;
|
||||
xs *p_vars = NULL;
|
||||
xs *l1, *l2;
|
||||
char *v;
|
||||
const char *v;
|
||||
|
||||
xs_socket_timeout(fileno(f), 2.0, 0.0);
|
||||
|
||||
|
@ -60,7 +60,8 @@ xs_dict *xs_httpd_request(FILE *f, xs_str **payload, int *p_size)
|
|||
p = xs_split_n(l, ": ", 1);
|
||||
|
||||
if (xs_list_len(p) == 2)
|
||||
req = xs_dict_append(req, xs_tolower_i(xs_list_get(p, 0)), xs_list_get(p, 1));
|
||||
req = xs_dict_append(req, xs_tolower_i(
|
||||
(xs_str *)xs_list_get(p, 0)), xs_list_get(p, 1));
|
||||
}
|
||||
|
||||
xs_socket_timeout(fileno(f), 5.0, 0.0);
|
||||
|
@ -98,8 +99,8 @@ void xs_httpd_response(FILE *f, int status, xs_dict *headers, xs_str *body, int
|
|||
/* sends an httpd response */
|
||||
{
|
||||
xs *proto;
|
||||
xs_str *k;
|
||||
xs_val *v;
|
||||
const xs_str *k;
|
||||
const xs_val *v;
|
||||
|
||||
proto = xs_fmt("HTTP/1.1 %d %s", status, status / 100 == 2 ? "OK" : "ERROR");
|
||||
fprintf(f, "%s\r\n", proto);
|
||||
|
|
23
xs_json.h
23
xs_json.h
|
@ -71,12 +71,12 @@ static void _xs_json_indent(int level, int indent, FILE *f)
|
|||
}
|
||||
|
||||
|
||||
static void _xs_json_dump(const xs_val *s_data, int level, int indent, FILE *f)
|
||||
static void _xs_json_dump(const xs_val *data, int level, int indent, FILE *f)
|
||||
/* dumps partial data as JSON */
|
||||
{
|
||||
int c = 0;
|
||||
xs_val *v;
|
||||
xs_val *data = (xs_val *)s_data;
|
||||
int ct = 0;
|
||||
const xs_val *v;
|
||||
|
||||
switch (xs_type(data)) {
|
||||
case XSTYPE_NULL:
|
||||
|
@ -98,7 +98,7 @@ static void _xs_json_dump(const xs_val *s_data, int level, int indent, FILE *f)
|
|||
case XSTYPE_LIST:
|
||||
fputc('[', f);
|
||||
|
||||
while (xs_list_iter(&data, &v)) {
|
||||
while (xs_list_next(data, &v, &ct)) {
|
||||
if (c != 0)
|
||||
fputc(',', f);
|
||||
|
||||
|
@ -116,10 +116,9 @@ static void _xs_json_dump(const xs_val *s_data, int level, int indent, FILE *f)
|
|||
case XSTYPE_DICT:
|
||||
fputc('{', f);
|
||||
|
||||
xs_str *k;
|
||||
int ct = 0;
|
||||
const xs_str *k;
|
||||
|
||||
while (xs_dict_next(s_data, &k, &v, &ct)) {
|
||||
while (xs_dict_next(data, &k, &v, &ct)) {
|
||||
if (c != 0)
|
||||
fputc(',', f);
|
||||
|
||||
|
@ -328,7 +327,7 @@ static xs_val *_xs_json_load_lexer(FILE *f, js_type *t)
|
|||
|
||||
int xs_json_load_array_iter(FILE *f, xs_val **value, xstype *pt, int *c)
|
||||
/* loads the next scalar value from the JSON stream */
|
||||
/* if the value ahead is complex, value is NULL and pt is filled */
|
||||
/* if the value ahead is compound, value is NULL and pt is set */
|
||||
{
|
||||
js_type t;
|
||||
|
||||
|
@ -348,7 +347,7 @@ int xs_json_load_array_iter(FILE *f, xs_val **value, xstype *pt, int *c)
|
|||
}
|
||||
|
||||
if (*value == NULL) {
|
||||
/* possible complex type ahead */
|
||||
/* possible compound type ahead */
|
||||
if (t == JS_OBRACK)
|
||||
*pt = XSTYPE_LIST;
|
||||
else
|
||||
|
@ -365,7 +364,7 @@ int xs_json_load_array_iter(FILE *f, xs_val **value, xstype *pt, int *c)
|
|||
|
||||
|
||||
xs_list *xs_json_load_array(FILE *f)
|
||||
/* loads a JSON array (after the initial OBRACK) */
|
||||
/* loads a full JSON array (after the initial OBRACK) */
|
||||
{
|
||||
xstype t;
|
||||
xs_list *l = xs_list_new();
|
||||
|
@ -406,7 +405,7 @@ xs_list *xs_json_load_array(FILE *f)
|
|||
|
||||
int xs_json_load_object_iter(FILE *f, xs_str **key, xs_val **value, xstype *pt, int *c)
|
||||
/* loads the next key and scalar value from the JSON stream */
|
||||
/* if the value ahead is complex, value is NULL and pt is filled */
|
||||
/* if the value ahead is compound, value is NULL and pt is set */
|
||||
{
|
||||
js_type t;
|
||||
|
||||
|
@ -453,7 +452,7 @@ int xs_json_load_object_iter(FILE *f, xs_str **key, xs_val **value, xstype *pt,
|
|||
|
||||
|
||||
xs_dict *xs_json_load_object(FILE *f)
|
||||
/* loads a JSON object (after the initial OCURLY) */
|
||||
/* loads a full JSON object (after the initial OCURLY) */
|
||||
{
|
||||
xstype t;
|
||||
xs_dict *d = xs_dict_new();
|
||||
|
|
22
xs_mime.h
22
xs_mime.h
|
@ -55,19 +55,23 @@ const char *xs_mime_by_ext(const char *file)
|
|||
const char *ext = strrchr(file, '.');
|
||||
|
||||
if (ext) {
|
||||
const char **p = xs_mime_types;
|
||||
xs *uext = xs_tolower_i(xs_dup(ext + 1));
|
||||
xs *uext = xs_tolower_i(xs_dup(ext + 1));
|
||||
int b = 0;
|
||||
int t = xs_countof(xs_mime_types) / 2 - 2;
|
||||
|
||||
while (*p) {
|
||||
int c;
|
||||
while (t >= b) {
|
||||
int n = (b + t) / 2;
|
||||
const char *p = xs_mime_types[n * 2];
|
||||
|
||||
if ((c = strcmp(*p, uext)) == 0)
|
||||
return p[1];
|
||||
int c = strcmp(uext, p);
|
||||
|
||||
if (c < 0)
|
||||
t = n - 1;
|
||||
else
|
||||
if (c > 0)
|
||||
break;
|
||||
|
||||
p += 2;
|
||||
b = n + 1;
|
||||
else
|
||||
return xs_mime_types[(n * 2) + 1];
|
||||
}
|
||||
}
|
||||
|
||||
|
|
43
xs_regex.h
43
xs_regex.h
|
@ -4,6 +4,7 @@
|
|||
|
||||
#define _XS_REGEX_H
|
||||
|
||||
int xs_regex_match(const char *str, const char *rx);
|
||||
xs_list *xs_regex_split_n(const char *str, const char *rx, int count);
|
||||
#define xs_regex_split(str, rx) xs_regex_split_n(str, rx, XS_ALL)
|
||||
xs_list *xs_regex_select_n(const char *str, const char *rx, int count);
|
||||
|
@ -15,21 +16,29 @@ xs_list *xs_regex_replace_in(xs_str *str, const char *rx, const char *rep, int c
|
|||
|
||||
#ifdef XS_IMPLEMENTATION
|
||||
|
||||
#ifdef __TINYC__
|
||||
/* fix a compilation error in tcc */
|
||||
#define _REGEX_NELTS(n)
|
||||
#endif
|
||||
|
||||
#include <regex.h>
|
||||
|
||||
xs_list *xs_regex_split_n(const char *str, const char *rx, int count)
|
||||
/* splits str by regex */
|
||||
/* splits str using regex as a separator, at most count times.
|
||||
Always returns a list:
|
||||
len == 0: regcomp error
|
||||
len == 1: full string (no matches)
|
||||
len == odd: first part [ separator / next part ]...
|
||||
*/
|
||||
{
|
||||
regex_t re;
|
||||
regmatch_t rm;
|
||||
int offset = 0;
|
||||
xs_list *list = NULL;
|
||||
xs_list *list = xs_list_new();
|
||||
const char *p;
|
||||
|
||||
if (regcomp(&re, rx, REG_EXTENDED))
|
||||
return NULL;
|
||||
|
||||
list = xs_list_new();
|
||||
return list;
|
||||
|
||||
while (count > 0 && !regexec(&re, (p = str + offset), 1, &rm, offset > 0 ? REG_NOTBOL : 0)) {
|
||||
/* add first the leading part of the string */
|
||||
|
@ -60,16 +69,15 @@ xs_list *xs_regex_select_n(const char *str, const char *rx, int count)
|
|||
{
|
||||
xs_list *list = xs_list_new();
|
||||
xs *split = NULL;
|
||||
xs_list *p;
|
||||
xs_val *v;
|
||||
const xs_val *v;
|
||||
int n = 0;
|
||||
int c = 0;
|
||||
|
||||
/* split */
|
||||
split = xs_regex_split_n(str, rx, count);
|
||||
|
||||
/* now iterate to get only the 'separators' (odd ones) */
|
||||
p = split;
|
||||
while (xs_list_iter(&p, &v)) {
|
||||
while (xs_list_next(split, &v, &c)) {
|
||||
if (n & 0x1)
|
||||
list = xs_list_append(list, v);
|
||||
|
||||
|
@ -86,13 +94,12 @@ xs_list *xs_regex_replace_in(xs_str *str, const char *rx, const char *rep, int c
|
|||
{
|
||||
xs_str *s = xs_str_new(NULL);
|
||||
xs *split = xs_regex_split_n(str, rx, count);
|
||||
xs_list *p;
|
||||
xs_val *v;
|
||||
const xs_val *v;
|
||||
int n = 0;
|
||||
int c = 0;
|
||||
int pholder = !!strchr(rep, '&');
|
||||
|
||||
p = split;
|
||||
while (xs_list_iter(&p, &v)) {
|
||||
while (xs_list_next(split, &v, &c)) {
|
||||
if (n & 0x1) {
|
||||
if (pholder) {
|
||||
/* rep has a placeholder; process char by char */
|
||||
|
@ -128,6 +135,16 @@ xs_list *xs_regex_replace_in(xs_str *str, const char *rx, const char *rep, int c
|
|||
return s;
|
||||
}
|
||||
|
||||
|
||||
int xs_regex_match(const char *str, const char *rx)
|
||||
/* returns if str matches the regex at least once */
|
||||
{
|
||||
xs *l = xs_regex_select_n(str, rx, 1);
|
||||
|
||||
return xs_list_len(l) == 1;
|
||||
}
|
||||
|
||||
|
||||
#endif /* XS_IMPLEMENTATION */
|
||||
|
||||
#endif /* XS_REGEX_H */
|
||||
|
|
8
xs_set.h
8
xs_set.h
|
@ -85,7 +85,7 @@ int xs_set_add(xs_set *s, const xs_val *data)
|
|||
{
|
||||
/* is it 'full'? */
|
||||
if (s->used >= s->elems / 2) {
|
||||
char *p, *v;
|
||||
const xs_val *v;
|
||||
|
||||
/* expand! */
|
||||
s->elems *= 2;
|
||||
|
@ -95,8 +95,8 @@ int xs_set_add(xs_set *s, const xs_val *data)
|
|||
memset(s->hash, '\0', s->elems * sizeof(int));
|
||||
|
||||
/* add the list elements back */
|
||||
p = s->list;
|
||||
while (xs_list_iter(&p, &v))
|
||||
int ct = 0;
|
||||
while (xs_list_next(s->list, &v, &ct))
|
||||
_store_hash(s, v, v - s->list);
|
||||
}
|
||||
|
||||
|
@ -104,7 +104,7 @@ int xs_set_add(xs_set *s, const xs_val *data)
|
|||
|
||||
/* if it's new, add the data */
|
||||
if (ret)
|
||||
s->list = xs_list_append_m(s->list, data, xs_size(data));
|
||||
s->list = xs_list_append(s->list, data);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
|
22
xs_unicode.h
22
xs_unicode.h
|
@ -6,7 +6,7 @@
|
|||
|
||||
int _xs_utf8_enc(char buf[4], unsigned int cpoint);
|
||||
int xs_is_utf8_cont_byte(char c);
|
||||
unsigned int xs_utf8_dec(char **str);
|
||||
unsigned int xs_utf8_dec(const char **str);
|
||||
int xs_unicode_width(unsigned int cpoint);
|
||||
int xs_is_surrogate(unsigned int cpoint);
|
||||
unsigned int xs_surrogate_dec(unsigned int p1, unsigned int p2);
|
||||
|
@ -27,8 +27,8 @@
|
|||
|
||||
#ifdef XS_IMPLEMENTATION
|
||||
|
||||
#ifndef countof
|
||||
#define countof(a) (sizeof((a)) / sizeof((*a)))
|
||||
#ifndef xs_countof
|
||||
#define xs_countof(a) (sizeof((a)) / sizeof((*a)))
|
||||
#endif
|
||||
|
||||
int _xs_utf8_enc(char buf[4], unsigned int cpoint)
|
||||
|
@ -66,10 +66,10 @@ int xs_is_utf8_cont_byte(char c)
|
|||
}
|
||||
|
||||
|
||||
unsigned int xs_utf8_dec(char **str)
|
||||
unsigned int xs_utf8_dec(const char **str)
|
||||
/* decodes an utf-8 char inside str and updates the pointer */
|
||||
{
|
||||
char *p = *str;
|
||||
const char *p = *str;
|
||||
unsigned int cpoint = 0;
|
||||
unsigned char c = *p++;
|
||||
int cb = 0;
|
||||
|
@ -125,7 +125,7 @@ int xs_unicode_width(unsigned int cpoint)
|
|||
/* returns the width in columns of a Unicode codepoint (somewhat simplified) */
|
||||
{
|
||||
int b = 0;
|
||||
int t = countof(xs_unicode_width_table) / 3 - 1;
|
||||
int t = xs_countof(xs_unicode_width_table) / 3 - 1;
|
||||
|
||||
while (t >= b) {
|
||||
int n = (b + t) / 2;
|
||||
|
@ -193,7 +193,7 @@ unsigned int *_xs_unicode_upper_search(unsigned int cpoint)
|
|||
/* searches for an uppercase codepoint in the case fold table */
|
||||
{
|
||||
int b = 0;
|
||||
int t = countof(xs_unicode_case_fold_table) / 2 + 1;
|
||||
int t = xs_countof(xs_unicode_case_fold_table) / 2 + 1;
|
||||
|
||||
while (t >= b) {
|
||||
int n = (b + t) / 2;
|
||||
|
@ -216,7 +216,7 @@ unsigned int *_xs_unicode_lower_search(unsigned int cpoint)
|
|||
/* searches for a lowercase codepoint in the case fold table */
|
||||
{
|
||||
unsigned int *p = xs_unicode_case_fold_table;
|
||||
unsigned int *e = p + countof(xs_unicode_case_fold_table);
|
||||
unsigned int *e = p + xs_countof(xs_unicode_case_fold_table);
|
||||
|
||||
while (p < e) {
|
||||
if (cpoint == p[1])
|
||||
|
@ -251,7 +251,7 @@ int xs_unicode_nfd(unsigned int cpoint, unsigned int *base, unsigned int *diac)
|
|||
/* applies unicode Normalization Form D */
|
||||
{
|
||||
int b = 0;
|
||||
int t = countof(xs_unicode_nfd_table) / 3 - 1;
|
||||
int t = xs_countof(xs_unicode_nfd_table) / 3 - 1;
|
||||
|
||||
while (t >= b) {
|
||||
int n = (b + t) / 2;
|
||||
|
@ -279,7 +279,7 @@ int xs_unicode_nfc(unsigned int base, unsigned int diac, unsigned int *cpoint)
|
|||
/* applies unicode Normalization Form C */
|
||||
{
|
||||
unsigned int *p = xs_unicode_nfd_table;
|
||||
unsigned int *e = p + countof(xs_unicode_nfd_table);
|
||||
unsigned int *e = p + xs_countof(xs_unicode_nfd_table);
|
||||
|
||||
while (p < e) {
|
||||
if (p[1] == base && p[2] == diac) {
|
||||
|
@ -298,7 +298,7 @@ int xs_unicode_is_alpha(unsigned int cpoint)
|
|||
/* checks if a codepoint is an alpha (i.e. a letter) */
|
||||
{
|
||||
int b = 0;
|
||||
int t = countof(xs_unicode_alpha_table) / 2 - 1;
|
||||
int t = xs_countof(xs_unicode_alpha_table) / 2 - 1;
|
||||
|
||||
while (t >= b) {
|
||||
int n = (b + t) / 2;
|
||||
|
|
13
xs_url.h
13
xs_url.h
|
@ -51,12 +51,11 @@ xs_dict *xs_url_vars(const char *str)
|
|||
/* split by arguments */
|
||||
xs *args = xs_split(str, "&");
|
||||
|
||||
xs_list *l;
|
||||
xs_val *v;
|
||||
int ct = 0;
|
||||
const xs_val *v;
|
||||
|
||||
l = args;
|
||||
while (xs_list_iter(&l, &v)) {
|
||||
xs *kv = xs_split_n(v, "=", 2);
|
||||
while (xs_list_next(args, &v, &ct)) {
|
||||
xs *kv = xs_split_n(v, "=", 1);
|
||||
|
||||
if (xs_list_len(kv) == 2) {
|
||||
const char *key = xs_list_get(kv, 0);
|
||||
|
@ -119,8 +118,8 @@ xs_dict *xs_multipart_form_data(const char *payload, int p_size, const char *hea
|
|||
while ((p = xs_memmem(payload + offset, p_size - offset, boundary, bsz)) != NULL) {
|
||||
xs *s1 = NULL;
|
||||
xs *l1 = NULL;
|
||||
char *vn = NULL;
|
||||
char *fn = NULL;
|
||||
const char *vn = NULL;
|
||||
const char *fn = NULL;
|
||||
char *q;
|
||||
int po, ps;
|
||||
|
||||
|
|
|
@ -1 +1 @@
|
|||
/* 0df383371d207b0adfda40912ee54b37e5b01152 2024-03-15T03:45:39+01:00 */
|
||||
/* 65769f25ed99b886a643522bef21628396cd118d 2024-05-25T08:18:51+02:00 */
|
||||
|
|
Loading…
Reference in a new issue