123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462 |
- package models
- import (
- "bytes"
- "database/sql/driver"
- "encoding/json"
- "errors"
- "fmt"
- "html/template"
- "regexp"
- "strings"
- "time"
- "github.com/jmoiron/sqlx"
- "github.com/jmoiron/sqlx/types"
- "github.com/lib/pq"
- "github.com/yuin/goldmark"
- "github.com/yuin/goldmark/extension"
- "github.com/yuin/goldmark/renderer/html"
- null "gopkg.in/volatiletech/null.v6"
- )
- // Enum values for various statuses.
- const (
- // Subscriber.
- SubscriberStatusEnabled = "enabled"
- SubscriberStatusDisabled = "disabled"
- SubscriberStatusBlockListed = "blocklisted"
- // Subscription.
- SubscriptionStatusUnconfirmed = "unconfirmed"
- SubscriptionStatusConfirmed = "confirmed"
- SubscriptionStatusUnsubscribed = "unsubscribed"
- // Campaign.
- CampaignStatusDraft = "draft"
- CampaignStatusScheduled = "scheduled"
- CampaignStatusRunning = "running"
- CampaignStatusPaused = "paused"
- CampaignStatusFinished = "finished"
- CampaignStatusCancelled = "cancelled"
- CampaignTypeRegular = "regular"
- CampaignTypeOptin = "optin"
- CampaignContentTypeRichtext = "richtext"
- CampaignContentTypeHTML = "html"
- CampaignContentTypeMarkdown = "markdown"
- CampaignContentTypePlain = "plain"
- // List.
- ListTypePrivate = "private"
- ListTypePublic = "public"
- ListOptinSingle = "single"
- ListOptinDouble = "double"
- // User.
- UserTypeSuperadmin = "superadmin"
- UserTypeUser = "user"
- UserStatusEnabled = "enabled"
- UserStatusDisabled = "disabled"
- // BaseTpl is the name of the base template.
- BaseTpl = "base"
- // ContentTpl is the name of the compiled message.
- ContentTpl = "content"
- // Headers attached to e-mails for bounce tracking.
- EmailHeaderSubscriberUUID = "X-Listmonk-Subscriber"
- EmailHeaderCampaignUUID = "X-Listmonk-Campaign"
- BounceTypeHard = "hard"
- BounceTypeSoft = "soft"
- )
- // regTplFunc represents contains a regular expression for wrapping and
- // substituting a Go template function from the user's shorthand to a full
- // function call.
- type regTplFunc struct {
- regExp *regexp.Regexp
- replace string
- }
- // Regular expression for matching {{ Track "http://link.com" }} in the template
- // and substituting it with {{ Track "http://link.com" .Campaign.UUID .Subscriber.UUID }}
- // before compilation. This string gimmick is to make linking easier for users.
- var regTplFuncs = []regTplFunc{
- regTplFunc{
- regExp: regexp.MustCompile("{{(\\s+)?TrackLink\\s+?(\"|`)(.+?)(\"|`)(\\s+)?}}"),
- replace: `{{ TrackLink "$3" . }}`,
- },
- regTplFunc{
- regExp: regexp.MustCompile(`{{(\s+)?(TrackView|UnsubscribeURL|OptinURL|MessageURL)(\s+)?}}`),
- replace: `{{ $2 . }}`,
- },
- }
- // AdminNotifCallback is a callback function that's called
- // when a campaign's status changes.
- type AdminNotifCallback func(subject string, data interface{}) error
- // Base holds common fields shared across models.
- type Base struct {
- ID int `db:"id" json:"id"`
- CreatedAt null.Time `db:"created_at" json:"created_at"`
- UpdatedAt null.Time `db:"updated_at" json:"updated_at"`
- }
- // User represents an admin user.
- type User struct {
- Base
- Email string `json:"email"`
- Name string `json:"name"`
- Password string `json:"-"`
- Type string `json:"type"`
- Status string `json:"status"`
- }
- // Subscriber represents an e-mail subscriber.
- type Subscriber struct {
- Base
- UUID string `db:"uuid" json:"uuid"`
- Email string `db:"email" json:"email"`
- Name string `db:"name" json:"name"`
- Attribs SubscriberAttribs `db:"attribs" json:"attribs"`
- Status string `db:"status" json:"status"`
- Lists types.JSONText `db:"lists" json:"lists"`
- }
- type subLists struct {
- SubscriberID int `db:"subscriber_id"`
- Lists types.JSONText `db:"lists"`
- }
- // SubscriberAttribs is the map of key:value attributes of a subscriber.
- type SubscriberAttribs map[string]interface{}
- // Subscribers represents a slice of Subscriber.
- type Subscribers []Subscriber
- // SubscriberExport represents a subscriber record that is exported to raw data.
- type SubscriberExport struct {
- Base
- UUID string `db:"uuid" json:"uuid"`
- Email string `db:"email" json:"email"`
- Name string `db:"name" json:"name"`
- Attribs string `db:"attribs" json:"attribs"`
- Status string `db:"status" json:"status"`
- }
- // List represents a mailing list.
- type List struct {
- Base
- UUID string `db:"uuid" json:"uuid"`
- Name string `db:"name" json:"name"`
- Type string `db:"type" json:"type"`
- Optin string `db:"optin" json:"optin"`
- Tags pq.StringArray `db:"tags" json:"tags"`
- SubscriberCount int `db:"subscriber_count" json:"subscriber_count"`
- SubscriberID int `db:"subscriber_id" json:"-"`
- // This is only relevant when querying the lists of a subscriber.
- SubscriptionStatus string `db:"subscription_status" json:"subscription_status,omitempty"`
- // Pseudofield for getting the total number of subscribers
- // in searches and queries.
- Total int `db:"total" json:"-"`
- }
- // Campaign represents an e-mail campaign.
- type Campaign struct {
- Base
- CampaignMeta
- UUID string `db:"uuid" json:"uuid"`
- Type string `db:"type" json:"type"`
- Name string `db:"name" json:"name"`
- Subject string `db:"subject" json:"subject"`
- FromEmail string `db:"from_email" json:"from_email"`
- Body string `db:"body" json:"body"`
- AltBody null.String `db:"altbody" json:"altbody"`
- SendAt null.Time `db:"send_at" json:"send_at"`
- Status string `db:"status" json:"status"`
- ContentType string `db:"content_type" json:"content_type"`
- Tags pq.StringArray `db:"tags" json:"tags"`
- TemplateID int `db:"template_id" json:"template_id"`
- Messenger string `db:"messenger" json:"messenger"`
- // TemplateBody is joined in from templates by the next-campaigns query.
- TemplateBody string `db:"template_body" json:"-"`
- Tpl *template.Template `json:"-"`
- SubjectTpl *template.Template `json:"-"`
- AltBodyTpl *template.Template `json:"-"`
- // Pseudofield for getting the total number of subscribers
- // in searches and queries.
- Total int `db:"total" json:"-"`
- }
- // CampaignMeta contains fields tracking a campaign's progress.
- type CampaignMeta struct {
- CampaignID int `db:"campaign_id" json:"-"`
- Views int `db:"views" json:"views"`
- Clicks int `db:"clicks" json:"clicks"`
- Bounces int `db:"bounces" json:"bounces"`
- // This is a list of {list_id, name} pairs unlike Subscriber.Lists[]
- // because lists can be deleted after a campaign is finished, resulting
- // in null lists data to be returned. For that reason, campaign_lists maintains
- // campaign-list associations with a historical record of id + name that persist
- // even after a list is deleted.
- Lists types.JSONText `db:"lists" json:"lists"`
- StartedAt null.Time `db:"started_at" json:"started_at"`
- ToSend int `db:"to_send" json:"to_send"`
- Sent int `db:"sent" json:"sent"`
- }
- // Campaigns represents a slice of Campaigns.
- type Campaigns []Campaign
- // Template represents a reusable e-mail template.
- type Template struct {
- Base
- Name string `db:"name" json:"name"`
- Body string `db:"body" json:"body,omitempty"`
- IsDefault bool `db:"is_default" json:"is_default"`
- }
- // Bounce represents a single bounce event.
- type Bounce struct {
- ID int `db:"id" json:"id"`
- Type string `db:"type" json:"type"`
- Source string `db:"source" json:"source"`
- Meta json.RawMessage `db:"meta" json:"meta"`
- CreatedAt time.Time `db:"created_at" json:"created_at"`
- // One of these should be provided.
- Email string `db:"email" json:"email,omitempty"`
- SubscriberUUID string `db:"subscriber_uuid" json:"subscriber_uuid,omitempty"`
- SubscriberID int `db:"subscriber_id" json:"subscriber_id,omitempty"`
- CampaignUUID string `db:"campaign_uuid" json:"campaign_uuid,omitempty"`
- Campaign *json.RawMessage `db:"campaign" json:"campaign"`
- // Pseudofield for getting the total number of bounces
- // in searches and queries.
- Total int `db:"total" json:"-"`
- }
- // markdown is a global instance of Markdown parser and renderer.
- var markdown = goldmark.New(
- goldmark.WithRendererOptions(
- html.WithXHTML(),
- html.WithUnsafe(),
- ),
- goldmark.WithExtensions(
- extension.Table,
- extension.Strikethrough,
- extension.TaskList,
- ),
- )
- // GetIDs returns the list of subscriber IDs.
- func (subs Subscribers) GetIDs() []int {
- IDs := make([]int, len(subs))
- for i, c := range subs {
- IDs[i] = c.ID
- }
- return IDs
- }
- // LoadLists lazy loads the lists for all the subscribers
- // in the Subscribers slice and attaches them to their []Lists property.
- func (subs Subscribers) LoadLists(stmt *sqlx.Stmt) error {
- var sl []subLists
- err := stmt.Select(&sl, pq.Array(subs.GetIDs()))
- if err != nil {
- return err
- }
- if len(subs) != len(sl) {
- return errors.New("campaign stats count does not match")
- }
- for i, s := range sl {
- if s.SubscriberID == subs[i].ID {
- subs[i].Lists = s.Lists
- }
- }
- return nil
- }
- // Value returns the JSON marshalled SubscriberAttribs.
- func (s SubscriberAttribs) Value() (driver.Value, error) {
- return json.Marshal(s)
- }
- // Scan unmarshals JSON into SubscriberAttribs.
- func (s SubscriberAttribs) Scan(src interface{}) error {
- if data, ok := src.([]byte); ok {
- return json.Unmarshal(data, &s)
- }
- return fmt.Errorf("Could not not decode type %T -> %T", src, s)
- }
- // GetIDs returns the list of campaign IDs.
- func (camps Campaigns) GetIDs() []int {
- IDs := make([]int, len(camps))
- for i, c := range camps {
- IDs[i] = c.ID
- }
- return IDs
- }
- // LoadStats lazy loads campaign stats onto a list of campaigns.
- func (camps Campaigns) LoadStats(stmt *sqlx.Stmt) error {
- var meta []CampaignMeta
- if err := stmt.Select(&meta, pq.Array(camps.GetIDs())); err != nil {
- return err
- }
- if len(camps) != len(meta) {
- return errors.New("campaign stats count does not match")
- }
- for i, c := range meta {
- if c.CampaignID == camps[i].ID {
- camps[i].Lists = c.Lists
- camps[i].Views = c.Views
- camps[i].Clicks = c.Clicks
- camps[i].Bounces = c.Bounces
- }
- }
- return nil
- }
- // CompileTemplate compiles a campaign body template into its base
- // template and sets the resultant template to Campaign.Tpl.
- func (c *Campaign) CompileTemplate(f template.FuncMap) error {
- // Compile the base template.
- body := c.TemplateBody
- for _, r := range regTplFuncs {
- body = r.regExp.ReplaceAllString(body, r.replace)
- }
- baseTPL, err := template.New(BaseTpl).Funcs(f).Parse(body)
- if err != nil {
- return fmt.Errorf("error compiling base template: %v", err)
- }
- // If the format is markdown, convert Markdown to HTML.
- if c.ContentType == CampaignContentTypeMarkdown {
- var b bytes.Buffer
- if err := markdown.Convert([]byte(c.Body), &b); err != nil {
- return err
- }
- body = b.String()
- } else {
- body = c.Body
- }
- // Compile the campaign message.
- for _, r := range regTplFuncs {
- body = r.regExp.ReplaceAllString(body, r.replace)
- }
- msgTpl, err := template.New(ContentTpl).Funcs(f).Parse(body)
- if err != nil {
- return fmt.Errorf("error compiling message: %v", err)
- }
- out, err := baseTPL.AddParseTree(ContentTpl, msgTpl.Tree)
- if err != nil {
- return fmt.Errorf("error inserting child template: %v", err)
- }
- c.Tpl = out
- // If the subject line has a template string, compile it.
- if strings.Contains(c.Subject, "{{") {
- subj := c.Subject
- for _, r := range regTplFuncs {
- subj = r.regExp.ReplaceAllString(subj, r.replace)
- }
- subjTpl, err := template.New(ContentTpl).Funcs(f).Parse(subj)
- if err != nil {
- return fmt.Errorf("error compiling subject: %v", err)
- }
- c.SubjectTpl = subjTpl
- }
- if strings.Contains(c.AltBody.String, "{{") {
- b := c.AltBody.String
- for _, r := range regTplFuncs {
- b = r.regExp.ReplaceAllString(b, r.replace)
- }
- bTpl, err := template.New(ContentTpl).Funcs(f).Parse(b)
- if err != nil {
- return fmt.Errorf("error compiling alt plaintext message: %v", err)
- }
- c.AltBodyTpl = bTpl
- }
- return nil
- }
- // ConvertContent converts a campaign's body from one format to another,
- // for example, Markdown to HTML.
- func (c *Campaign) ConvertContent(from, to string) (string, error) {
- body := c.Body
- for _, r := range regTplFuncs {
- body = r.regExp.ReplaceAllString(body, r.replace)
- }
- // If the format is markdown, convert Markdown to HTML.
- var out string
- if from == CampaignContentTypeMarkdown &&
- (to == CampaignContentTypeHTML || to == CampaignContentTypeRichtext) {
- var b bytes.Buffer
- if err := markdown.Convert([]byte(c.Body), &b); err != nil {
- return out, err
- }
- out = b.String()
- } else {
- return out, errors.New("unknown formats to convert")
- }
- return out, nil
- }
- // FirstName splits the name by spaces and returns the first chunk
- // of the name that's greater than 2 characters in length, assuming
- // that it is the subscriber's first name.
- func (s Subscriber) FirstName() string {
- for _, s := range strings.Split(s.Name, " ") {
- if len(s) > 2 {
- return s
- }
- }
- return s.Name
- }
- // LastName splits the name by spaces and returns the last chunk
- // of the name that's greater than 2 characters in length, assuming
- // that it is the subscriber's last name.
- func (s Subscriber) LastName() string {
- chunks := strings.Split(s.Name, " ")
- for i := len(chunks) - 1; i >= 0; i-- {
- chunk := chunks[i]
- if len(chunk) > 2 {
- return chunk
- }
- }
- return s.Name
- }
|