Compare commits

..

No commits in common. "418e4a4a149fed1a40021eb7f3a4f6c5c05492c7" and "cbc868ae35078fa7d4a2496381e156eeb8a36252" have entirely different histories.

5 changed files with 49 additions and 67 deletions

View File

@ -18,7 +18,6 @@ import (
const keyfileName = "authorized_keys2" const keyfileName = "authorized_keys2"
func quit(msg string, code int) { func quit(msg string, code int) {
// TODO print to stderr
fmt.Println(msg) fmt.Println(msg)
os.Exit(code) os.Exit(code)
} }

View File

@ -1,9 +1,7 @@
package main package main
import ( import (
"errors"
"fmt" "fmt"
"os"
"git.tilde.town/tildetown/town/email" "git.tilde.town/tildetown/town/email"
) )
@ -25,27 +23,8 @@ Follow the instructions there to add your new key and restore access to your acc
best, best,
~vilmibm` ~vilmibm`
func loadPassword() (string, error) {
f, err := os.Open("/town/docs/smtp_help.pw")
if err != nil {
return "", fmt.Errorf("could not open smtp password file: %w", err)
}
pw := make([]byte, 100)
n, err := f.Read(pw)
if err != nil {
return "", fmt.Errorf("could not read smtp password file: %w", err)
}
if n == 0 {
return "", errors.New("smtp password file was empty")
}
return string(pw[0:n]), nil
}
func sendAuthCodeEmail(ac AuthCode) error { func sendAuthCodeEmail(ac AuthCode) error {
pw, err := loadPassword() pw, err := email.LoadPassword()
if err != nil { if err != nil {
return err return err
} }

View File

@ -5,7 +5,6 @@ import (
"database/sql" "database/sql"
"errors" "errors"
"fmt" "fmt"
"log"
"os" "os"
"os/exec" "os/exec"
"strconv" "strconv"
@ -118,10 +117,6 @@ func (p *Prompter) Select(prompt string, opts []string) (int, error) {
} }
func _main(cs colorScheme) error { func _main(cs colorScheme) error {
logFilename := fmt.Sprintf("/town/var/log/help/%d", time.Now().Unix)
logFile, err := os.OpenFile(logFilename, os.O_WRONLY|os.O_CREATE|os.O_EXCL, 0600)
l := log.New(logFile, "", log.Ldate|log.Ltime|log.LUTC|log.Lshortfile)
db, err := connectDB() db, err := connectDB()
if err != nil { if err != nil {
return fmt.Errorf("could not connect to database. please let root@tilde.town know about this.") return fmt.Errorf("could not connect to database. please let root@tilde.town know about this.")
@ -153,9 +148,9 @@ func _main(cs colorScheme) error {
switch c { switch c {
case 0: case 0:
return collectEmail(l, db, cs, p) return collectEmail(db, cs, p)
case 1: case 1:
return redeemCode(l, db, cs, p) return redeemCode(db, cs, p)
case 2: case 2:
return nil return nil
} }
@ -176,7 +171,7 @@ func emailToUsername(email string) (string, error) {
return stdoutBuff.String(), nil return stdoutBuff.String(), nil
} }
func collectEmail(l *log.Logger, db *sql.DB, cs colorScheme, p *Prompter) error { func collectEmail(db *sql.DB, cs colorScheme, p *Prompter) error {
fmt.Println(cs.Header("We can send a authorization code to an email associated with your town account.")) fmt.Println(cs.Header("We can send a authorization code to an email associated with your town account."))
email, err := p.String("email to send reset code to?") email, err := p.String("email to send reset code to?")
if err != nil { if err != nil {
@ -196,12 +191,12 @@ func collectEmail(l *log.Logger, db *sql.DB, cs colorScheme, p *Prompter) error
} }
} }
if found != len(mustHave) { if found != len(mustHave) {
l.Printf("corrupt email '%s'", email) // TODO log
return nil return nil
} }
if _, err = emailToUsername(email); err != nil { if _, err = emailToUsername(email); err != nil {
l.Printf("no user for '%s'", email) // TODO log
return nil return nil
} }
@ -213,30 +208,31 @@ func collectEmail(l *log.Logger, db *sql.DB, cs colorScheme, p *Prompter) error
} }
if err = ac.Insert(db); err != nil { if err = ac.Insert(db); err != nil {
l.Printf("database error: %s", err.Error()) // TODO log
return errors.New("the database was sad") return err
} }
if err = sendAuthCodeEmail(*ac); err != nil { if err = sendAuthCodeEmail(*ac); err != nil {
l.Printf("mail send error: %s", err.Error()) // TODO log
return errors.New("email sending failed") return err
} }
return nil return nil
} }
func redeemCode(l *log.Logger, db *sql.DB, cs colorScheme, p *Prompter) error { func redeemCode(db *sql.DB, cs colorScheme, p *Prompter) error {
fmt.Println(cs.Header("redeem an auth code and add a new public key")) fmt.Println(cs.Header("redeem an auth code and add a new public key"))
fmt.Println()
c, err := p.String("paste your auth code and hit enter to submit:") c, err := p.String("paste your auth code and hit enter to submit:")
if err != nil { if err != nil {
l.Printf("failed to prompt: %s", err.Error()) // TODO log
fmt.Println(cs.Error("sorry, I couldn't read that.")) fmt.Println(cs.Error("sorry, I couldn't read that."))
return nil return nil
} }
parts, err := codes.Decode(c) parts, err := codes.Decode(c)
if err != nil { if err != nil {
l.Printf("failed to decode auth code: %s", err.Error()) // TODO log
fmt.Println(cs.Error("sorry, that doesn't look like an auth code...")) fmt.Println(cs.Error("sorry, that doesn't look like an auth code..."))
return nil return nil
} }
@ -248,7 +244,7 @@ func redeemCode(l *log.Logger, db *sql.DB, cs colorScheme, p *Prompter) error {
err = code.Hydrate(db) err = code.Hydrate(db)
if err != nil { if err != nil {
l.Printf("hydrate failed: %s", err.Error()) // TODO log
return errors.New("the database is sad") return errors.New("the database is sad")
} }
@ -259,14 +255,14 @@ func redeemCode(l *log.Logger, db *sql.DB, cs colorScheme, p *Prompter) error {
username, err := emailToUsername(code.Email) username, err := emailToUsername(code.Email)
if err != nil { if err != nil {
l.Printf("could not find user: %s", err.Error())
fmt.Println(cs.Error("That code doesn't seem to match an account.")) fmt.Println(cs.Error("That code doesn't seem to match an account."))
// TODO log
return nil return nil
} }
key, err := p.String("paste your new public key and hit enter to submit:") key, err := p.String("paste your new public key and hit enter to submit:")
if err != nil { if err != nil {
l.Printf("failed to prompt: %s", err.Error()) // TODO log
fmt.Println(cs.Error("sorry, I couldn't read that.")) fmt.Println(cs.Error("sorry, I couldn't read that."))
return nil return nil
} }
@ -287,19 +283,27 @@ func redeemCode(l *log.Logger, db *sql.DB, cs colorScheme, p *Prompter) error {
stdoutBuff := bytes.NewBuffer([]byte{}) stdoutBuff := bytes.NewBuffer([]byte{})
cmd.Stdout = stdoutBuff cmd.Stdout = stdoutBuff
if err = cmd.Run(); err != nil { if err = cmd.Run(); err != nil {
l.Printf("appendkeyfile failed with '%s': %s", stdoutBuff.String(), err.Error()) // TODO log error
//return fmt.Errorf("appendkeyfile failed with '%s': %w", string(stdoutBuff.Bytes()), err)
return errors.New("adding to keys file failed") return errors.New("adding to keys file failed")
} }
err = code.MarkUsed(db) err = code.MarkUsed(db)
if err != nil { if err != nil {
l.Printf("failed to mark used: %s", err.Error()) // TODO log err
return errors.New("database was sad") return errors.New("database was sad")
} }
return nil return nil
} }
// TODO db plan:
// add new db, codes (modeled after invites)
// add new helper, emailtouser, that can access town.db and report on what user matches a given email
// drop table from town.db
// update sshapps.md
func main() { func main() {
cs := newColorScheme() cs := newColorScheme()
err := _main(cs) err := _main(cs)
@ -322,7 +326,7 @@ type AuthCode struct {
func (c *AuthCode) Insert(db *sql.DB) error { func (c *AuthCode) Insert(db *sql.DB) error {
stmt, err := db.Prepare(` stmt, err := db.Prepare(`
INSERT INTO auth_codes (code, email) INSERT INTO auth_codes (code, email)
VALUES (?, ?)`) VALUES ?, ?`)
if err != nil { if err != nil {
return err return err
} }

View File

@ -1,9 +1,7 @@
package main package main
import ( import (
"errors"
"fmt" "fmt"
"os"
"git.tilde.town/tildetown/town/email" "git.tilde.town/tildetown/town/email"
"git.tilde.town/tildetown/town/invites" "git.tilde.town/tildetown/town/invites"
@ -28,25 +26,6 @@ If you end up stuck, e-mail root@tilde.town with any questions.
See you on the server, See you on the server,
~vilmibm` ~vilmibm`
func loadPassword() (string, error) {
f, err := os.Open("/town/docs/smtp.pw")
if err != nil {
return "", fmt.Errorf("could not open smtp password file: %w", err)
}
pw := make([]byte, 100)
n, err := f.Read(pw)
if err != nil {
return "", fmt.Errorf("could not read smtp password file: %w", err)
}
if n == 0 {
return "", errors.New("smtp password file was empty")
}
return string(pw[0:n]), nil
}
func sendInviteEmail(invite invites.Invite) error { func sendInviteEmail(invite invites.Invite) error {
pw, err := email.LoadPassword() pw, err := email.LoadPassword()
if err != nil { if err != nil {

View File

@ -3,8 +3,10 @@ package email
import ( import (
"bytes" "bytes"
"crypto/tls" "crypto/tls"
"errors"
"fmt" "fmt"
"net/smtp" "net/smtp"
"os"
"os/exec" "os/exec"
) )
@ -14,6 +16,25 @@ const (
SMTPPort = 465 SMTPPort = 465
) )
func LoadPassword() (string, error) {
f, err := os.Open("/town/docs/smtp.pw")
if err != nil {
return "", fmt.Errorf("could not open smtp password file: %w", err)
}
pw := make([]byte, 100)
n, err := f.Read(pw)
if err != nil {
return "", fmt.Errorf("could not read smtp password file: %w", err)
}
if n == 0 {
return "", errors.New("smtp password file was empty")
}
return string(pw[0:n]), nil
}
func SendLocalEmail(username, subject, body string) error { func SendLocalEmail(username, subject, body string) error {
cmd := exec.Command("/usr/sbin/sendmail", username) cmd := exec.Command("/usr/sbin/sendmail", username)
cmd.Stdin = bytes.NewBufferString(fmt.Sprintf("Subject: %s\n\n%s", subject, body)) cmd.Stdin = bytes.NewBufferString(fmt.Sprintf("Subject: %s\n\n%s", subject, body))