]> code.citadel.org Git - citadel.git/blobdiff - citadel/modules/listsub/serv_listsub.c
Big change to mailing list subscription/unsubscription!
[citadel.git] / citadel / modules / listsub / serv_listsub.c
index c67582da088de6b3790abfc558b217e06d9a0a09..0c8335a50fd515faee319a58d72447297ef30a8c 100644 (file)
@@ -1,16 +1,16 @@
-/*
- * This module handles self-service subscription/unsubscription to mail lists.
- *
- * Copyright (c) 2002-2016 by the citadel.org team
- *
- * This program is open source software; you can redistribute it and/or modify
- * it under the terms of the GNU General Public License version 3.
- *  
- * This program is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
- * GNU General Public License for more details.
- */
+// This module handles self-service subscription/unsubscription to mail lists.
+//
+// Copyright (c) 2002-2022 by the citadel.org team
+//
+// This program is open source software.  It runs great on the
+// Linux operating system (and probably elsewhere).  You can use,
+// copy, and run it under the terms of the GNU General Public
+// License version 3.
+//
+// This program is distributed in the hope that it will be useful,
+// but WITHOUT ANY WARRANTY; without even the implied warranty of
+// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+// GNU General Public License for more details.
 
 #include "sysdep.h"
 #include <stdlib.h>
@@ -27,6 +27,7 @@
 #include <sys/wait.h>
 #include <string.h>
 #include <limits.h>
+#include <crypt.h>
 #include <libcitadel.h>
 #include "citadel.h"
 #include "server.h"
 #include "ctdl_module.h"
 
 
+enum {                         // one of these gets passed to do_subscribe_or_unsubscribe() so it knows what we asked for
+       UNSUBSCRIBE,
+       SUBSCRIBE
+};
 
-// FIXME rewrite the subscribe-o-matic AJC 2021
 
+// The confirmation token will be generated by combining the room name and email address with the host key,
+// and then generating an encrypted hash of that string.  The encrypted hash is included as part of the
+// confirmation link.
+void generate_confirmation_token(char *token_buf, size_t token_buf_len, char *roomname, char *emailaddr) {
+       char string_to_hash[1024];
+       struct crypt_data cd;
+       char *ptr;
 
-/* 
- * process subscribe/unsubscribe requests and confirmations
- */
-void cmd_subs(char *cmdbuf)
-{
-       cprintf("%d Invalid command\n", ERROR + ILLEGAL_VALUE);
+       snprintf(string_to_hash, sizeof string_to_hash, "%s|%s|%s", roomname, emailaddr, CtdlGetConfigStr("host_key"));
+       memset(&cd, 0, sizeof cd);
+
+       strncpy(token_buf, crypt_r(string_to_hash, "$1$ctdl", &cd), token_buf_len);
+
+       for (ptr=token_buf; *ptr; ++ptr) {
+               if (!isalnum((char)*ptr)) *ptr='X';
+       }
+}
+
+
+// This generates an email with a link the user clicks to confirm a list subscription.
+void send_subscribe_confirmation_email(char *roomname, char *emailaddr, char *url, char *confirmation_token) {
+       // We need a URL-safe representation of the room name
+       char urlroom[ROOMNAMELEN+10];
+       urlesc(urlroom, sizeof(urlroom), roomname);
+
+       char from_address[1024];
+       snprintf(from_address, sizeof from_address, "noreply@%s", CtdlGetConfigStr("c_fqdn"));
+
+       char emailtext[SIZ];
+       snprintf(emailtext, sizeof emailtext,
+               "MIME-Version: 1.0\n"
+               "Content-Type: multipart/alternative; boundary=\"__ctdlmultipart__\"\n"
+               "\n"
+               "This is a multipart message in MIME format.\n"
+               "\n"
+               "--__ctdlmultipart__\n"
+               "Content-type: text/plain\n"
+               "\n"
+               "Someone (probably you) has submitted a request to subscribe\n"
+               "<%s> to the <%s> mailing list.\n"
+               "\n"
+               "Please go here to confirm this request:\n"
+               "%s?cmd=confirm_subscribe&email=%s&room=%s&token=%s\n"
+               "\n"
+               "If this request has been submitted in error and you do not\n"
+               "wish to receive the <%s> mailing list, simply do nothing,\n"
+               "and you will not receive any further mailings.\n"
+               "\n"
+               "--__ctdlmultipart__\n"
+               "Content-type: text/html\n"
+               "\n"
+               "<html><body><p>Someone (probably you) has submitted a request to subscribe "
+               "<strong>%s</strong> to the <strong>%s</strong> mailing list.</p>"
+               "<p>Please go here to confirm this request:</p>"
+               "<p><a href=\"%s?cmd=confirm_subscribe&email=%s&room=%s&token=%s\">"
+               "%s?cmd=confirm_subscribe&email=%s&room=%s&token=%s</a></p>"
+               "<p>If this request has been submitted in error and you do not "
+               "wish to receive the <strong>%s<strong> mailing list, simply do nothing, "
+               "and you will not receive any further mailings.</p>"
+               "</body></html>\n"
+               "\n"
+               "--__ctdlmultipart__--\n"
+               ,
+               emailaddr, roomname,
+               url, emailaddr, urlroom, confirmation_token,
+               roomname
+               ,
+               emailaddr, roomname,
+               url, emailaddr, urlroom, confirmation_token,
+               url, emailaddr, urlroom, confirmation_token,
+               roomname
+       );
+
+       quickie_message("Citadel", from_address, emailaddr, NULL, emailtext, FMT_RFC822, "Please confirm your list subscription");
+       cprintf("%d confirmation email sent\n", CIT_OK);
+}
+
+
+// This generates an email with a link the user clicks to confirm a list unsubscription.
+void send_unsubscribe_confirmation_email(char *roomname, char *emailaddr, char *url, char *confirmation_token) {
+       // We need a URL-safe representation of the room name
+       char urlroom[ROOMNAMELEN+10];
+       urlesc(urlroom, sizeof(urlroom), roomname);
+
+       char from_address[1024];
+       snprintf(from_address, sizeof from_address, "noreply@%s", CtdlGetConfigStr("c_fqdn"));
+
+       char emailtext[SIZ];
+       snprintf(emailtext, sizeof emailtext,
+               "MIME-Version: 1.0\n"
+               "Content-Type: multipart/alternative; boundary=\"__ctdlmultipart__\"\n"
+               "\n"
+               "This is a multipart message in MIME format.\n"
+               "\n"
+               "--__ctdlmultipart__\n"
+               "Content-type: text/plain\n"
+               "\n"
+               "Someone (probably you) has submitted a request to unsubscribe\n"
+               "<%s> from the <%s> mailing list.\n"
+               "\n"
+               "Please go here to confirm this request:\n"
+               "%s?cmd=confirm_unsubscribe&email=%s&room=%s&token=%s\n"
+               "\n"
+               "If this request has been submitted in error and you still\n"
+               "wish to receive the <%s> mailing list, simply do nothing,\n"
+               "and you will remain subscribed.\n"
+               "\n"
+               "--__ctdlmultipart__\n"
+               "Content-type: text/html\n"
+               "\n"
+               "<html><body><p>Someone (probably you) has submitted a request to unsubscribe "
+               "<strong>%s</strong> from the <strong>%s</strong> mailing list.</p>"
+               "<p>Please go here to confirm this request:</p>"
+               "<p><a href=\"%s?cmd=confirm_unsubscribe&email=%s&room=%s&token=%s\">"
+               "%s?cmd=confirm_unsubscribe&email=%s&room=%s&token=%s</a></p>"
+               "<p>If this request has been submitted in error and you still "
+               "wish to receive the <strong>%s<strong> mailing list, simply do nothing, "
+               "and you will remain subscribed.</p>"
+               "</body></html>\n"
+               "\n"
+               "--__ctdlmultipart__--\n"
+               ,
+               emailaddr, roomname,
+               url, emailaddr, urlroom, confirmation_token,
+               roomname
+               ,
+               emailaddr, roomname,
+               url, emailaddr, urlroom, confirmation_token,
+               url, emailaddr, urlroom, confirmation_token,
+               roomname
+       );
+
+       quickie_message("Citadel", from_address, emailaddr, NULL, emailtext, FMT_RFC822, "Please confirm your list unsubscription");
+       cprintf("%d confirmation email sent\n", CIT_OK);
+}
+
+
+// Confirm a list subscription or unsubscription
+void do_confirm(int cmd, char *roomname, char *emailaddr, char *url, char *generated_token, char *supplied_token) {
+       int i;
+       char buf[1024];
+       int config_lines = 0;
+       char *oldnetconfig, *newnetconfig;
+
+       // The server has generated a persistent confirmation token for the user+room combination.
+       // Let's see if the user has supplied the same token.
+       if (strcmp(generated_token, supplied_token)) {
+               cprintf("%d This request could not be authenticated.\n", ERROR + PASSWORD_REQUIRED);
+               return;
+       }
+
+       // If the generated token matches the supplied token, the request is authentic.  Do what it says.
+
+       // Load the room's network configuration...
+        oldnetconfig = LoadRoomNetConfigFile(CC->room.QRnumber);
+        if (!oldnetconfig) {
+               oldnetconfig = strdup("");
+       }
+
+       // The new netconfig begins with an empty buffer...
+       begin_critical_section(S_NETCONFIGS);
+       newnetconfig = malloc(strlen(oldnetconfig) + 1024);
+       newnetconfig[0] = 0;
+
+       // Load the config lines in one by one, skipping any that reference this subscriber.  Also remove blank lines.
+       config_lines = num_tokens(oldnetconfig, '\n');
+       for (i=0; i<config_lines; ++i) {
+               char buf_email[256];
+               extract_token(buf, oldnetconfig, i, '\n', sizeof buf);
+               extract_token(buf_email, buf, 1, '|', sizeof buf_email);
+               if ( !IsEmptyStr(buf) && (strcasecmp(buf_email, emailaddr)) ) {
+                       sprintf(&newnetconfig[strlen(newnetconfig)], "%s\n", buf);
+               }
+       }
+
+       // We have now removed all lines containing the subscriber's email address.  This deletes any pending requests.
+       // If this was an unsubscribe operation, they're now gone from the list.
+       // But if this was a subscribe operation, we now need to add them.
+       if (cmd == SUBSCRIBE) {
+               sprintf(&newnetconfig[strlen(newnetconfig)], "listrecp|%s\n", emailaddr);
+       }
+
+       // write it back to disk
+       SaveRoomNetConfigFile(CC->room.QRnumber, newnetconfig);
+       end_critical_section(S_NETCONFIGS);
+       free(oldnetconfig);
+       free(newnetconfig);
+       cprintf("%d The pending request was confirmed.\n", CIT_OK);
+}
+
+
+// process subscribe/unsubscribe requests and confirmations
+void cmd_lsub(char *cmdbuf) {
+       char cmd[20];
+       char roomname[ROOMNAMELEN];
+       char emailaddr[1024];
+       char url[1024];
+       char generated_token[128];
+       char supplied_token[128];
+
+       extract_token(cmd, cmdbuf, 0, '|', sizeof cmd);                         // token 0 is the sub-command being sent
+       extract_token(roomname, cmdbuf, 1, '|', sizeof roomname);               // token 1 is always a room name
+       extract_token(emailaddr, cmdbuf, 2, '|', sizeof emailaddr);             // token 2 is the subscriber's email address
+       extract_token(url, cmdbuf, 3, '|', sizeof url);                         // token 3 is the URL at which we subscribed
+       extract_token(supplied_token, cmdbuf, 4, '|', sizeof supplied_token);   // token 4 is the token supplied by the caller
+
+       syslog(LOG_DEBUG, "cmd_lsub(cmd=%s, roomname=%s, emailaddr=%s, url=%s, token=%s",
+               cmd, roomname, emailaddr, url, supplied_token
+       );
+
+       // First confirm that the caller is referencing a room that actually exists.
+       if (CtdlGetRoom(&CC->room, roomname) != 0) {
+               cprintf("%d There is no list called '%s'\n", ERROR + ROOM_NOT_FOUND, roomname);
+               return;
+       }
+
+       if ((CC->room.QRflags2 & QR2_SELFLIST) == 0) {
+               cprintf("%d '%s' does not accept subscribe/unsubscribe requests.\n", ERROR + ROOM_NOT_FOUND, roomname);
+               return;
+       }
+
+       // Room confirmed, now parse the command.
+       generate_confirmation_token(generated_token, sizeof generated_token, roomname, emailaddr);
+
+       if (!strcasecmp(cmd, "subscribe")) {
+               send_subscribe_confirmation_email(roomname, emailaddr, url, generated_token);
+       }
+
+       else if (!strcasecmp(cmd, "unsubscribe")) {
+               send_unsubscribe_confirmation_email(roomname, emailaddr, url, generated_token);
+       }
+
+       else if (!strcasecmp(cmd, "confirm_subscribe")) {
+               do_confirm(SUBSCRIBE, roomname, emailaddr, url, generated_token, supplied_token);
+       }
+
+       else if (!strcasecmp(cmd, "confirm_unsubscribe")) {
+               do_confirm(UNSUBSCRIBE, roomname, emailaddr, url, generated_token, supplied_token);
+       }
+
+       else {                                                                  // sorry man, I can't deal with that
+               cprintf("%d Invalid command '%s'\n", ERROR + ILLEGAL_VALUE, cmd);
+       }
 }
 
 
@@ -61,7 +301,7 @@ CTDL_MODULE_INIT(listsub)
 {
        if (!threading)
        {
-               CtdlRegisterProtoHook(cmd_subs, "SUBS", "List subscribe/unsubscribe");
+               CtdlRegisterProtoHook(cmd_lsub, "LSUB", "List subscribe/unsubscribe");
        }
        
        /* return our module name for the log */