mirror of
https://github.com/directorz/mailfull-go.git
synced 2025-12-18 10:07:03 +00:00
Implement some parts for loading data from a directory
This commit is contained in:
34
aliasdomain.go
Normal file
34
aliasdomain.go
Normal file
@@ -0,0 +1,34 @@
|
|||||||
|
package mailfull
|
||||||
|
|
||||||
|
// AliasDomain represents a AliasDomain.
|
||||||
|
type AliasDomain struct {
|
||||||
|
name string
|
||||||
|
target string
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewAliasDomain creates a new AliasDomain instance.
|
||||||
|
func NewAliasDomain(name, target string) (*AliasDomain, error) {
|
||||||
|
if !validAliasDomainName(name) {
|
||||||
|
return nil, ErrInvalidAliasDomainName
|
||||||
|
}
|
||||||
|
if !validAliasDomainTarget(target) {
|
||||||
|
return nil, ErrInvalidAliasDomainTarget
|
||||||
|
}
|
||||||
|
|
||||||
|
ad := &AliasDomain{
|
||||||
|
name: name,
|
||||||
|
target: target,
|
||||||
|
}
|
||||||
|
|
||||||
|
return ad, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Name returns name.
|
||||||
|
func (ad *AliasDomain) Name() string {
|
||||||
|
return ad.name
|
||||||
|
}
|
||||||
|
|
||||||
|
// Target returns target.
|
||||||
|
func (ad *AliasDomain) Target() string {
|
||||||
|
return ad.target
|
||||||
|
}
|
||||||
48
aliasuser.go
Normal file
48
aliasuser.go
Normal file
@@ -0,0 +1,48 @@
|
|||||||
|
package mailfull
|
||||||
|
|
||||||
|
import "errors"
|
||||||
|
|
||||||
|
// Errors for parameter.
|
||||||
|
var (
|
||||||
|
ErrNotEnoughAliasUserTargets = errors.New("AliasUser: targets not enough")
|
||||||
|
)
|
||||||
|
|
||||||
|
// AliasUser represents a AliasUser.
|
||||||
|
type AliasUser struct {
|
||||||
|
name string
|
||||||
|
targets []string
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewAliasUser creates a new AliasUser instance.
|
||||||
|
func NewAliasUser(name string, targets []string) (*AliasUser, error) {
|
||||||
|
if !validAliasUserName(name) {
|
||||||
|
return nil, ErrInvalidAliasUserName
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(targets) < 1 {
|
||||||
|
return nil, ErrNotEnoughAliasUserTargets
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, target := range targets {
|
||||||
|
if !validAliasUserTarget(target) {
|
||||||
|
return nil, ErrInvalidAliasUserTarget
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
au := &AliasUser{
|
||||||
|
name: name,
|
||||||
|
targets: targets,
|
||||||
|
}
|
||||||
|
|
||||||
|
return au, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Name returns name.
|
||||||
|
func (au *AliasUser) Name() string {
|
||||||
|
return au.name
|
||||||
|
}
|
||||||
|
|
||||||
|
// Targets returns targets.
|
||||||
|
func (au *AliasUser) Targets() []string {
|
||||||
|
return au.targets
|
||||||
|
}
|
||||||
24
catchalluser.go
Normal file
24
catchalluser.go
Normal file
@@ -0,0 +1,24 @@
|
|||||||
|
package mailfull
|
||||||
|
|
||||||
|
// CatchAllUser represents a CatchAllUser.
|
||||||
|
type CatchAllUser struct {
|
||||||
|
name string
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewCatchAllUser creates a new CatchAllUser instance.
|
||||||
|
func NewCatchAllUser(name string) (*CatchAllUser, error) {
|
||||||
|
if !validCatchAllUserName(name) {
|
||||||
|
return nil, ErrInvalidCatchAllUserName
|
||||||
|
}
|
||||||
|
|
||||||
|
cu := &CatchAllUser{
|
||||||
|
name: name,
|
||||||
|
}
|
||||||
|
|
||||||
|
return cu, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Name returns name.
|
||||||
|
func (cu *CatchAllUser) Name() string {
|
||||||
|
return cu.name
|
||||||
|
}
|
||||||
13
const.go
Normal file
13
const.go
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
package mailfull
|
||||||
|
|
||||||
|
// Filenames that are contained in the Repository.
|
||||||
|
const (
|
||||||
|
DirNameConfig = ".mailfull"
|
||||||
|
FileNameConfig = "config"
|
||||||
|
|
||||||
|
FileNameAliasDomains = ".valiasdomains"
|
||||||
|
FileNameUsersPassword = ".vpasswd"
|
||||||
|
FileNameUserForwards = ".forward"
|
||||||
|
FileNameAliasUsers = ".valiases"
|
||||||
|
FileNameCatchAllUser = ".vcatchall"
|
||||||
|
)
|
||||||
27
domain.go
Normal file
27
domain.go
Normal file
@@ -0,0 +1,27 @@
|
|||||||
|
package mailfull
|
||||||
|
|
||||||
|
// Domain represents a Domain.
|
||||||
|
type Domain struct {
|
||||||
|
name string
|
||||||
|
Users []*User
|
||||||
|
AliasUsers []*AliasUser
|
||||||
|
CatchAllUser *CatchAllUser
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewDomain creates a new Domain instance.
|
||||||
|
func NewDomain(name string) (*Domain, error) {
|
||||||
|
if !validDomainName(name) {
|
||||||
|
return nil, ErrInvalidDomainName
|
||||||
|
}
|
||||||
|
|
||||||
|
d := &Domain{
|
||||||
|
name: name,
|
||||||
|
}
|
||||||
|
|
||||||
|
return d, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Name returns name.
|
||||||
|
func (d *Domain) Name() string {
|
||||||
|
return d.name
|
||||||
|
}
|
||||||
399
repository.go
Normal file
399
repository.go
Normal file
@@ -0,0 +1,399 @@
|
|||||||
|
package mailfull
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bufio"
|
||||||
|
"errors"
|
||||||
|
"io/ioutil"
|
||||||
|
"os"
|
||||||
|
"path/filepath"
|
||||||
|
"strings"
|
||||||
|
"syscall"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Errors for the operation of the Repository.
|
||||||
|
var (
|
||||||
|
ErrDomainNotExist = errors.New("Domain: not exist")
|
||||||
|
ErrUserNotExist = errors.New("User: not exist")
|
||||||
|
|
||||||
|
ErrInvalidFormatUsersPassword = errors.New("User: password file invalid format")
|
||||||
|
ErrInvalidFormatAliasDomain = errors.New("AliasDomain: file invalid format")
|
||||||
|
ErrInvalidFormatAliasUsers = errors.New("AliasUsers: file invalid format")
|
||||||
|
)
|
||||||
|
|
||||||
|
// Repository represents a Repository.
|
||||||
|
type Repository struct {
|
||||||
|
*RepositoryConfig
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewRepository creates a new Repository instance.
|
||||||
|
func NewRepository(c *RepositoryConfig) (*Repository, error) {
|
||||||
|
r := &Repository{
|
||||||
|
RepositoryConfig: c,
|
||||||
|
}
|
||||||
|
|
||||||
|
return r, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Domains returns a Domain slice.
|
||||||
|
func (r *Repository) Domains() ([]*Domain, error) {
|
||||||
|
fileInfos, err := ioutil.ReadDir(r.DirMailDataPath)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
domains := make([]*Domain, 0, len(fileInfos))
|
||||||
|
|
||||||
|
for _, fileInfo := range fileInfos {
|
||||||
|
if !fileInfo.IsDir() {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
name := fileInfo.Name()
|
||||||
|
|
||||||
|
domain, err := NewDomain(name)
|
||||||
|
if err != nil {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
domains = append(domains, domain)
|
||||||
|
}
|
||||||
|
|
||||||
|
return domains, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Domain returns a Domain of the input name.
|
||||||
|
func (r *Repository) Domain(domainName string) (*Domain, error) {
|
||||||
|
if !validDomainName(domainName) {
|
||||||
|
return nil, ErrInvalidDomainName
|
||||||
|
}
|
||||||
|
|
||||||
|
fileInfo, err := os.Stat(filepath.Join(r.DirMailDataPath, domainName))
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if !fileInfo.IsDir() {
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
name := domainName
|
||||||
|
|
||||||
|
domain, err := NewDomain(name)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return domain, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// AliasDomains returns a AliasDomain slice.
|
||||||
|
func (r *Repository) AliasDomains() ([]*AliasDomain, error) {
|
||||||
|
file, err := os.Open(filepath.Join(r.DirMailDataPath, FileNameAliasDomains))
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
aliasDomains := make([]*AliasDomain, 0, 10)
|
||||||
|
|
||||||
|
scanner := bufio.NewScanner(file)
|
||||||
|
for scanner.Scan() {
|
||||||
|
words := strings.Split(scanner.Text(), ":")
|
||||||
|
if len(words) != 2 {
|
||||||
|
return nil, ErrInvalidFormatAliasDomain
|
||||||
|
}
|
||||||
|
|
||||||
|
name := words[0]
|
||||||
|
target := words[1]
|
||||||
|
|
||||||
|
aliasDomain, err := NewAliasDomain(name, target)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
aliasDomains = append(aliasDomains, aliasDomain)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := scanner.Err(); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return aliasDomains, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// AliasDomain returns a AliasDomain of the input name.
|
||||||
|
func (r *Repository) AliasDomain(aliasDomainName string) (*AliasDomain, error) {
|
||||||
|
aliasDomains, err := r.AliasDomains()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, aliasDomain := range aliasDomains {
|
||||||
|
if aliasDomain.Name() == aliasDomainName {
|
||||||
|
return aliasDomain, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Users returns a User slice.
|
||||||
|
func (r *Repository) Users(domainName string) ([]*User, error) {
|
||||||
|
domain, err := r.Domain(domainName)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
if domain == nil {
|
||||||
|
return nil, ErrDomainNotExist
|
||||||
|
}
|
||||||
|
|
||||||
|
hashedPasswords, err := r.usersHashedPassword(domainName)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
fileInfos, err := ioutil.ReadDir(filepath.Join(r.DirMailDataPath, domainName))
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
users := make([]*User, 0, len(fileInfos))
|
||||||
|
|
||||||
|
for _, fileInfo := range fileInfos {
|
||||||
|
if !fileInfo.IsDir() {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
name := fileInfo.Name()
|
||||||
|
|
||||||
|
forwards, err := r.userForwards(domainName, name)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
hashedPassword, ok := hashedPasswords[name]
|
||||||
|
if !ok {
|
||||||
|
hashedPassword = ""
|
||||||
|
}
|
||||||
|
|
||||||
|
user, err := NewUser(name, hashedPassword, forwards)
|
||||||
|
if err != nil {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
users = append(users, user)
|
||||||
|
}
|
||||||
|
|
||||||
|
return users, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// User returns a User of the input name.
|
||||||
|
func (r *Repository) User(domainName, userName string) (*User, error) {
|
||||||
|
domain, err := r.Domain(domainName)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
if domain == nil {
|
||||||
|
return nil, ErrDomainNotExist
|
||||||
|
}
|
||||||
|
|
||||||
|
if !validUserName(userName) {
|
||||||
|
return nil, ErrInvalidUserName
|
||||||
|
}
|
||||||
|
|
||||||
|
hashedPasswords, err := r.usersHashedPassword(domainName)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
fileInfo, err := os.Stat(filepath.Join(r.DirMailDataPath, domainName, userName))
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if !fileInfo.IsDir() {
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
name := userName
|
||||||
|
|
||||||
|
forwards, err := r.userForwards(domainName, name)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
hashedPassword, ok := hashedPasswords[name]
|
||||||
|
if !ok {
|
||||||
|
hashedPassword = ""
|
||||||
|
}
|
||||||
|
|
||||||
|
user, err := NewUser(name, hashedPassword, forwards)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return user, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// usersHashedPassword returns a string map of usernames to the hashed password.
|
||||||
|
func (r *Repository) usersHashedPassword(domainName string) (map[string]string, error) {
|
||||||
|
domain, err := r.Domain(domainName)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
if domain == nil {
|
||||||
|
return nil, ErrDomainNotExist
|
||||||
|
}
|
||||||
|
|
||||||
|
file, err := os.Open(filepath.Join(r.DirMailDataPath, domainName, FileNameUsersPassword))
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
hashedPasswords := map[string]string{}
|
||||||
|
|
||||||
|
scanner := bufio.NewScanner(file)
|
||||||
|
for scanner.Scan() {
|
||||||
|
words := strings.Split(scanner.Text(), ":")
|
||||||
|
if len(words) != 2 {
|
||||||
|
return nil, ErrInvalidFormatUsersPassword
|
||||||
|
}
|
||||||
|
|
||||||
|
name := words[0]
|
||||||
|
hashedPassword := words[1]
|
||||||
|
|
||||||
|
hashedPasswords[name] = hashedPassword
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := scanner.Err(); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return hashedPasswords, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// userForwards returns a string slice of forwards that the input name has.
|
||||||
|
func (r *Repository) userForwards(domainName, userName string) ([]string, error) {
|
||||||
|
user, err := r.User(domainName, userName)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
if user == nil {
|
||||||
|
return nil, ErrUserNotExist
|
||||||
|
}
|
||||||
|
|
||||||
|
file, err := os.Open(filepath.Join(r.DirMailDataPath, domainName, userName, FileNameUserForwards))
|
||||||
|
if err != nil {
|
||||||
|
if err.(*os.PathError).Err == syscall.ENOENT {
|
||||||
|
return []string{}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
forwards := make([]string, 0, 5)
|
||||||
|
|
||||||
|
scanner := bufio.NewScanner(file)
|
||||||
|
for scanner.Scan() {
|
||||||
|
forwards = append(forwards, scanner.Text())
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := scanner.Err(); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return forwards, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// AliasUsers returns a AliasUser slice.
|
||||||
|
func (r *Repository) AliasUsers(domainName string) ([]*AliasUser, error) {
|
||||||
|
domain, err := r.Domain(domainName)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
if domain == nil {
|
||||||
|
return nil, ErrDomainNotExist
|
||||||
|
}
|
||||||
|
|
||||||
|
file, err := os.Open(filepath.Join(r.DirMailDataPath, domainName, FileNameAliasUsers))
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
aliasUsers := make([]*AliasUser, 0, 50)
|
||||||
|
|
||||||
|
scanner := bufio.NewScanner(file)
|
||||||
|
for scanner.Scan() {
|
||||||
|
words := strings.Split(scanner.Text(), ":")
|
||||||
|
if len(words) != 2 {
|
||||||
|
return nil, ErrInvalidFormatAliasUsers
|
||||||
|
}
|
||||||
|
|
||||||
|
name := words[0]
|
||||||
|
targets := strings.Split(words[1], ",")
|
||||||
|
|
||||||
|
aliasUser, err := NewAliasUser(name, targets)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
aliasUsers = append(aliasUsers, aliasUser)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := scanner.Err(); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return aliasUsers, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// AliasUser returns a AliasUser of the input name.
|
||||||
|
func (r *Repository) AliasUser(domainName, aliasUserName string) (*AliasUser, error) {
|
||||||
|
aliasUsers, err := r.AliasUsers(domainName)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, aliasUser := range aliasUsers {
|
||||||
|
if aliasUser.Name() == aliasUserName {
|
||||||
|
return aliasUser, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// CatchAllUser returns a CatchAllUser that the input name has.
|
||||||
|
func (r *Repository) CatchAllUser(domainName string) (*CatchAllUser, error) {
|
||||||
|
domain, err := r.Domain(domainName)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
if domain == nil {
|
||||||
|
return nil, ErrDomainNotExist
|
||||||
|
}
|
||||||
|
|
||||||
|
file, err := os.Open(filepath.Join(r.DirMailDataPath, domainName, FileNameCatchAllUser))
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
scanner := bufio.NewScanner(file)
|
||||||
|
scanner.Scan()
|
||||||
|
|
||||||
|
name := scanner.Text()
|
||||||
|
|
||||||
|
if err := scanner.Err(); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if name == "" {
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
catchAllUser, err := NewCatchAllUser(name)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return catchAllUser, nil
|
||||||
|
}
|
||||||
214
repositoryconfig.go
Normal file
214
repositoryconfig.go
Normal file
@@ -0,0 +1,214 @@
|
|||||||
|
package mailfull
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"os"
|
||||||
|
"path/filepath"
|
||||||
|
"syscall"
|
||||||
|
|
||||||
|
"github.com/BurntSushi/toml"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Errors for the Repository.
|
||||||
|
var (
|
||||||
|
ErrInvalidRepository = errors.New("invalid repository")
|
||||||
|
ErrNotRepository = errors.New("not a Mailfull repository (or any of the parent directories)")
|
||||||
|
ErrRepositoryExist = errors.New("a Mailfull repository exists")
|
||||||
|
)
|
||||||
|
|
||||||
|
// RepositoryConfig is used to configure a Repository.
|
||||||
|
type RepositoryConfig struct {
|
||||||
|
DirDatabasePath string `toml:"dir_database"`
|
||||||
|
DirMailDataPath string `toml:"dir_maildata"`
|
||||||
|
Username string `toml:"username"`
|
||||||
|
Groupname string `toml:"groupname"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// DefaultRepositoryConfig returns a RepositoryConfig with default parameter.
|
||||||
|
func DefaultRepositoryConfig() *RepositoryConfig {
|
||||||
|
c := &RepositoryConfig{
|
||||||
|
DirDatabasePath: "./etc",
|
||||||
|
DirMailDataPath: "./domains",
|
||||||
|
Username: "mailfull",
|
||||||
|
Groupname: "mailfull",
|
||||||
|
}
|
||||||
|
|
||||||
|
return c
|
||||||
|
}
|
||||||
|
|
||||||
|
// OpenRepository opens a Repository and creates a new Repository instance.
|
||||||
|
func OpenRepository(basePath string) (*Repository, error) {
|
||||||
|
rootPath, err := filepath.Abs(basePath)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
for {
|
||||||
|
configDirPath := filepath.Join(rootPath, DirNameConfig)
|
||||||
|
|
||||||
|
fi, errStat := os.Stat(configDirPath)
|
||||||
|
if errStat != nil {
|
||||||
|
if errStat.(*os.PathError).Err != syscall.ENOENT {
|
||||||
|
return nil, errStat
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if fi.IsDir() {
|
||||||
|
break
|
||||||
|
} else {
|
||||||
|
return nil, ErrInvalidRepository
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
parentPath := filepath.Clean(filepath.Join(rootPath, ".."))
|
||||||
|
if rootPath == parentPath {
|
||||||
|
return nil, ErrNotRepository
|
||||||
|
}
|
||||||
|
rootPath = parentPath
|
||||||
|
}
|
||||||
|
|
||||||
|
configFilePath := filepath.Join(rootPath, DirNameConfig, FileNameConfig)
|
||||||
|
|
||||||
|
fi, err := os.Stat(configFilePath)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
if fi.IsDir() {
|
||||||
|
return nil, ErrInvalidRepository
|
||||||
|
}
|
||||||
|
|
||||||
|
configFile, err := os.Open(configFilePath)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
defer configFile.Close()
|
||||||
|
|
||||||
|
c := DefaultRepositoryConfig()
|
||||||
|
if _, err = toml.DecodeReader(configFile, c); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if !filepath.IsAbs(c.DirDatabasePath) {
|
||||||
|
c.DirDatabasePath = filepath.Join(rootPath, c.DirDatabasePath)
|
||||||
|
}
|
||||||
|
if !filepath.IsAbs(c.DirMailDataPath) {
|
||||||
|
c.DirMailDataPath = filepath.Join(rootPath, c.DirMailDataPath)
|
||||||
|
}
|
||||||
|
|
||||||
|
r, err := NewRepository(c)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return r, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// InitRepository initializes the input directory as a Repository.
|
||||||
|
func InitRepository(rootPath string) error {
|
||||||
|
rootPath, err := filepath.Abs(rootPath)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
configDirPath := filepath.Join(rootPath, DirNameConfig)
|
||||||
|
|
||||||
|
fi, err := os.Stat(configDirPath)
|
||||||
|
if err != nil {
|
||||||
|
if err.(*os.PathError).Err == syscall.ENOENT {
|
||||||
|
if err = os.Mkdir(configDirPath, 0777); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if !fi.IsDir() {
|
||||||
|
return ErrInvalidRepository
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
configFilePath := filepath.Join(configDirPath, FileNameConfig)
|
||||||
|
|
||||||
|
fi, err = os.Stat(configFilePath)
|
||||||
|
if err != nil {
|
||||||
|
if err.(*os.PathError).Err != syscall.ENOENT {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if fi.IsDir() {
|
||||||
|
return ErrInvalidRepository
|
||||||
|
}
|
||||||
|
|
||||||
|
return ErrRepositoryExist
|
||||||
|
}
|
||||||
|
|
||||||
|
configFile, err := os.Create(configFilePath)
|
||||||
|
if err != nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
defer configFile.Close()
|
||||||
|
|
||||||
|
c := DefaultRepositoryConfig()
|
||||||
|
|
||||||
|
enc := toml.NewEncoder(configFile)
|
||||||
|
if err := enc.Encode(c); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
if !filepath.IsAbs(c.DirDatabasePath) {
|
||||||
|
c.DirDatabasePath = filepath.Join(rootPath, c.DirDatabasePath)
|
||||||
|
}
|
||||||
|
if !filepath.IsAbs(c.DirMailDataPath) {
|
||||||
|
c.DirMailDataPath = filepath.Join(rootPath, c.DirMailDataPath)
|
||||||
|
}
|
||||||
|
|
||||||
|
fi, err = os.Stat(c.DirDatabasePath)
|
||||||
|
if err != nil {
|
||||||
|
if err.(*os.PathError).Err == syscall.ENOENT {
|
||||||
|
if err = os.Mkdir(c.DirDatabasePath, 0777); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if !fi.IsDir() {
|
||||||
|
return ErrInvalidRepository
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fi, err = os.Stat(c.DirMailDataPath)
|
||||||
|
if err != nil {
|
||||||
|
if err.(*os.PathError).Err == syscall.ENOENT {
|
||||||
|
if err = os.Mkdir(c.DirMailDataPath, 0777); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if !fi.IsDir() {
|
||||||
|
return ErrInvalidRepository
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
aliasDomainFileName := filepath.Join(c.DirMailDataPath, FileNameAliasDomains)
|
||||||
|
|
||||||
|
fi, err = os.Stat(aliasDomainFileName)
|
||||||
|
if err != nil {
|
||||||
|
if err.(*os.PathError).Err != syscall.ENOENT {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if fi.IsDir() {
|
||||||
|
return ErrInvalidRepository
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
aliasDomainFile, err := os.Create(aliasDomainFileName)
|
||||||
|
if err != nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
defer aliasDomainFile.Close()
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
38
user.go
Normal file
38
user.go
Normal file
@@ -0,0 +1,38 @@
|
|||||||
|
package mailfull
|
||||||
|
|
||||||
|
// User represents a User.
|
||||||
|
type User struct {
|
||||||
|
name string
|
||||||
|
hashedPassword string
|
||||||
|
forwards []string
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewUser creates a new User instance.
|
||||||
|
func NewUser(name, hashedPassword string, forwards []string) (*User, error) {
|
||||||
|
if !validUserName(name) {
|
||||||
|
return nil, ErrInvalidUserName
|
||||||
|
}
|
||||||
|
|
||||||
|
u := &User{
|
||||||
|
name: name,
|
||||||
|
hashedPassword: hashedPassword,
|
||||||
|
forwards: forwards,
|
||||||
|
}
|
||||||
|
|
||||||
|
return u, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Name returns name.
|
||||||
|
func (u *User) Name() string {
|
||||||
|
return u.name
|
||||||
|
}
|
||||||
|
|
||||||
|
// HashedPassword returns hashedPassword.
|
||||||
|
func (u *User) HashedPassword() string {
|
||||||
|
return u.hashedPassword
|
||||||
|
}
|
||||||
|
|
||||||
|
// Forwards returns forwards.
|
||||||
|
func (u *User) Forwards() []string {
|
||||||
|
return u.forwards
|
||||||
|
}
|
||||||
52
valid.go
Normal file
52
valid.go
Normal file
@@ -0,0 +1,52 @@
|
|||||||
|
package mailfull
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"regexp"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Errors for incorrect format.
|
||||||
|
var (
|
||||||
|
ErrInvalidDomainName = errors.New("Domain: name incorrect format")
|
||||||
|
ErrInvalidAliasDomainName = errors.New("AliasDomain: name incorrect format")
|
||||||
|
ErrInvalidAliasDomainTarget = errors.New("AliasDomain: target incorrect format")
|
||||||
|
ErrInvalidUserName = errors.New("User: name incorrect format")
|
||||||
|
ErrInvalidAliasUserName = errors.New("AliasUser: name incorrect format")
|
||||||
|
ErrInvalidAliasUserTarget = errors.New("AliasUser: target incorrect format")
|
||||||
|
ErrInvalidCatchAllUserName = errors.New("CatchAllUser: name incorrect format")
|
||||||
|
)
|
||||||
|
|
||||||
|
// validDomainName returns true if the input is correct format.
|
||||||
|
func validDomainName(name string) bool {
|
||||||
|
return regexp.MustCompile(`^([A-Za-z0-9][A-Za-z0-9\-]{1,61}[A-Za-z0-9]\.)*[A-Za-z]+$`).MatchString(name)
|
||||||
|
}
|
||||||
|
|
||||||
|
// validAliasDomainName returns true if the input is correct format.
|
||||||
|
func validAliasDomainName(name string) bool {
|
||||||
|
return regexp.MustCompile(`^([A-Za-z0-9][A-Za-z0-9\-]{1,61}[A-Za-z0-9]\.)*[A-Za-z]+$`).MatchString(name)
|
||||||
|
}
|
||||||
|
|
||||||
|
// validAliasDomainTarget returns true if the input is correct format.
|
||||||
|
func validAliasDomainTarget(target string) bool {
|
||||||
|
return regexp.MustCompile(`^([A-Za-z0-9][A-Za-z0-9\-]{1,61}[A-Za-z0-9]\.)*[A-Za-z]+$`).MatchString(target)
|
||||||
|
}
|
||||||
|
|
||||||
|
// validUserName returns true if the input is correct format.
|
||||||
|
func validUserName(name string) bool {
|
||||||
|
return regexp.MustCompile(`^[^\s]+$`).MatchString(name)
|
||||||
|
}
|
||||||
|
|
||||||
|
// validAliasUserName returns true if the input is correct format.
|
||||||
|
func validAliasUserName(name string) bool {
|
||||||
|
return regexp.MustCompile(`^[^\s]+$`).MatchString(name)
|
||||||
|
}
|
||||||
|
|
||||||
|
// validAliasUserTarget returns true if the input is correct format.
|
||||||
|
func validAliasUserTarget(target string) bool {
|
||||||
|
return regexp.MustCompile(`^[^\s]+@([A-Za-z0-9][A-Za-z0-9\-]{1,61}[A-Za-z0-9]\.)*[A-Za-z]+$`).MatchString(target)
|
||||||
|
}
|
||||||
|
|
||||||
|
// validCatchAllUserName returns true if the input is correct format.
|
||||||
|
func validCatchAllUserName(name string) bool {
|
||||||
|
return regexp.MustCompile(`^[^\s]+$`).MatchString(name)
|
||||||
|
}
|
||||||
4
version.go
Normal file
4
version.go
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
package mailfull
|
||||||
|
|
||||||
|
// Version is a version number.
|
||||||
|
const Version = "0.0.0"
|
||||||
Reference in New Issue
Block a user