gogcli/internal/cmd/gmail_attachment.go
salmonumbrella 07ffcb5d84
fix(paths): expand ~ in user-provided file paths (#56)
* fix(paths): expand ~ in user-provided file paths

When users specify paths with ~ (e.g., --out ~/Downloads/file.pdf) and
the path is quoted in the shell command, the tilde is not expanded by
the shell. This caused files to be written to a literal ~/Downloads
directory instead of the user's home directory.

Add config.ExpandPath() function that expands ~ at the beginning of
paths to the user's home directory. Apply this fix to all user-provided
file paths across:

- gmail attachment download (--out)
- drive download/export (--out)
- drive upload (localPath argument)
- auth token export (--out)
- auth credentials/import/keep (input paths)
- gmail thread attachments (--out-dir)
- gmail send/drafts (--attach)

Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>

* fix(lint): address wrapcheck and wsl issues

* fix(calendar): support ISO 8601 time format and add 'list' alias

- Add parsing for ISO 8601 datetime with numeric timezone without colon
  (e.g., 2026-01-09T16:38:41-0800), which is the format produced by
  macOS `date +%Y-%m-%dT%H:%M:%S%z`
- Add 'list' as an alias for 'events' subcommand for more intuitive CLI
  usage (gog calendar list instead of gog calendar events)

Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>

* chore(changelog): note PR #56

* chore(lint): dedupe file string

---------

Co-authored-by: Claude Opus 4.5 <noreply@anthropic.com>
Co-authored-by: Peter Steinberger <steipete@gmail.com>
2026-01-10 02:19:30 +00:00

137 lines
3.7 KiB
Go

package cmd
import (
"context"
"encoding/base64"
"errors"
"fmt"
"os"
"path/filepath"
"strings"
"google.golang.org/api/gmail/v1"
"github.com/steipete/gogcli/internal/config"
"github.com/steipete/gogcli/internal/outfmt"
"github.com/steipete/gogcli/internal/ui"
)
type GmailAttachmentCmd struct {
MessageID string `arg:"" name:"messageId" help:"Message ID"`
AttachmentID string `arg:"" name:"attachmentId" help:"Attachment ID"`
Output OutputPathFlag `embed:""`
Name string `name:"name" help:"Filename (only used when --out is empty)"`
}
func (c *GmailAttachmentCmd) Run(ctx context.Context, flags *RootFlags) error {
u := ui.FromContext(ctx)
account, err := requireAccount(flags)
if err != nil {
return err
}
messageID := strings.TrimSpace(c.MessageID)
attachmentID := strings.TrimSpace(c.AttachmentID)
if messageID == "" || attachmentID == "" {
return usage("messageId/attachmentId required")
}
svc, err := newGmailService(ctx, account)
if err != nil {
return err
}
if strings.TrimSpace(c.Output.Path) == "" {
dir, dirErr := config.EnsureGmailAttachmentsDir()
if dirErr != nil {
return dirErr
}
filename := strings.TrimSpace(c.Name)
if filename == "" {
filename = "attachment.bin"
}
safeFilename := filepath.Base(filename)
if safeFilename == "" || safeFilename == "." || safeFilename == ".." {
safeFilename = "attachment.bin"
}
shortID := attachmentID
if len(shortID) > 8 {
shortID = shortID[:8]
}
destPath := filepath.Join(dir, fmt.Sprintf("%s_%s_%s", messageID, shortID, safeFilename))
path, cached, bytes, dlErr := downloadAttachmentToPath(ctx, svc, messageID, attachmentID, destPath, -1)
if dlErr != nil {
return dlErr
}
if outfmt.IsJSON(ctx) {
return outfmt.WriteJSON(os.Stdout, map[string]any{"path": path, "cached": cached, "bytes": bytes})
}
u.Out().Printf("path\t%s", path)
u.Out().Printf("cached\t%t", cached)
u.Out().Printf("bytes\t%d", bytes)
return nil
}
outPath, err := config.ExpandPath(c.Output.Path)
if err != nil {
return err
}
path, cached, bytes, err := downloadAttachmentToPath(ctx, svc, messageID, attachmentID, outPath, -1)
if err != nil {
return err
}
if outfmt.IsJSON(ctx) {
return outfmt.WriteJSON(os.Stdout, map[string]any{"path": path, "cached": cached, "bytes": bytes})
}
u.Out().Printf("path\t%s", path)
u.Out().Printf("cached\t%t", cached)
u.Out().Printf("bytes\t%d", bytes)
return nil
}
func downloadAttachmentToPath(
ctx context.Context,
svc *gmail.Service,
messageID string,
attachmentID string,
outPath string,
expectedSize int64,
) (string, bool, int64, error) {
if strings.TrimSpace(outPath) == "" {
return "", false, 0, errors.New("missing outPath")
}
if expectedSize > 0 {
if st, err := os.Stat(outPath); err == nil && st.Size() == expectedSize {
return outPath, true, st.Size(), nil
}
} else if expectedSize == -1 {
if st, err := os.Stat(outPath); err == nil && st.Size() > 0 {
return outPath, true, st.Size(), nil
}
}
body, err := svc.Users.Messages.Attachments.Get("me", messageID, attachmentID).Context(ctx).Do()
if err != nil {
return "", false, 0, err
}
if body == nil || body.Data == "" {
return "", false, 0, errors.New("empty attachment data")
}
data, err := base64.RawURLEncoding.DecodeString(body.Data)
if err != nil {
// Gmail can return padded base64url; accept both.
data, err = base64.URLEncoding.DecodeString(body.Data)
if err != nil {
return "", false, 0, err
}
}
if err := os.MkdirAll(filepath.Dir(outPath), 0o700); err != nil {
return "", false, 0, err
}
if err := os.WriteFile(outPath, data, 0o600); err != nil {
return "", false, 0, err
}
return outPath, false, int64(len(data)), nil
}