mirror of
https://github.com/foxcpp/maddy.git
synced 2025-04-04 21:47:40 +03:00
parent
c777be4c95
commit
e7d5418b88
8 changed files with 48 additions and 40 deletions
|
@ -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
|
||||||
}
|
}
|
||||||
|
|
|
@ -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
|
||||||
|
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
package sql
|
package imapsql
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"flag"
|
"flag"
|
|
@ -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)
|
||||||
}
|
}
|
|
@ -1,4 +1,4 @@
|
||||||
package sql
|
package imapsql
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"errors"
|
"errors"
|
|
@ -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"
|
10
maddy.conf
10
maddy.conf
|
@ -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
|
||||||
}
|
}
|
||||||
|
|
||||||
# ----------------------------------------------------------------------------
|
# ----------------------------------------------------------------------------
|
||||||
|
|
2
maddy.go
2
maddy.go
|
@ -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"
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue