dnote/pkg/cli/cmd/login/login.go
2025-10-31 23:41:21 -07:00

184 lines
4.4 KiB
Go

/* Copyright 2025 Dnote Authors
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package login
import (
"fmt"
"net/url"
"strconv"
"github.com/dnote/dnote/pkg/cli/client"
"github.com/dnote/dnote/pkg/cli/consts"
"github.com/dnote/dnote/pkg/cli/context"
"github.com/dnote/dnote/pkg/cli/database"
"github.com/dnote/dnote/pkg/cli/infra"
"github.com/dnote/dnote/pkg/cli/log"
"github.com/dnote/dnote/pkg/cli/ui"
"github.com/pkg/errors"
"github.com/spf13/cobra"
)
var example = `
dnote login`
var usernameFlag, passwordFlag, apiEndpointFlag string
// NewCmd returns a new login command
func NewCmd(ctx context.DnoteCtx) *cobra.Command {
cmd := &cobra.Command{
Use: "login",
Short: "Login to dnote server",
Example: example,
RunE: newRun(ctx),
}
f := cmd.Flags()
f.StringVarP(&usernameFlag, "username", "u", "", "email address for authentication")
f.StringVarP(&passwordFlag, "password", "p", "", "password for authentication")
f.StringVar(&apiEndpointFlag, "apiEndpoint", "", "API endpoint to connect to (defaults to value in config)")
return cmd
}
// Do dervies credentials on the client side and requests a session token from the server
func Do(ctx context.DnoteCtx, email, password string) error {
signinResp, err := client.Signin(ctx, email, password)
if err != nil {
return errors.Wrap(err, "requesting session")
}
db := ctx.DB
tx, err := db.Begin()
if err != nil {
return errors.Wrap(err, "beginning a transaction")
}
if err := database.UpsertSystem(tx, consts.SystemSessionKey, signinResp.Key); err != nil {
return errors.Wrap(err, "saving session key")
}
if err := database.UpsertSystem(tx, consts.SystemSessionKeyExpiry, strconv.FormatInt(signinResp.ExpiresAt, 10)); err != nil {
return errors.Wrap(err, "saving session key")
}
tx.Commit()
return nil
}
func getUsername() (string, error) {
if usernameFlag != "" {
return usernameFlag, nil
}
var email string
if err := ui.PromptInput("email", &email); err != nil {
return "", errors.Wrap(err, "getting email input")
}
if email == "" {
return "", errors.New("Email is empty")
}
return email, nil
}
func getPassword() (string, error) {
if passwordFlag != "" {
return passwordFlag, nil
}
var password string
if err := ui.PromptPassword("password", &password); err != nil {
return "", errors.Wrap(err, "getting password input")
}
if password == "" {
return "", errors.New("Password is empty")
}
return password, nil
}
func getBaseURL(rawURL string) (string, error) {
u, err := url.Parse(rawURL)
if err != nil {
return "", errors.Wrap(err, "parsing url")
}
if u.Scheme == "" || u.Host == "" {
return "", nil
}
return fmt.Sprintf("%s://%s", u.Scheme, u.Host), nil
}
func getServerDisplayURL(ctx context.DnoteCtx) string {
baseURL, err := getBaseURL(ctx.APIEndpoint)
if err != nil {
return ""
}
return baseURL
}
func getGreeting(ctx context.DnoteCtx) string {
base := "Welcome to Dnote"
serverURL := getServerDisplayURL(ctx)
if serverURL == "" {
return fmt.Sprintf("%s\n", base)
}
return fmt.Sprintf("%s (%s)\n", base, serverURL)
}
func newRun(ctx context.DnoteCtx) infra.RunEFunc {
return func(cmd *cobra.Command, args []string) error {
// Override APIEndpoint if flag was provided
if apiEndpointFlag != "" {
ctx.APIEndpoint = apiEndpointFlag
}
greeting := getGreeting(ctx)
log.Plain(greeting)
email, err := getUsername()
if err != nil {
return errors.Wrap(err, "getting email input")
}
password, err := getPassword()
if err != nil {
return errors.Wrap(err, "getting password input")
}
if password == "" {
return errors.New("Password is empty")
}
log.Debug("Logging in with email: %s and password: (length %d)\n", email, len(password))
err = Do(ctx, email, password)
if errors.Cause(err) == client.ErrInvalidLogin {
log.Error("wrong login\n")
return nil
} else if err != nil {
return errors.Wrap(err, "logging in")
}
log.Success("logged in\n")
return nil
}
}