123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361 |
- package main
- import (
- "bytes"
- "database/sql"
- "html/template"
- "image"
- "image/png"
- "io"
- "net/http"
- "strconv"
- "strings"
- "github.com/knadh/listmonk/messenger"
- "github.com/knadh/listmonk/models"
- "github.com/knadh/listmonk/subimporter"
- "github.com/labstack/echo"
- "github.com/lib/pq"
- )
- const (
- tplMessage = "message"
- )
- // tplRenderer wraps a template.tplRenderer for echo.
- type tplRenderer struct {
- templates *template.Template
- RootURL string
- LogoURL string
- FaviconURL string
- }
- // tplData is the data container that is injected
- // into public templates for accessing data.
- type tplData struct {
- RootURL string
- LogoURL string
- FaviconURL string
- Data interface{}
- }
- type publicTpl struct {
- Title string
- Description string
- }
- type unsubTpl struct {
- publicTpl
- SubUUID string
- AllowBlacklist bool
- AllowExport bool
- AllowWipe bool
- }
- type optinTpl struct {
- publicTpl
- SubUUID string
- ListUUIDs []string `query:"l" form:"l"`
- Lists []models.List `query:"-" form:"-"`
- }
- type msgTpl struct {
- publicTpl
- MessageTitle string
- Message string
- }
- type subForm struct {
- subimporter.SubReq
- SubListUUIDs []string `form:"l"`
- }
- var (
- pixelPNG = drawTransparentImage(3, 14)
- )
- // Render executes and renders a template for echo.
- func (t *tplRenderer) Render(w io.Writer, name string, data interface{}, c echo.Context) error {
- return t.templates.ExecuteTemplate(w, name, tplData{
- RootURL: t.RootURL,
- LogoURL: t.LogoURL,
- FaviconURL: t.FaviconURL,
- Data: data,
- })
- }
- // handleSubscriptionPage renders the subscription management page and
- // handles unsubscriptions.
- func handleSubscriptionPage(c echo.Context) error {
- var (
- app = c.Get("app").(*App)
- campUUID = c.Param("campUUID")
- subUUID = c.Param("subUUID")
- unsub, _ = strconv.ParseBool(c.FormValue("unsubscribe"))
- blacklist, _ = strconv.ParseBool(c.FormValue("blacklist"))
- out = unsubTpl{}
- )
- out.SubUUID = subUUID
- out.Title = "Unsubscribe from mailing list"
- out.AllowBlacklist = app.Constants.Privacy.AllowBlacklist
- out.AllowExport = app.Constants.Privacy.AllowExport
- out.AllowWipe = app.Constants.Privacy.AllowWipe
- // Unsubscribe.
- if unsub {
- // Is blacklisting allowed?
- if !app.Constants.Privacy.AllowBlacklist {
- blacklist = false
- }
- if _, err := app.Queries.Unsubscribe.Exec(campUUID, subUUID, blacklist); err != nil {
- app.Logger.Printf("error unsubscribing: %v", err)
- return c.Render(http.StatusInternalServerError, tplMessage,
- makeMsgTpl("Error", "",
- `Error processing request. Please retry.`))
- }
- return c.Render(http.StatusOK, tplMessage,
- makeMsgTpl("Unsubscribed", "",
- `You have been successfully unsubscribed.`))
- }
- return c.Render(http.StatusOK, "subscription", out)
- }
- // handleOptinPage handles a double opt-in confirmation from subscribers.
- func handleOptinPage(c echo.Context) error {
- var (
- app = c.Get("app").(*App)
- subUUID = c.Param("subUUID")
- confirm, _ = strconv.ParseBool(c.FormValue("confirm"))
- out = optinTpl{}
- )
- out.SubUUID = subUUID
- out.Title = "Confirm subscriptions"
- out.SubUUID = subUUID
- // Get and validate fields.
- if err := c.Bind(&out); err != nil {
- return err
- }
- // Validate list UUIDs if there are incoming UUIDs in the request.
- if len(out.ListUUIDs) > 0 {
- for _, l := range out.ListUUIDs {
- if !reUUID.MatchString(l) {
- return c.Render(http.StatusBadRequest, tplMessage,
- makeMsgTpl("Invalid request", "",
- `One or more UUIDs in the request are invalid.`))
- }
- }
- }
- // Get the list of subscription lists where the subscriber hasn't confirmed.
- if err := app.Queries.GetSubscriberLists.Select(&out.Lists, 0, subUUID,
- nil, pq.StringArray(out.ListUUIDs), models.SubscriptionStatusUnconfirmed, nil); err != nil {
- app.Logger.Printf("error fetching lists for opt-in: %s", pqErrMsg(err))
- return c.Render(http.StatusInternalServerError, tplMessage,
- makeMsgTpl("Error", "", `Error fetching lists. Please retry.`))
- }
- // There are no lists to confirm.
- if len(out.Lists) == 0 {
- return c.Render(http.StatusInternalServerError, tplMessage,
- makeMsgTpl("No subscriptions", "",
- `There are no subscriptions to confirm.`))
- }
- // Confirm.
- if confirm {
- if _, err := app.Queries.ConfirmSubscriptionOptin.Exec(subUUID, pq.StringArray(out.ListUUIDs)); err != nil {
- app.Logger.Printf("error unsubscribing: %v", err)
- return c.Render(http.StatusInternalServerError, tplMessage,
- makeMsgTpl("Error", "",
- `Error processing request. Please retry.`))
- }
- return c.Render(http.StatusOK, tplMessage,
- makeMsgTpl("Confirmed", "",
- `Your subscriptions have been confirmed.`))
- }
- return c.Render(http.StatusOK, "optin", out)
- }
- // handleOptinPage handles a double opt-in confirmation from subscribers.
- func handleSubscriptionForm(c echo.Context) error {
- var (
- app = c.Get("app").(*App)
- req subForm
- )
- // Get and validate fields.
- if err := c.Bind(&req); err != nil {
- return err
- }
- if len(req.SubListUUIDs) == 0 {
- return c.Render(http.StatusInternalServerError, tplMessage,
- makeMsgTpl("Error", "",
- `No lists to subscribe to.`))
- }
- // If there's no name, use the name bit from the e-mail.
- req.Email = strings.ToLower(req.Email)
- if req.Name == "" {
- req.Name = strings.Split(req.Email, "@")[0]
- }
- // Validate fields.
- if err := subimporter.ValidateFields(req.SubReq); err != nil {
- return c.Render(http.StatusInternalServerError, tplMessage,
- makeMsgTpl("Error", "", err.Error()))
- }
- // Insert the subscriber into the DB.
- req.Status = models.SubscriberStatusEnabled
- req.ListUUIDs = pq.StringArray(req.SubListUUIDs)
- if _, err := insertSubscriber(req.SubReq, app); err != nil {
- return err
- }
- return c.Render(http.StatusInternalServerError, tplMessage,
- makeMsgTpl("Done", "", `Subscribed successfully.`))
- }
- // handleLinkRedirect handles link UUID to real link redirection.
- func handleLinkRedirect(c echo.Context) error {
- var (
- app = c.Get("app").(*App)
- linkUUID = c.Param("linkUUID")
- campUUID = c.Param("campUUID")
- subUUID = c.Param("subUUID")
- )
- var url string
- if err := app.Queries.RegisterLinkClick.Get(&url, linkUUID, campUUID, subUUID); err != nil {
- if err != sql.ErrNoRows {
- app.Logger.Printf("error fetching redirect link: %s", err)
- }
- return c.Render(http.StatusInternalServerError, tplMessage,
- makeMsgTpl("Error opening link", "",
- "There was an error opening the link. Please try later."))
- }
- return c.Redirect(http.StatusTemporaryRedirect, url)
- }
- // handleRegisterCampaignView registers a campaign view which comes in
- // the form of an pixel image request. Regardless of errors, this handler
- // should always render the pixel image bytes.
- func handleRegisterCampaignView(c echo.Context) error {
- var (
- app = c.Get("app").(*App)
- campUUID = c.Param("campUUID")
- subUUID = c.Param("subUUID")
- )
- if _, err := app.Queries.RegisterCampaignView.Exec(campUUID, subUUID); err != nil {
- app.Logger.Printf("error registering campaign view: %s", err)
- }
- c.Response().Header().Set("Cache-Control", "no-cache")
- return c.Blob(http.StatusOK, "image/png", pixelPNG)
- }
- // handleSelfExportSubscriberData pulls the subscriber's profile,
- // list subscriptions, campaign views and clicks and produces
- // a JSON report. This is a privacy feature and depends on the
- // configuration in app.Constants.Privacy.
- func handleSelfExportSubscriberData(c echo.Context) error {
- var (
- app = c.Get("app").(*App)
- subUUID = c.Param("subUUID")
- )
- // Is export allowed?
- if !app.Constants.Privacy.AllowExport {
- return c.Render(http.StatusBadRequest, tplMessage,
- makeMsgTpl("Invalid request", "", "The feature is not available."))
- }
- // Get the subscriber's data. A single query that gets the profile,
- // list subscriptions, campaign views, and link clicks. Names of
- // private lists are replaced with "Private list".
- data, b, err := exportSubscriberData(0, subUUID, app.Constants.Privacy.Exportable, app)
- if err != nil {
- app.Logger.Printf("error exporting subscriber data: %s", err)
- return c.Render(http.StatusInternalServerError, tplMessage,
- makeMsgTpl("Error processing request", "",
- "There was an error processing your request. Please try later."))
- }
- // Send the data out to the subscriber as an atachment.
- var msg bytes.Buffer
- if err := app.NotifTpls.ExecuteTemplate(&msg, notifSubscriberData, data); err != nil {
- app.Logger.Printf("error compiling notification template '%s': %v",
- notifSubscriberData, err)
- return c.Render(http.StatusInternalServerError, tplMessage,
- makeMsgTpl("Error preparing data", "",
- "There was an error preparing your data. Please try later."))
- }
- const fname = "profile.json"
- if err := app.Messenger.Push(app.Constants.FromEmail,
- []string{data.Email},
- "Your profile data",
- msg.Bytes(),
- []*messenger.Attachment{
- &messenger.Attachment{
- Name: fname,
- Content: b,
- Header: messenger.MakeAttachmentHeader(fname, "base64"),
- },
- },
- ); err != nil {
- app.Logger.Printf("error e-mailing subscriber profile: %s", err)
- return c.Render(http.StatusInternalServerError, tplMessage,
- makeMsgTpl("Error e-mailing data", "",
- "There was an error e-mailing your data. Please try later."))
- }
- return c.Render(http.StatusOK, tplMessage,
- makeMsgTpl("Data e-mailed", "",
- `Your data has been e-mailed to you as an attachment.`))
- }
- // handleWipeSubscriberData allows a subscriber to self-delete their data. The
- // profile and subscriptions are deleted, while the campaign_views and link
- // clicks remain as orphan data unconnected to any subscriber.
- func handleWipeSubscriberData(c echo.Context) error {
- var (
- app = c.Get("app").(*App)
- subUUID = c.Param("subUUID")
- )
- // Is wiping allowed?
- if !app.Constants.Privacy.AllowExport {
- return c.Render(http.StatusBadRequest, tplMessage,
- makeMsgTpl("Invalid request", "",
- "The feature is not available."))
- }
- if _, err := app.Queries.DeleteSubscribers.Exec(nil, pq.StringArray{subUUID}); err != nil {
- app.Logger.Printf("error wiping subscriber data: %s", err)
- return c.Render(http.StatusInternalServerError, tplMessage,
- makeMsgTpl("Error processing request", "",
- "There was an error processing your request. Please try later."))
- }
- return c.Render(http.StatusOK, tplMessage,
- makeMsgTpl("Data removed", "",
- `Your subscriptions and all associated data has been removed.`))
- }
- // drawTransparentImage draws a transparent PNG of given dimensions
- // and returns the PNG bytes.
- func drawTransparentImage(h, w int) []byte {
- var (
- img = image.NewRGBA(image.Rect(0, 0, w, h))
- out = &bytes.Buffer{}
- )
- _ = png.Encode(out, img)
- return out.Bytes()
- }
|