package accesslog

import (
	"fmt"
	"io"
	"os"
	"path/filepath"
	"sync"

	"github.com/yusing/go-proxy/internal/logging"
	"github.com/yusing/go-proxy/internal/utils"
)

type File struct {
	f *os.File

	// os.File.Name() may not equal to key of `openedFiles`.
	// Store it for later delete from `openedFiles`.
	path string

	refCount *utils.RefCount
}

var (
	openedFiles   = make(map[string]*File)
	openedFilesMu sync.Mutex
)

func newFileIO(path string) (SupportRotate, error) {
	openedFilesMu.Lock()
	defer openedFilesMu.Unlock()

	var file *File
	path = filepath.Clean(path)
	if opened, ok := openedFiles[path]; ok {
		opened.refCount.Add()
		return opened, nil
	} else {
		// cannot open as O_APPEND as we need Seek and WriteAt
		f, err := os.OpenFile(path, os.O_CREATE|os.O_RDWR, 0o644)
		if err != nil {
			return nil, fmt.Errorf("access log open error: %w", err)
		}
		if _, err := f.Seek(0, io.SeekEnd); err != nil {
			return nil, fmt.Errorf("access log seek error: %w", err)
		}
		file = &File{f: f, path: path, refCount: utils.NewRefCounter()}
		openedFiles[path] = file
		go file.closeOnZero()
	}

	return file, nil
}

func (f *File) Name() string {
	return f.f.Name()
}

func (f *File) Write(p []byte) (n int, err error) {
	return f.f.Write(p)
}

func (f *File) ReadAt(p []byte, off int64) (n int, err error) {
	return f.f.ReadAt(p, off)
}

func (f *File) WriteAt(p []byte, off int64) (n int, err error) {
	return f.f.WriteAt(p, off)
}

func (f *File) Seek(offset int64, whence int) (int64, error) {
	return f.f.Seek(offset, whence)
}

func (f *File) Size() (int64, error) {
	stat, err := f.f.Stat()
	if err != nil {
		return 0, err
	}
	return stat.Size(), nil
}

func (f *File) Truncate(size int64) error {
	return f.f.Truncate(size)
}

func (f *File) Close() error {
	f.refCount.Sub()
	return nil
}

func (f *File) closeOnZero() {
	defer logging.Debug().
		Str("path", f.path).
		Msg("access log closed")

	<-f.refCount.Zero()

	openedFilesMu.Lock()
	delete(openedFiles, f.path)
	openedFilesMu.Unlock()
	f.f.Close()
}