// Copyright (C) MongoDB, Inc. 2017-present.
//
// Licensed under the Apache License, Version 2.0 (the "License"); you may
// not use this file except in compliance with the License. You may obtain
// a copy of the License at http://www.apache.org/licenses/LICENSE-2.0

package bson

import (
	"fmt"
	"io"
	"reflect"
	"strings"
	"testing"

	"gitea.psichedelico.com/go/bson/internal/assert"
)

func TestExtJSONValueWriter(t *testing.T) {
	oid := ObjectID{0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0A, 0x0B, 0x0C}
	testCases := []struct {
		name   string
		fn     interface{}
		params []interface{}
	}{
		{
			"WriteBinary",
			(*extJSONValueWriter).WriteBinary,
			[]interface{}{[]byte{0x01, 0x02, 0x03}},
		},
		{
			"WriteBinaryWithSubtype (not 0x02)",
			(*extJSONValueWriter).WriteBinaryWithSubtype,
			[]interface{}{[]byte{0x01, 0x02, 0x03}, byte(0xFF)},
		},
		{
			"WriteBinaryWithSubtype (0x02)",
			(*extJSONValueWriter).WriteBinaryWithSubtype,
			[]interface{}{[]byte{0x01, 0x02, 0x03}, byte(0x02)},
		},
		{
			"WriteBoolean",
			(*extJSONValueWriter).WriteBoolean,
			[]interface{}{true},
		},
		{
			"WriteDBPointer",
			(*extJSONValueWriter).WriteDBPointer,
			[]interface{}{"bar", oid},
		},
		{
			"WriteDateTime",
			(*extJSONValueWriter).WriteDateTime,
			[]interface{}{int64(12345678)},
		},
		{
			"WriteDecimal128",
			(*extJSONValueWriter).WriteDecimal128,
			[]interface{}{NewDecimal128(10, 20)},
		},
		{
			"WriteDouble",
			(*extJSONValueWriter).WriteDouble,
			[]interface{}{float64(3.14159)},
		},
		{
			"WriteInt32",
			(*extJSONValueWriter).WriteInt32,
			[]interface{}{int32(123456)},
		},
		{
			"WriteInt64",
			(*extJSONValueWriter).WriteInt64,
			[]interface{}{int64(1234567890)},
		},
		{
			"WriteJavascript",
			(*extJSONValueWriter).WriteJavascript,
			[]interface{}{"var foo = 'bar';"},
		},
		{
			"WriteMaxKey",
			(*extJSONValueWriter).WriteMaxKey,
			[]interface{}{},
		},
		{
			"WriteMinKey",
			(*extJSONValueWriter).WriteMinKey,
			[]interface{}{},
		},
		{
			"WriteNull",
			(*extJSONValueWriter).WriteNull,
			[]interface{}{},
		},
		{
			"WriteObjectID",
			(*extJSONValueWriter).WriteObjectID,
			[]interface{}{oid},
		},
		{
			"WriteRegex",
			(*extJSONValueWriter).WriteRegex,
			[]interface{}{"bar", "baz"},
		},
		{
			"WriteString",
			(*extJSONValueWriter).WriteString,
			[]interface{}{"hello, world!"},
		},
		{
			"WriteSymbol",
			(*extJSONValueWriter).WriteSymbol,
			[]interface{}{"symbollolz"},
		},
		{
			"WriteTimestamp",
			(*extJSONValueWriter).WriteTimestamp,
			[]interface{}{uint32(10), uint32(20)},
		},
		{
			"WriteUndefined",
			(*extJSONValueWriter).WriteUndefined,
			[]interface{}{},
		},
	}

	for _, tc := range testCases {
		t.Run(tc.name, func(t *testing.T) {
			fn := reflect.ValueOf(tc.fn)
			if fn.Kind() != reflect.Func {
				t.Fatalf("fn must be of kind Func but it is a %v", fn.Kind())
			}
			if fn.Type().NumIn() != len(tc.params)+1 || fn.Type().In(0) != reflect.TypeOf((*extJSONValueWriter)(nil)) {
				t.Fatalf("fn must have at least one parameter and the first parameter must be a *valueWriter")
			}
			if fn.Type().NumOut() != 1 || fn.Type().Out(0) != reflect.TypeOf((*error)(nil)).Elem() {
				t.Fatalf("fn must have one return value and it must be an error.")
			}
			params := make([]reflect.Value, 1, len(tc.params)+1)
			ejvw := newExtJSONWriter(io.Discard, true, true, false)
			params[0] = reflect.ValueOf(ejvw)
			for _, param := range tc.params {
				params = append(params, reflect.ValueOf(param))
			}

			t.Run("incorrect transition", func(t *testing.T) {
				results := fn.Call(params)
				got := results[0].Interface().(error)
				fnName := tc.name
				if strings.Contains(fnName, "WriteBinary") {
					fnName = "WriteBinaryWithSubtype"
				}
				want := TransitionError{current: mTopLevel, name: fnName, modes: []mode{mElement, mValue},
					action: "write"}
				if !assert.CompareErrors(got, want) {
					t.Errorf("Errors do not match. got %v; want %v", got, want)
				}
			})
		})
	}

	t.Run("WriteArray", func(t *testing.T) {
		ejvw := newExtJSONWriter(io.Discard, true, true, false)
		ejvw.push(mArray)
		want := TransitionError{current: mArray, destination: mArray, parent: mTopLevel,
			name: "WriteArray", modes: []mode{mElement, mValue}, action: "write"}
		_, got := ejvw.WriteArray()
		if !assert.CompareErrors(got, want) {
			t.Errorf("Did not get expected error. got %v; want %v", got, want)
		}
	})
	t.Run("WriteCodeWithScope", func(t *testing.T) {
		ejvw := newExtJSONWriter(io.Discard, true, true, false)
		ejvw.push(mArray)
		want := TransitionError{current: mArray, destination: mCodeWithScope, parent: mTopLevel,
			name: "WriteCodeWithScope", modes: []mode{mElement, mValue}, action: "write"}
		_, got := ejvw.WriteCodeWithScope("")
		if !assert.CompareErrors(got, want) {
			t.Errorf("Did not get expected error. got %v; want %v", got, want)
		}
	})
	t.Run("WriteDocument", func(t *testing.T) {
		ejvw := newExtJSONWriter(io.Discard, true, true, false)
		ejvw.push(mArray)
		want := TransitionError{current: mArray, destination: mDocument, parent: mTopLevel,
			name: "WriteDocument", modes: []mode{mElement, mValue, mTopLevel}, action: "write"}
		_, got := ejvw.WriteDocument()
		if !assert.CompareErrors(got, want) {
			t.Errorf("Did not get expected error. got %v; want %v", got, want)
		}
	})
	t.Run("WriteDocumentElement", func(t *testing.T) {
		ejvw := newExtJSONWriter(io.Discard, true, true, false)
		ejvw.push(mElement)
		want := TransitionError{current: mElement,
			destination: mElement,
			parent:      mTopLevel,
			name:        "WriteDocumentElement",
			modes:       []mode{mDocument, mTopLevel, mCodeWithScope},
			action:      "write"}
		_, got := ejvw.WriteDocumentElement("")
		if !assert.CompareErrors(got, want) {
			t.Errorf("Did not get expected error. got %v; want %v", got, want)
		}
	})
	t.Run("WriteDocumentEnd", func(t *testing.T) {
		ejvw := newExtJSONWriter(io.Discard, true, true, false)
		ejvw.push(mElement)
		want := fmt.Errorf("incorrect mode to end document: %s", mElement)
		got := ejvw.WriteDocumentEnd()
		if !assert.CompareErrors(got, want) {
			t.Errorf("Did not get expected error. got %v; want %v", got, want)
		}
	})
	t.Run("WriteArrayElement", func(t *testing.T) {
		ejvw := newExtJSONWriter(io.Discard, true, true, false)
		ejvw.push(mElement)
		want := TransitionError{current: mElement,
			destination: mValue,
			parent:      mTopLevel,
			name:        "WriteArrayElement",
			modes:       []mode{mArray},
			action:      "write"}
		_, got := ejvw.WriteArrayElement()
		if !assert.CompareErrors(got, want) {
			t.Errorf("Did not get expected error. got %v; want %v", got, want)
		}
	})
	t.Run("WriteArrayEnd", func(t *testing.T) {
		ejvw := newExtJSONWriter(io.Discard, true, true, false)
		ejvw.push(mElement)
		want := fmt.Errorf("incorrect mode to end array: %s", mElement)
		got := ejvw.WriteArrayEnd()
		if !assert.CompareErrors(got, want) {
			t.Errorf("Did not get expected error. got %v; want %v", got, want)
		}
	})

	t.Run("WriteBytes", func(t *testing.T) {
		t.Run("writeElementHeader error", func(t *testing.T) {
			ejvw := newExtJSONWriterFromSlice(nil, true, true)
			want := TransitionError{current: mTopLevel, destination: mode(0),
				name: "WriteBinaryWithSubtype", modes: []mode{mElement, mValue}, action: "write"}
			got := ejvw.WriteBinaryWithSubtype(nil, (byte)(TypeEmbeddedDocument))
			if !assert.CompareErrors(got, want) {
				t.Errorf("Did not received expected error. got %v; want %v", got, want)
			}
		})
	})

	t.Run("FormatDoubleWithExponent", func(t *testing.T) {
		want := "3E-12"
		got := formatDouble(float64(0.000000000003))
		if got != want {
			t.Errorf("Did not receive expected string. got %s: want %s", got, want)
		}
	})
}