storage/imapsql: Rename and clarify docs

See #212.
This commit is contained in:
fox.cpp 2020-03-05 21:03:57 +03:00
parent c777be4c95
commit e7d5418b88
No known key found for this signature in database
GPG key ID: E76D97CCEDE90B6C
8 changed files with 48 additions and 40 deletions

View file

@ -2,13 +2,13 @@ package main
import ( import (
"github.com/foxcpp/maddy/internal/config" "github.com/foxcpp/maddy/internal/config"
"github.com/foxcpp/maddy/internal/storage/sql" "github.com/foxcpp/maddy/internal/storage/imapsql"
_ "github.com/go-sql-driver/mysql" _ "github.com/go-sql-driver/mysql"
_ "github.com/lib/pq" _ "github.com/lib/pq"
) )
func sqlFromCfgBlock(root, node config.Node) (*sql.Storage, error) { func sqlFromCfgBlock(root, node config.Node) (*imapsql.Storage, error) {
// Global variables relevant for sql module. // Global variables relevant for sql module.
globals := config.NewMap(nil, root) globals := config.NewMap(nil, root)
// None now... // None now...
@ -23,7 +23,7 @@ func sqlFromCfgBlock(root, node config.Node) (*sql.Storage, error) {
instName = node.Args[0] instName = node.Args[0]
} }
mod, err := sql.New("sql", instName, nil, nil) mod, err := imapsql.New("sql", instName, nil, nil)
if err != nil { if err != nil {
return nil, err return nil, err
} }
@ -31,5 +31,5 @@ func sqlFromCfgBlock(root, node config.Node) (*sql.Storage, error) {
return nil, err return nil, err
} }
return mod.(*sql.Storage), nil return mod.(*imapsql.Storage), nil
} }

View file

@ -9,30 +9,34 @@ flags, message UIDs, etc defined as in RFC 3501.
This man page lists supported storage backends along with supported This man page lists supported storage backends along with supported
configuration directives for each. configuration directives for each.
Most likely, you are going to modules listed here in 'storage' directive for Most likely, you are going to use modules listed here in 'storage' directive
IMAP endpoint module (see *maddy-imap*(5)). for IMAP endpoint module (see *maddy-imap*(5)).
# SQL-based database module (sql) # SQL-based database module (imapsql)
Module that stores message metadata and indexes in a SQL-based relational The 'imapsql' module implements unified database for IMAP index and the user
database. credentials using SQL-based relational database. This allows easier management
as there is no storage accounts and no authentication accounts. There are just
accounts that can be created and removed using 'maddyctl' command.
Message contents are stored in an "external store", currently the only Message contents are stored in an "external store", currently the only
supported "external store" is a filesystem directory, used by default. supported "external store" is a filesystem directory, used by default.
All messages are stored in StateDirectory/messages under random IDs. By default, all messages are stored in StateDirectory/messages under random IDs.
Supported RDBMS: Supported RDBMS:
- SQLite 3.25.0 - SQLite 3.25.0
- PostgreSQL 9.6 or newer - PostgreSQL 9.6 or newer
Can be used as a storage backend (for IMAP), authentication backend (IMAP &
SMTP) or delivery target (SMTP).
Account names are required to have the form of a email address and are Account names are required to have the form of a email address and are
case-insensitive. UTF-8 names are supported with restrictions defined in the case-insensitive. UTF-8 names are supported with restrictions defined in the
PRECIS UsernameCaseMapped profile. PRECIS UsernameCaseMapped profile.
The database can be managed using the maddyctl utility. ```
imapsql {
driver sqlite3
dsn imapsql.db
}
```
## Arguments ## Arguments

View file

@ -1,4 +1,4 @@
package sql package imapsql
import ( import (
"flag" "flag"

View file

@ -1,11 +1,11 @@
// Package sql implements SQL-based storage module // Package imapsql implements SQL-based storage module
// using go-imap-sql library (github.com/foxcpp/go-imap-sql). // using go-imap-sql library (github.com/foxcpp/go-imap-sql).
// //
// Interfaces implemented: // Interfaces implemented:
// - module.StorageBackend // - module.StorageBackend
// - module.PlainAuth // - module.PlainAuth
// - module.DeliveryTarget // - module.DeliveryTarget
package sql package imapsql
import ( import (
"context" "context"
@ -187,12 +187,12 @@ func (store *Storage) InstanceName() string {
func New(_, instName string, _, inlineArgs []string) (module.Module, error) { func New(_, instName string, _, inlineArgs []string) (module.Module, error) {
store := &Storage{ store := &Storage{
instName: instName, instName: instName,
Log: log.Logger{Name: "sql"}, Log: log.Logger{Name: "imapsql"},
resolver: dns.DefaultResolver(), resolver: dns.DefaultResolver(),
} }
if len(inlineArgs) != 0 { if len(inlineArgs) != 0 {
if len(inlineArgs) == 1 { if len(inlineArgs) == 1 {
return nil, errors.New("sql: expected at least 2 arguments") return nil, errors.New("imapsql: expected at least 2 arguments")
} }
store.driver = inlineArgs[0] store.driver = inlineArgs[0]
@ -238,10 +238,10 @@ func (store *Storage) Init(cfg *config.Map) error {
} }
if dsn == nil { if dsn == nil {
return errors.New("sql: dsn is required") return errors.New("imapsql: dsn is required")
} }
if driver == "" { if driver == "" {
return errors.New("sql: driver is required") return errors.New("imapsql: driver is required")
} }
opts.Log = &store.Log opts.Log = &store.Log
@ -252,7 +252,7 @@ func (store *Storage) Init(cfg *config.Map) error {
// int is 32-bit on some platforms, so cut off values we can't actually // int is 32-bit on some platforms, so cut off values we can't actually
// use. // use.
if int(uint32(appendlimitVal)) != appendlimitVal { if int(uint32(appendlimitVal)) != appendlimitVal {
return errors.New("sql: appendlimit value is too big") return errors.New("imapsql: appendlimit value is too big")
} }
opts.MaxMsgBytes = new(uint32) opts.MaxMsgBytes = new(uint32)
*opts.MaxMsgBytes = uint32(appendlimitVal) *opts.MaxMsgBytes = uint32(appendlimitVal)
@ -273,24 +273,24 @@ func (store *Storage) Init(cfg *config.Map) error {
if len(compression) == 2 { if len(compression) == 2 {
opts.CompressAlgoParams = compression[1] opts.CompressAlgoParams = compression[1]
if _, err := strconv.Atoi(compression[1]); err != nil { if _, err := strconv.Atoi(compression[1]); err != nil {
return errors.New("sql: first argument for lz4 and zstd is compression level") return errors.New("imapsql: first argument for lz4 and zstd is compression level")
} }
} }
if len(compression) > 2 { if len(compression) > 2 {
return errors.New("sql: expected at most 2 arguments") return errors.New("imapsql: expected at most 2 arguments")
} }
case "off": case "off":
if len(compression) > 1 { if len(compression) > 1 {
return errors.New("sql: expected at most 1 arguments") return errors.New("imapsql: expected at most 1 arguments")
} }
default: default:
return errors.New("sql: unknown compression algorithm") return errors.New("imapsql: unknown compression algorithm")
} }
} }
store.Back, err = imapsql.New(driver, dsnStr, extStore, opts) store.Back, err = imapsql.New(driver, dsnStr, extStore, opts)
if err != nil { if err != nil {
return fmt.Errorf("sql: %s", err) return fmt.Errorf("imapsql: %s", err)
} }
store.Log.Debugln("go-imap-sql version", imapsql.VersionStr) store.Log.Debugln("go-imap-sql version", imapsql.VersionStr)
@ -306,7 +306,7 @@ func (store *Storage) EnableUpdatePipe(mode updatepipe.BackendMode) error {
return nil return nil
} }
if store.updates != nil { if store.updates != nil {
panic("sql: EnableUpdatePipe called after Updates") panic("imapsql: EnableUpdatePipe called after Updates")
} }
upds := store.Back.Updates() upds := store.Back.Updates()
@ -321,7 +321,7 @@ func (store *Storage) EnableUpdatePipe(mode updatepipe.BackendMode) error {
Log: log.Logger{Name: "sql/updpipe", Debug: store.Log.Debug}, Log: log.Logger{Name: "sql/updpipe", Debug: store.Log.Debug},
} }
default: default:
return errors.New("sql: driver does not have an update pipe implementation") return errors.New("imapsql: driver does not have an update pipe implementation")
} }
wrapped := make(chan backend.Update, cap(upds)*2) wrapped := make(chan backend.Update, cap(upds)*2)
@ -397,7 +397,7 @@ func (store *Storage) EnableChildrenExt() bool {
func prepareUsername(username string) (string, error) { func prepareUsername(username string) (string, error) {
mbox, domain, err := address.Split(username) mbox, domain, err := address.Split(username)
if err != nil { if err != nil {
return "", fmt.Errorf("sql: username prepare: %w", err) return "", fmt.Errorf("imapsql: username prepare: %w", err)
} }
// PRECIS is not included in the regular address.ForLookup since it reduces // PRECIS is not included in the regular address.ForLookup since it reduces
@ -409,12 +409,12 @@ func prepareUsername(username string) (string, error) {
// CompareKey and String. // CompareKey and String.
mbox, err = precis.UsernameCaseMapped.CompareKey(mbox) mbox, err = precis.UsernameCaseMapped.CompareKey(mbox)
if err != nil { if err != nil {
return "", fmt.Errorf("sql: username prepare: %w", err) return "", fmt.Errorf("imapsql: username prepare: %w", err)
} }
domain, err = dns.ForLookup(domain) domain, err = dns.ForLookup(domain)
if err != nil { if err != nil {
return "", fmt.Errorf("sql: username prepare: %w", err) return "", fmt.Errorf("imapsql: username prepare: %w", err)
} }
return mbox + "@" + domain, nil return mbox + "@" + domain, nil
@ -422,7 +422,7 @@ func prepareUsername(username string) (string, error) {
func (store *Storage) AuthPlain(username, password string) error { func (store *Storage) AuthPlain(username, password string) error {
// TODO: Pass session context there. // TODO: Pass session context there.
defer trace.StartRegion(context.Background(), "sql/AuthPlain").End() defer trace.StartRegion(context.Background(), "imapsql/AuthPlain").End()
accountName, err := prepareUsername(username) accountName, err := prepareUsername(username)
if err != nil { if err != nil {
@ -468,5 +468,5 @@ func (store *Storage) Close() error {
} }
func init() { func init() {
module.Register("sql", New) module.Register("imapsql", New)
} }

View file

@ -1,4 +1,4 @@
package sql package imapsql
import ( import (
"errors" "errors"

View file

@ -1,5 +1,5 @@
// +build !nosqlite3,cgo // +build !nosqlite3,cgo
package sql package imapsql
import _ "github.com/mattn/go-sqlite3" import _ "github.com/mattn/go-sqlite3"

View file

@ -1,4 +1,4 @@
## maddy 0.1 - default configuration file (2020-02-15T12:39Z) ## maddy 0.2 - default configuration file (2020-03-05)
# Suitable for small-scale deployments. Uses its own format for local users DB, # Suitable for small-scale deployments. Uses its own format for local users DB,
# should be managed via maddyctl utility. # should be managed via maddyctl utility.
# #
@ -21,9 +21,13 @@ tls /etc/maddy/certs/$(hostname)/fullchain.pem \
# ---------------------------------------------------------------------------- # ----------------------------------------------------------------------------
# Local storage & authentication # Local storage & authentication
sql local_mailboxes local_authdb { # imapsql modules provides unified database that is used both for user
# credentials and IMAP index. Use 'maddyctl users' utility to manage accounts
# and 'maddyctl imap-*' commands to inspect stored messages.
imapsql local_mailboxes local_authdb {
driver sqlite3 driver sqlite3
dsn all.db dsn imapsql.db
} }
# ---------------------------------------------------------------------------- # ----------------------------------------------------------------------------

View file

@ -34,7 +34,7 @@ import (
_ "github.com/foxcpp/maddy/internal/endpoint/smtp" _ "github.com/foxcpp/maddy/internal/endpoint/smtp"
_ "github.com/foxcpp/maddy/internal/modify" _ "github.com/foxcpp/maddy/internal/modify"
_ "github.com/foxcpp/maddy/internal/modify/dkim" _ "github.com/foxcpp/maddy/internal/modify/dkim"
_ "github.com/foxcpp/maddy/internal/storage/sql" _ "github.com/foxcpp/maddy/internal/storage/imapsql"
_ "github.com/foxcpp/maddy/internal/table" _ "github.com/foxcpp/maddy/internal/table"
_ "github.com/foxcpp/maddy/internal/target/queue" _ "github.com/foxcpp/maddy/internal/target/queue"
_ "github.com/foxcpp/maddy/internal/target/remote" _ "github.com/foxcpp/maddy/internal/target/remote"