/*
   Copyright The containerd Authors.

   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

   Unless required by applicable law or agreed to in writing, software
   distributed under the License is distributed on an "AS IS" BASIS,
   WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
   See the License for the specific language governing permissions and
   limitations under the License.
*/

package builder

import (
	"fmt"
	"path/filepath"
	"reflect"
	"runtime"
	"testing"

	specs "github.com/opencontainers/image-spec/specs-go/v1"
	"go.uber.org/mock/gomock"
	"gotest.tools/v3/assert"
)

type MockParse struct {
	ctrl     *gomock.Controller
	recorder *MockParseRecorder
}

type MockParseRecorder struct {
	mock *MockParse
}

func newMockParser(ctrl *gomock.Controller) *MockParse {
	mock := &MockParse{ctrl: ctrl}
	mock.recorder = &MockParseRecorder{mock}
	return mock
}

func (m *MockParse) EXPECT() *MockParseRecorder {
	return m.recorder
}

func (m *MockParse) Parse(platform string) (specs.Platform, error) {
	m.ctrl.T.Helper()
	ret := m.ctrl.Call(m, "Parse")
	ret0, _ := ret[0].(specs.Platform)
	ret1, _ := ret[1].(error)
	return ret0, ret1
}

func (m *MockParseRecorder) Parse(platform string) *gomock.Call {
	m.mock.ctrl.T.Helper()
	return m.mock.ctrl.RecordCallWithMethodType(m.mock, "Parse", reflect.TypeOf((*MockParse)(nil).Parse))
}

func (m *MockParse) DefaultSpec() specs.Platform {
	m.ctrl.T.Helper()
	ret := m.ctrl.Call(m, "DefaultSpec")
	ret0, _ := ret[0].(specs.Platform)
	return ret0
}

func (m *MockParseRecorder) DefaultSpec() *gomock.Call {
	m.mock.ctrl.T.Helper()
	return m.mock.ctrl.RecordCallWithMethodType(m.mock, "DefaultSpec", reflect.TypeOf((*MockParse)(nil).DefaultSpec))
}

func TestIsMatchingRuntimePlatform(t *testing.T) {
	t.Parallel()

	testCases := []struct {
		name string
		mock func(*MockParse)
		want bool
	}{
		{
			name: "Image is shareable when Runtime and build platform match for os, arch and variant",
			mock: func(mockParser *MockParse) {
				mockParser.EXPECT().Parse("test").Return(specs.Platform{OS: "mockOS", Architecture: "mockArch", Variant: "mockVariant"}, nil)
				mockParser.EXPECT().DefaultSpec().Return(specs.Platform{OS: "mockOS", Architecture: "mockArch", Variant: "mockVariant"})
			},
			want: true,
		},
		{
			name: "Image is shareable when Runtime and build platform match for os, arch. Variant is not defined",
			mock: func(mockParser *MockParse) {
				mockParser.EXPECT().Parse("test").Return(specs.Platform{OS: "mockOS", Architecture: "mockArch", Variant: ""}, nil)
				mockParser.EXPECT().DefaultSpec().Return(specs.Platform{OS: "mockOS", Architecture: "mockArch", Variant: "mockVariant"})
			},
			want: true,
		},
		{
			name: "Image is not shareable when Runtime and build platform donot math OS",
			mock: func(mockParser *MockParse) {
				mockParser.EXPECT().Parse("test").Return(specs.Platform{OS: "OS", Architecture: "mockArch", Variant: ""}, nil)
				mockParser.EXPECT().DefaultSpec().Return(specs.Platform{OS: "mockOS", Architecture: "mockArch", Variant: "mockVariant"})
			},
			want: false,
		},
		{
			name: "Image is not shareable when Runtime and build platform donot math Arch",
			mock: func(mockParser *MockParse) {
				mockParser.EXPECT().Parse("test").Return(specs.Platform{OS: "mockOS", Architecture: "Arch", Variant: ""}, nil)
				mockParser.EXPECT().DefaultSpec().Return(specs.Platform{OS: "mockOS", Architecture: "mockArch", Variant: "mockVariant"})
			},
			want: false,
		},
		{
			name: "Image is not shareable when Runtime and build platform donot math Variant",
			mock: func(mockParser *MockParse) {
				mockParser.EXPECT().Parse("test").Return(specs.Platform{OS: "mockOS", Architecture: "mockArch", Variant: "Variant"}, nil)
				mockParser.EXPECT().DefaultSpec().Return(specs.Platform{OS: "mockOS", Architecture: "mockArch", Variant: "mockVariant"})
			},
			want: false,
		},
	}

	for _, tc := range testCases {
		tc := tc
		t.Run(tc.name, func(t *testing.T) {
			t.Parallel()

			ctrl := gomock.NewController(t)
			mockParser := newMockParser(ctrl)
			tc.mock(mockParser)
			r := isMatchingRuntimePlatform("test", mockParser)
			assert.Equal(t, r, tc.want, tc.name)
		})
	}
}

func TestIsBuildPlatformDefault(t *testing.T) {
	t.Parallel()

	testCases := []struct {
		name     string
		mock     func(*MockParse)
		platform []string
		want     bool
	}{
		{
			name:     "Image is shreable when len of platform is 0",
			platform: make([]string, 0),
			want:     true,
		},
		{
			name:     "Image is shareable when Runtime and build platform match for os, arch and variant",
			platform: []string{"test"},
			mock: func(mockParser *MockParse) {
				mockParser.EXPECT().Parse("test").Return(specs.Platform{OS: "mockOS", Architecture: "mockArch", Variant: "mockVariant"}, nil)
				mockParser.EXPECT().DefaultSpec().Return(specs.Platform{OS: "mockOS", Architecture: "mockArch", Variant: "mockVariant"})
			},
			want: true,
		},
		{
			name:     "Image is not shareable when Runtime build platform dont match",
			platform: []string{"test"},
			mock: func(mockParser *MockParse) {
				mockParser.EXPECT().Parse("test").Return(specs.Platform{OS: "OS", Architecture: "mockArch", Variant: "mockVariant"}, nil)
				mockParser.EXPECT().DefaultSpec().Return(specs.Platform{OS: "mockOS", Architecture: "mockArch", Variant: "mockVariant"})
			},
			want: false,
		},
		{
			name:     "Image is not shareable when more than 2 platforms are passed",
			platform: []string{"test1", "test2"},
			want:     false,
		},
	}

	for _, tc := range testCases {
		tc := tc
		t.Run(tc.name, func(t *testing.T) {
			t.Parallel()

			ctrl := gomock.NewController(t)
			mockParser := newMockParser(ctrl)
			if len(tc.platform) == 1 {
				tc.mock(mockParser)
			}
			r := isBuildPlatformDefault(tc.platform, mockParser)
			assert.Equal(t, r, tc.want, tc.name)
		})
	}
}

func TestParseBuildctlArgsForOCILayout(t *testing.T) {
	tests := []struct {
		name          string
		ociLayoutName string
		ociLayoutPath string
		expectedArgs  []string
		errorIsNil    bool
		expectedErr   string
	}{
		{
			name:          "PrefixNotFoundError",
			ociLayoutName: "unit-test",
			ociLayoutPath: "/tmp/oci-layout/",
			expectedArgs:  []string{},
			expectedErr:   ErrOCILayoutPrefixNotFound.Error(),
		},
		{
			name:          "DirectoryNotFoundError",
			ociLayoutName: "unit-test",
			ociLayoutPath: "oci-layout:///tmp/oci-layout",
			expectedArgs:  []string{},
			expectedErr:   "open /tmp/oci-layout/index.json: no such file or directory",
		},
	}

	if runtime.GOOS == "windows" {
		abspath, err := filepath.Abs("/tmp/oci-layout")
		assert.NilError(t, err)
		tests[1].expectedErr = fmt.Sprintf(
			"open %s\\index.json: The system cannot find the path specified.",
			abspath,
		)
	}

	for _, test := range tests {
		t.Run(test.name, func(t *testing.T) {
			args, err := parseBuildContextFromOCILayout(test.ociLayoutName, test.ociLayoutPath)
			if test.errorIsNil {
				assert.NilError(t, err)
			} else {
				assert.Error(t, err, test.expectedErr)
			}
			assert.Equal(t, len(args), len(test.expectedArgs))
			assert.DeepEqual(t, args, test.expectedArgs)
		})
	}
}

func TestGetEffectiveSourcePolicyFile(t *testing.T) {
	// Cannot use t.Parallel() since subtests modify environment variables

	tests := []struct {
		name        string
		optionValue string
		envValue    string
		expected    string
	}{
		{
			name:        "option value takes precedence over env var",
			optionValue: "/path/from/flag.json",
			envValue:    "/path/from/env.json",
			expected:    "/path/from/flag.json",
		},
		{
			name:        "env var is used when option is empty",
			optionValue: "",
			envValue:    "/path/from/env.json",
			expected:    "/path/from/env.json",
		},
		{
			name:        "empty when both are unset",
			optionValue: "",
			envValue:    "",
			expected:    "",
		},
		{
			name:        "option value used when env var is empty",
			optionValue: "/path/from/flag.json",
			envValue:    "",
			expected:    "/path/from/flag.json",
		},
	}

	for _, tc := range tests {
		t.Run(tc.name, func(t *testing.T) {
			// Set up the environment variable for this test
			t.Setenv("EXPERIMENTAL_BUILDKIT_SOURCE_POLICY", tc.envValue)

			result := GetEffectiveSourcePolicyFile(tc.optionValue)
			assert.Equal(t, result, tc.expected)
		})
	}
}
