package fuse

import (
	"errors"
	"io"
	"math"
	"os"
	"sync"

	"git.kmsign.ru/royalcat/tstor/src/fs"
	"github.com/billziss-gh/cgofuse/fuse"

	"github.com/rs/zerolog"
	"github.com/rs/zerolog/log"
)

type FS struct {
	fuse.FileSystemBase
	fh *fileHandler

	log zerolog.Logger
}

func NewFS(fs fs.Filesystem) fuse.FileSystemInterface {
	l := log.Logger.With().Str("component", "fuse").Logger()
	return &FS{
		fh:  &fileHandler{fs: fs},
		log: l,
	}
}

func (fs *FS) Open(path string, flags int) (errc int, fh uint64) {
	fh, err := fs.fh.OpenHolder(path)
	if os.IsNotExist(err) {
		fs.log.Debug().Str("path", path).Msg("file does not exists")
		return -fuse.ENOENT, fhNone

	}
	if err != nil {
		fs.log.Error().Err(err).Str("path", path).Msg("error opening file")
		return -fuse.EIO, fhNone
	}

	return 0, fh
}

// Unlink removes a file.
// The FileSystemBase implementation returns -ENOSYS.
func (fs *FS) Unlink(path string) int {
	return -fuse.ENOSYS
}

func (fs *FS) Opendir(path string) (errc int, fh uint64) {
	return fs.Open(path, 0)
}

func (fs *FS) Getattr(path string, stat *fuse.Stat_t, fh uint64) (errc int) {
	if path == "/" {
		stat.Mode = fuse.S_IFDIR | 0555
		return 0
	}

	file, err := fs.fh.GetFile(path, fh)
	if os.IsNotExist(err) {
		fs.log.Debug().Str("path", path).Msg("file does not exists")
		return -fuse.ENOENT

	}
	if err != nil {
		fs.log.Error().Err(err).Str("path", path).Msg("error getting holder when reading file attributes")
		return -fuse.EIO
	}

	if file.IsDir() {
		stat.Mode = fuse.S_IFDIR | 0555
	} else {
		stat.Mode = fuse.S_IFREG | 0444
		stat.Size = file.Size()
	}

	return 0
}

func (fs *FS) Read(path string, dest []byte, off int64, fh uint64) int {
	file, err := fs.fh.GetFile(path, fh)
	if os.IsNotExist(err) {
		fs.log.Error().Err(err).Str("path", path).Msg("file not found on READ operation")
		return -fuse.ENOENT

	}
	if err != nil {
		fs.log.Error().Err(err).Str("path", path).Msg("error getting holder reading data from file")
		return -fuse.EIO
	}

	end := int(math.Min(float64(len(dest)), float64(int64(file.Size())-off)))
	if end < 0 {
		end = 0
	}

	buf := dest[:end]

	n, err := file.ReadAt(buf, off)
	if err != nil && err != io.EOF {
		log.Error().Err(err).Str("path", path).Msg("error reading data")
		return -fuse.EIO
	}

	dest = buf[:n]
	return n
}

func (fs *FS) Release(path string, fh uint64) int {
	if err := fs.fh.Remove(fh); err != nil {
		fs.log.Error().Err(err).Str("path", path).Msg("error getting holder when releasing file")
		return -fuse.EIO
	}

	return 0
}

func (fs *FS) Releasedir(path string, fh uint64) int {
	return fs.Release(path, fh)
}

func (fs *FS) Readdir(path string,
	fill func(name string, stat *fuse.Stat_t, ofst int64) bool,
	ofst int64,
	fh uint64) (errc int) {
	fill(".", nil, 0)
	fill("..", nil, 0)

	//TODO improve this function to make use of fh index if possible
	paths, err := fs.fh.ListDir(path)
	if err != nil {
		fs.log.Error().Err(err).Str("path", path).Msg("error reading directory")
		return -fuse.ENOSYS
	}

	for _, p := range paths {
		if !fill(p, nil, 0) {
			fs.log.Error().Str("path", path).Msg("error adding directory")
			break
		}
	}

	return 0
}

const fhNone = ^uint64(0)

var ErrHolderEmpty = errors.New("file holder is empty")
var ErrBadHolderIndex = errors.New("holder index too big")

type fileHandler struct {
	mu     sync.RWMutex
	opened []fs.File
	fs     fs.Filesystem
}

func (fh *fileHandler) GetFile(path string, fhi uint64) (fs.File, error) {
	fh.mu.RLock()
	defer fh.mu.RUnlock()

	if fhi == fhNone {
		return fh.lookupFile(path)
	}
	return fh.get(fhi)
}

func (fh *fileHandler) ListDir(path string) ([]string, error) {
	fh.mu.RLock()
	defer fh.mu.RUnlock()

	var out []string
	files, err := fh.fs.ReadDir(path)
	if err != nil {
		return nil, err
	}
	for p := range files {
		out = append(out, p)
	}

	return out, nil
}

func (fh *fileHandler) OpenHolder(path string) (uint64, error) {
	file, err := fh.lookupFile(path)
	if err != nil {
		return fhNone, err
	}

	fh.mu.Lock()
	defer fh.mu.Unlock()

	for i, old := range fh.opened {
		if old == nil {
			fh.opened[i] = file
			return uint64(i), nil
		}
	}
	fh.opened = append(fh.opened, file)

	return uint64(len(fh.opened) - 1), nil
}

func (fh *fileHandler) get(fhi uint64) (fs.File, error) {
	if int(fhi) >= len(fh.opened) {
		return nil, ErrBadHolderIndex
	}
	// TODO check opened slice to avoid panics
	h := fh.opened[int(fhi)]
	if h == nil {
		return nil, ErrHolderEmpty
	}

	return h, nil
}

func (fh *fileHandler) Remove(fhi uint64) error {
	fh.mu.Lock()
	defer fh.mu.Unlock()

	if fhi == fhNone {
		return nil
	}

	// TODO check opened slice to avoid panics
	f := fh.opened[int(fhi)]
	if f == nil {
		return ErrHolderEmpty
	}

	if err := f.Close(); err != nil {
		return err
	}

	fh.opened[int(fhi)] = nil

	return nil
}

func (fh *fileHandler) lookupFile(path string) (fs.File, error) {
	file, err := fh.fs.Open(path)
	if err != nil {
		return nil, err
	}

	if file != nil {
		return file, nil
	}

	return nil, os.ErrNotExist
}