package cgi

import (
	"bytes"
	"context"
	"fmt"
	"strings"

	sr "tildegit.org/tjp/sliderule"
	"tildegit.org/tjp/sliderule/logging"
	"tildegit.org/tjp/sliderule/spartan"
)

// SpartanCGIDirectory runs executable files relative to a root directory in the file system.
//
// It will also find any run any executable _part way_ through the path, so for example a
// request for /foo/bar/baz can also run an executable found at /foo or /foo/bar. In such
// a case the PATH_INFO environment variable will include the remaining portion of the URI.
func SpartanCGIDirectory(pathRoot, fsRoot string) sr.Handler {
	fsRoot = strings.TrimRight(fsRoot, "/")
	return sr.HandlerFunc(func(ctx context.Context, request *sr.Request) *sr.Response {
		if !strings.HasPrefix(request.Path, pathRoot) {
			return nil
		}

		filepath, pathinfo, err := ResolveCGI(request.Path[len(pathRoot):], fsRoot)
		if err != nil {
			return spartan.ServerError(err)
		}
		if filepath == "" {
			return nil
		}

		stderr := &bytes.Buffer{}
		stdout, exitCode, err := RunCGI(ctx, request, filepath, pathinfo, stderr)
		if err != nil {
			return spartan.ServerError(err)
		}
		if exitCode != 0 {
			ctx.Value("warnlog").(logging.Logger).Log(
				"msg", "cgi exited with non-zero exit code",
				"code", exitCode,
				"stderr", stderr.String(),
			)
			return spartan.ServerError(fmt.Errorf("CGI process exited with status %d", exitCode))
		}

		response, err := spartan.ParseResponse(stdout)
		if err != nil {
			return spartan.ServerError(err)
		}
		return response
	})
}