package wasm
import (
	"context"
	"errors"
	"flag"
	"fmt"
	"io/ioutil"
	"log"
	"net/http"
	"os"
	"os/exec"
	"regexp"
	"strings"
	"testing"
	"time"
	"github.com/chromedp/cdproto/cdp"
	"github.com/chromedp/chromedp"
)
var addr = flag.String("addr", ":8826", "Host:port to listen on for wasm test server")
var wasmTmpDir string // set in TestMain to a temp directory for build output
func TestMain(m *testing.M) {
	flag.Parse()
	os.Exit(func() int {
		var err error
		wasmTmpDir, err = ioutil.TempDir("", "wasm_test")
		if err != nil {
			log.Fatalf("unable to create temp dir: %v", err)
		}
		defer os.RemoveAll(wasmTmpDir) // cleanup even on panic and before os.Exit
		startServer(wasmTmpDir)
		return m.Run()
	}())
}
func run(cmdline string) error {
	args := strings.Fields(cmdline)
	return runargs(args...)
}
func runargs(args ...string) error {
	cmd := exec.Command(args[0], args[1:]...)
	b, err := cmd.CombinedOutput()
	log.Printf("Command: %s; err=%v; full output:\n%s", strings.Join(args, " "), err, b)
	if err != nil {
		return err
	}
	return nil
}
func chromectx(timeout time.Duration) (context.Context, context.CancelFunc) {
	var ctx context.Context
	// looks for locally installed Chrome
	ctx, _ = chromedp.NewContext(context.Background())
	ctx, cancel := context.WithTimeout(ctx, timeout)
	return ctx, cancel
}
func startServer(tmpDir string) {
	fsh := http.FileServer(http.Dir(tmpDir))
	h := http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
		if r.URL.Path == "/wasm_exec.js" {
			http.ServeFile(w, r, "../../targets/wasm_exec.js")
			return
		}
		if r.URL.Path == "/run" {
			fmt.Fprintf(w, `
Test
`, r.FormValue("file"))
			return
		}
		fsh.ServeHTTP(w, r)
	})
	log.Printf("Starting server at %q for dir: %s", *addr, tmpDir)
	go func() {
		log.Fatal(http.ListenAndServe(*addr, h))
	}()
}
// waitLog blocks until the log output equals the text provided (ignoring whitespace before and after)
func waitLog(logText string) chromedp.QueryAction {
	return waitInnerTextTrimEq("#log", strings.TrimSpace(logText))
}
// waitLogRe blocks until the log output matches this regular expression
func waitLogRe(restr string) chromedp.QueryAction {
	return waitInnerTextMatch("#log", regexp.MustCompile(restr))
}
// waitInnerTextTrimEq will wait for the innerText of the specified element to match a specific text pattern (ignoring whitespace before and after)
func waitInnerTextTrimEq(sel string, innerText string) chromedp.QueryAction {
	return waitInnerTextMatch(sel, regexp.MustCompile(`^\s*`+regexp.QuoteMeta(innerText)+`\s*$`))
}
// waitInnerTextMatch will wait for the innerText of the specified element to match a specific regexp pattern
func waitInnerTextMatch(sel string, re *regexp.Regexp) chromedp.QueryAction {
	return chromedp.Query(sel, func(s *chromedp.Selector) {
		chromedp.WaitFunc(func(ctx context.Context, cur *cdp.Frame, ids ...cdp.NodeID) ([]*cdp.Node, error) {
			nodes := make([]*cdp.Node, len(ids))
			cur.RLock()
			for i, id := range ids {
				nodes[i] = cur.Nodes[id]
				if nodes[i] == nil {
					cur.RUnlock()
					// not yet ready
					return nil, nil
				}
			}
			cur.RUnlock()
			var ret string
			err := chromedp.EvaluateAsDevTools("document.querySelector('"+sel+"').innerText", &ret).Do(ctx)
			if err != nil {
				return nodes, err
			}
			if !re.MatchString(ret) {
				// log.Printf("found text: %s", ret)
				return nodes, errors.New("unexpected value: " + ret)
			}
			// log.Printf("NodeValue: %#v", nodes[0])
			// return nil, errors.New("not ready yet")
			return nodes, nil
		})(s)
	})
}