maddy/internal/auth/shadow/read.go
fox.cpp bf188e454f
Move most code from the repo root into subdirectories
The intention is to keep to repo root clean while the list of packages
is slowly growing.

Additionally, a bunch of small (~30 LoC) files in the repo root is
merged into a single maddy.go file, for the same reason.

Most of the internal code is moved into the internal/ directory. Go
toolchain will make it impossible to import these packages from external
applications.

Some packages are renamed and moved into the pkg/ directory in the root.
According to https://github.com/golang-standards/project-layout this is
the de-facto standard to place "library code that's ok to use by
external applications" in.

To clearly define the purpose of top-level directories, README.md files
are added to each.
2019-12-06 01:35:12 +03:00

80 lines
1.5 KiB
Go

package shadow
import (
"bufio"
"errors"
"fmt"
"os"
"strconv"
"strings"
)
var ErrNoSuchUser = errors.New("shadow: user entry is not present in database")
var ErrWrongPassword = errors.New("shadow: wrong password")
// Read reads system shadow passwords database and returns all entires in it.
func Read() ([]Entry, error) {
f, err := os.Open("/etc/shadow")
if err != nil {
return nil, err
}
scnr := bufio.NewScanner(f)
var res []Entry
for scnr.Scan() {
ent, err := parseEntry(scnr.Text())
if err != nil {
return res, err
}
res = append(res, *ent)
}
if err := scnr.Err(); err != nil {
return res, err
}
return res, nil
}
func parseEntry(line string) (*Entry, error) {
parts := strings.Split(line, ":")
if len(parts) != 9 {
return nil, errors.New("read: malformed entry")
}
res := &Entry{
Name: parts[0],
Pass: parts[1],
}
for i, value := range [...]*int{
&res.LastChange, &res.MinPassAge, &res.MaxPassAge,
&res.WarnPeriod, &res.InactivityPeriod, &res.AcctExpiry, &res.Flags,
} {
if parts[2+i] == "" {
*value = -1
} else {
var err error
*value, err = strconv.Atoi(parts[2+i])
if err != nil {
return nil, fmt.Errorf("read: invalid value for field %d", 2+i)
}
}
}
return res, nil
}
func Lookup(name string) (*Entry, error) {
entries, err := Read()
if err != nil {
return nil, err
}
for _, entry := range entries {
if entry.Name == name {
return &entry, nil
}
}
return nil, ErrNoSuchUser
}