// Package fastcgi has middleware that acts as a FastCGI client. Requests
// that get forwarded to FastCGI stop the middleware execution chain.
// The most common use for this layer is to serve PHP websites via php-fpm.
package fastcgi

import (
	"io"
	"io/ioutil"
	"log"
	"net/http"
	"path/filepath"
	"strings"

	"github.com/mholt/caddy/middleware"
)

// New generates a new FastCGI middleware.
func New(c middleware.Controller) (middleware.Middleware, error) {
	root := c.Root()

	var rules []fastCgi
	for c.Next() {
		rule := fastCgi{}
		if !c.Args(&rule.path, &rule.address) {
			return nil, c.ArgErr()
		}
		rules = append(rules, rule)
	}

	return func(next http.HandlerFunc) http.HandlerFunc {
		return func(w http.ResponseWriter, r *http.Request) {

			servedFcgi := false
			for _, rule := range rules {
				if middleware.Path(r.URL.Path).Matches(rule.path) {
					servedFcgi = true

					// Get absolute file paths
					absPath, err := filepath.Abs(root + r.URL.Path)
					if err != nil {
						// TODO!
						log.Fatal(err)
					}

					// Get absolute file paths
					absRootPath, err := filepath.Abs(root)
					if err != nil {
						// TODO!
						log.Fatal(err)
					}

					// Separate remote IP and port
					var ip, port string
					if idx := strings.Index(r.RemoteAddr, ":"); idx > -1 {
						ip = r.RemoteAddr[idx:]
						port = r.RemoteAddr[:idx]
					} else {
						ip = r.RemoteAddr
					}

					// TODO: Do we really have to make this map from scratch for each request?
					// TODO: We have quite a few more to map, too.
					env := make(map[string]string)
					env["SERVER_SOFTWARE"] = "caddy" // TODO: Obtain version info...
					env["SERVER_PROTOCOL"] = r.Proto
					env["SCRIPT_FILENAME"] = absPath
					env["REMOTE_ADDR"] = ip
					env["REMOTE_PORT"] = port
					env["REQUEST_METHOD"] = r.Method
					env["QUERY_STRING"] = r.URL.RawQuery
					env["DOCUMENT_URI"] = r.URL.Path
					env["DOCUMENT_ROOT"] = absRootPath

					fcgi, err := Dial("tcp", rule.address)
					if err != nil {
						// TODO!
					}

					resp, err := fcgi.Get(env)
					if err != nil && err != io.EOF {
						// TODO!
					}

					body, err := ioutil.ReadAll(resp.Body)
					if err != nil {
						// TODO!
					}

					for key, vals := range resp.Header {
						for _, val := range vals {
							w.Header().Add(key, val)
						}
					}

					w.WriteHeader(resp.StatusCode)
					w.Write(body)

					break
				}
			}

			if !servedFcgi {
				next(w, r)
			}
		}
	}, nil
}

type fastCgi struct {
	path    string
	address string
}