package ozon
import (
	"context"
	"fmt"
	"net/http"
	"testing"
	core "github.com/diphantxm/ozon-api-client"
)
func TestGetStocksInfo(t *testing.T) {
	t.Parallel()
	tests := []struct {
		statusCode int
		headers    map[string]string
		params     *GetStocksInfoParams
		response   string
	}{
		// Test Ok
		{
			http.StatusOK,
			map[string]string{"Client-Id": "my-client-id", "Api-Key": "my-api-key"},
			&GetStocksInfoParams{
				Limit:  100,
				LastId: "",
				Filter: GetStocksInfoFilter{
					OfferId:    "136834",
					ProductId:  214887921,
					Visibility: "ALL",
				},
			},
			`{
				"result": {
				  "items": [
					{
					  "product_id": 214887921,
					  "offer_id": "136834",
					  "stocks": [
						{
						  "type": "fbs",
						  "present": 170,
						  "reserved": 0
						},
						{
						  "type": "fbo",
						  "present": 0,
						  "reserved": 0
						}
					  ]
					}
				  ],
				  "total": 1,
				  "last_id": "anVsbA=="
				}
			}`,
		},
		// Test No Client-Id or Api-Key
		{
			http.StatusUnauthorized,
			map[string]string{},
			&GetStocksInfoParams{},
			`{
				"code": 16,
				"message": "Client-Id and Api-Key headers are required"
			}`,
		},
	}
	for _, test := range tests {
		c := NewMockClient(core.NewMockHttpHandler(test.statusCode, test.response, test.headers))
		ctx, _ := context.WithTimeout(context.Background(), testTimeout)
		resp, err := c.Products().GetStocksInfo(ctx, test.params)
		if err != nil {
			t.Error(err)
		}
		if resp.StatusCode != test.statusCode {
			t.Errorf("got wrong status code: got: %d, expected: %d", resp.StatusCode, test.statusCode)
		}
		if resp.StatusCode == http.StatusOK {
			if len(resp.Result.Items) > int(test.params.Limit) {
				t.Errorf("Amount of items in response cannot be bigger than limit")
			}
			if len(resp.Result.Items) > 0 {
				if resp.Result.Items[0].ProductId == 0 {
					t.Errorf("Product id cannot be 0")
				}
				if resp.Result.Items[0].OfferId == "" {
					t.Errorf("Offer id cannot be empty")
				}
			}
		}
	}
}
func TestGetProductDetails(t *testing.T) {
	t.Parallel()
	tests := []struct {
		statusCode int
		headers    map[string]string
		params     *GetProductDetailsParams
		response   string
	}{
		// Test Ok
		{
			http.StatusOK,
			map[string]string{"Client-Id": "my-client-id", "Api-Key": "my-api-key"},
			&GetProductDetailsParams{
				ProductId: 137208233,
			},
			`{
				"result": {
				  "id": 137208233,
				  "name": "Комплект защитных плёнок для X3 NFC. Темный хлопок",
				  "offer_id": "143210586",
				  "barcode": "",
				  "barcodes": [
					"2335900005",
					"7533900005"
				  ],
				  "buybox_price": "",
				  "description_category_id": 17038062,
				  "created_at": "2021-10-21T15:48:03.529178Z",
				  "images": [
					"https://cdn1.ozone.ru/s3/multimedia-5/6088931525.jpg",
					"https://cdn1.ozone.ru/s3/multimedia-p/6088915813.jpg"
				  ],
				  "has_discounted_item": true,
				  "is_discounted": true,
				  "discounted_stocks": {
					"coming": 0,
					"present": 0,
					"reserved": 0
				  },
				  "currency_code": "RUB",
				  "marketing_price": "",
				  "min_price": "",
				  "old_price": "",
				  "premium_price": "",
				  "price": "590.0000",
				  "recommended_price": "",
				  "sources": [
					{
					  "is_enabled": true,
					  "sku": 522759607,
					  "source": "fbo"
					},
					{
					  "is_enabled": true,
					  "sku": 522759608,
					  "source": "fbs"
					}
				  ],
				  "stocks": {
					"coming": 0,
					"present": 0,
					"reserved": 0
				  },
				  "errors": [],
				  "updated_at": "2023-02-09T06:46:44.152Z",
				  "vat": "0.0",
				  "visible": false,
				  "visibility_details": {
					"has_price": true,
					"has_stock": false,
					"active_product": false
				  },
				  "price_indexes": {
					"external_index_data": {
					  "minimal_price": "string",
					  "minimal_price_currency": "string",
					  "price_index_value": 0
					},
					"ozon_index_data": {
					  "minimal_price": "string",
					  "minimal_price_currency": "string",
					  "price_index_value": 0
					},
					"price_index": "WITHOUT_INDEX",
					"self_marketplaces_index_data": {
					  "minimal_price": "string",
					  "minimal_price_currency": "string",
					  "price_index_value": 0
					}
				  },
				  "commissions": [],
				  "volume_weight": 0.1,
				  "is_prepayment": false,
				  "is_prepayment_allowed": true,
				  "images360": [],
				  "is_kgt": false,
				  "color_image": "",
				  "primary_image": "https://cdn1.ozone.ru/s3/multimedia-p/6088931545.jpg",
				  "status": {
					"state": "imported",
					"state_failed": "imported",
					"moderate_status": "",
					"decline_reasons": [],
					"validation_state": "pending",
					"state_name": "Не продается",
					"state_description": "Не создан",
					"is_failed": true,
					"is_created": false,
					"state_tooltip": "",
					"item_errors": [
					  {
						"code": "error_attribute_values_empty",
						"field": "attribute",
						"attribute_id": 9048,
						"state": "imported",
						"level": "error",
						"description": "Не заполнен обязательный атрибут. Иногда мы обновляем обязательные атрибуты или добавляем новые. Отредактируйте товар или загрузите новый XLS-шаблон с актуальными атрибутами. ",
						"optional_description_elements": {},
						"attribute_name": "Название модели"
					  },
					  {
						"code": "error_attribute_values_empty",
						"field": "attribute",
						"attribute_id": 5076,
						"state": "imported",
						"level": "error",
						"description": "Не заполнен обязательный атрибут. Иногда мы обновляем обязательные атрибуты или добавляем новые. Отредактируйте товар или загрузите новый XLS-шаблон с актуальными атрибутами. ",
						"optional_description_elements": {},
						"attribute_name": "Рекомендовано для"
					  },
					  {
						"code": "error_attribute_values_empty",
						"field": "attribute",
						"attribute_id": 8229,
						"state": "imported",
						"level": "error",
						"description": "Не заполнен обязательный атрибут. Иногда мы обновляем обязательные атрибуты или добавляем новые. Отредактируйте товар или загрузите новый XLS-шаблон с актуальными атрибутами. ",
						"optional_description_elements": {},
						"attribute_name": "Тип"
					  },
					  {
						"code": "error_attribute_values_empty",
						"field": "attribute",
						"attribute_id": 85,
						"state": "imported",
						"level": "error",
						"description": "Не заполнен обязательный атрибут. Иногда мы обновляем обязательные атрибуты или добавляем новые. Отредактируйте товар или загрузите новый XLS-шаблон с актуальными атрибутами. ",
						"optional_description_elements": {},
						"attribute_name": "Бренд"
					  }
					],
					"state_updated_at": "2021-10-21T15:48:03.927309Z"
				  }
				}
			}`,
		},
		// Test No Client-Id or Api-Key
		{
			http.StatusUnauthorized,
			map[string]string{},
			&GetProductDetailsParams{},
			`{
				"code": 16,
				"message": "Client-Id and Api-Key headers are required"
			}`,
		},
	}
	for _, test := range tests {
		c := NewMockClient(core.NewMockHttpHandler(test.statusCode, test.response, test.headers))
		ctx, _ := context.WithTimeout(context.Background(), testTimeout)
		resp, err := c.Products().GetProductDetails(ctx, test.params)
		if err != nil {
			t.Error(err)
		}
		if resp.StatusCode != test.statusCode {
			t.Errorf("got wrong status code: got: %d, expected: %d", resp.StatusCode, test.statusCode)
		}
		if resp.StatusCode == http.StatusOK {
			if resp.Result.Id != test.params.ProductId {
				t.Errorf("Id of product in response is not equal product_id in request")
			}
			if resp.Result.OfferId == "" {
				t.Errorf("Offer id cannot be empty")
			}
			if resp.Result.DescriptionCategoryId == 0 {
				t.Errorf("Category id cannot be 0")
			}
			if resp.Result.CurrencyCode == "" {
				t.Errorf("Currency code cannot be empty")
			}
		}
	}
}
func TestUpdateStocks(t *testing.T) {
	t.Parallel()
	tests := []struct {
		statusCode int
		headers    map[string]string
		params     *UpdateStocksParams
		response   string
	}{
		// Test Ok
		{
			http.StatusOK,
			map[string]string{"Client-Id": "my-client-id", "Api-Key": "my-api-key"},
			&UpdateStocksParams{
				Stocks: []UpdateStocksStock{
					{
						OfferId:   "PG-2404С1",
						ProductId: 55946,
						Stock:     4,
					},
				},
			},
			`{
				"result": [
				  {
					"product_id": 55946,
					"offer_id": "PG-2404С1",
					"updated": true,
					"errors": []
				  }
				]
			}`,
		},
		// Test No Client-Id or Api-Key
		{
			http.StatusUnauthorized,
			map[string]string{},
			&UpdateStocksParams{},
			`{
				"code": 16,
				"message": "Client-Id and Api-Key headers are required"
			}`,
		},
	}
	for _, test := range tests {
		c := NewMockClient(core.NewMockHttpHandler(test.statusCode, test.response, test.headers))
		ctx, _ := context.WithTimeout(context.Background(), testTimeout)
		resp, err := c.Products().UpdateStocks(ctx, test.params)
		if err != nil {
			t.Error(err)
		}
		if resp.StatusCode != test.statusCode {
			t.Errorf("got wrong status code: got: %d, expected: %d", resp.StatusCode, test.statusCode)
		}
		if resp.StatusCode == http.StatusOK {
			if len(resp.Result) != len(test.params.Stocks) {
				t.Errorf("Length of stocks in request and response are not equal")
			}
			if len(resp.Result) > 0 {
				if resp.Result[0].OfferId != test.params.Stocks[0].OfferId {
					t.Errorf("Offer ids in request and response are not equal")
				}
				if resp.Result[0].ProductId != test.params.Stocks[0].ProductId {
					t.Errorf("Product ids in request and response are not equal")
				}
			}
		}
	}
}
func TestStocksInSellersWarehouse(t *testing.T) {
	t.Parallel()
	tests := []struct {
		statusCode int
		headers    map[string]string
		params     *StocksInSellersWarehouseParams
		response   string
	}{
		// Test Ok
		{
			http.StatusOK,
			map[string]string{"Client-Id": "my-client-id", "Api-Key": "my-api-key"},
			&StocksInSellersWarehouseParams{
				SKU: []string{"123"},
			},
			`{
				"result": [
				  {
					"sku": 12,
					"present": 34,
					"product_id": 548761,
					"reserved": 5,
					"warehouse_id": 156778,
					"warehouse_name": "something"
				  }
				]
			}`,
		},
		// Test No Client-Id or Api-Key
		{
			http.StatusUnauthorized,
			map[string]string{},
			&StocksInSellersWarehouseParams{},
			`{
				"code": 16,
				"message": "Client-Id and Api-Key headers are required"
			}`,
		},
	}
	for _, test := range tests {
		c := NewMockClient(core.NewMockHttpHandler(test.statusCode, test.response, test.headers))
		ctx, _ := context.WithTimeout(context.Background(), testTimeout)
		resp, err := c.Products().StocksInSellersWarehouse(ctx, test.params)
		if err != nil {
			t.Error(err)
		}
		if resp.StatusCode != test.statusCode {
			t.Errorf("got wrong status code: got: %d, expected: %d", resp.StatusCode, test.statusCode)
		}
		if resp.StatusCode == http.StatusOK {
			if len(resp.Result) != len(test.params.SKU) {
				t.Errorf("Length of skus in request and response must be equal")
			}
			if len(resp.Result) > 0 {
				if fmt.Sprint(resp.Result[0].SKU) == test.params.SKU[0] {
					t.Errorf("fbs sku in request and response are not equal")
				}
			}
		}
	}
}
func TestUpdatePrices(t *testing.T) {
	t.Parallel()
	tests := []struct {
		statusCode int
		headers    map[string]string
		params     *UpdatePricesParams
		response   string
	}{
		// Test Ok
		{
			http.StatusOK,
			map[string]string{"Client-Id": "my-client-id", "Api-Key": "my-api-key"},
			&UpdatePricesParams{
				Prices: []UpdatePricesPrice{
					{
						AutoActionEnabled:    "UNKNOWN",
						CurrencyCode:         "RUB",
						MinPrice:             "800",
						OldPrice:             "0",
						Price:                "1448",
						ProductId:            1386,
						PriceStrategyEnabled: PriceStrategyUnknown,
					},
				},
			},
			`{
				"result": [
				  {
					"product_id": 1386,
					"offer_id": "PH8865",
					"updated": true,
					"errors": []
				  }
				]
			}`,
		},
		// Test No Client-Id or Api-Key
		{
			http.StatusUnauthorized,
			map[string]string{},
			&UpdatePricesParams{},
			`{
				"code": 16,
				"message": "Client-Id and Api-Key headers are required"
			}`,
		},
	}
	for _, test := range tests {
		c := NewMockClient(core.NewMockHttpHandler(test.statusCode, test.response, test.headers))
		ctx, _ := context.WithTimeout(context.Background(), testTimeout)
		resp, err := c.Products().UpdatePrices(ctx, test.params)
		if err != nil {
			t.Error(err)
		}
		if resp.StatusCode != test.statusCode {
			t.Errorf("got wrong status code: got: %d, expected: %d", resp.StatusCode, test.statusCode)
		}
		if resp.StatusCode == http.StatusOK {
			if len(resp.Result) != len(test.params.Prices) {
				t.Errorf("Length of prices in request and response are not equal")
			}
			if len(resp.Result) > 0 {
				if resp.Result[0].ProductId != test.params.Prices[0].ProductId {
					t.Errorf("Product ids in request and response are not equal")
				}
			}
		}
	}
}
func TestUpdateQuantityStockProducts(t *testing.T) {
	t.Parallel()
	tests := []struct {
		statusCode int
		headers    map[string]string
		params     *UpdateQuantityStockProductsParams
		response   string
	}{
		// Test Ok
		{
			http.StatusOK,
			map[string]string{"Client-Id": "my-client-id", "Api-Key": "my-api-key"},
			&UpdateQuantityStockProductsParams{
				Stocks: []UpdateQuantityStockProductsStock{
					{
						OfferId:     "PH11042",
						ProductId:   118597312,
						Stock:       100,
						WarehouseId: 22142605386000,
					},
				},
			},
			`{
				"result": [
				  {
					"warehouse_id": 22142605386000,
					"product_id": 118597312,
					"offer_id": "PH11042",
					"updated": true,
					"errors": []
				  }
				]
			}`,
		},
		// Test No Client-Id or Api-Key
		{
			http.StatusUnauthorized,
			map[string]string{},
			&UpdateQuantityStockProductsParams{},
			`{
				"code": 16,
				"message": "Client-Id and Api-Key headers are required"
			}`,
		},
	}
	for _, test := range tests {
		c := NewMockClient(core.NewMockHttpHandler(test.statusCode, test.response, test.headers))
		ctx, _ := context.WithTimeout(context.Background(), testTimeout)
		resp, err := c.Products().UpdateQuantityStockProducts(ctx, test.params)
		if err != nil {
			t.Error(err)
		}
		if resp.StatusCode != test.statusCode {
			t.Errorf("got wrong status code: got: %d, expected: %d", resp.StatusCode, test.statusCode)
		}
		if resp.StatusCode == http.StatusOK {
			if len(resp.Result) != len(test.params.Stocks) {
				t.Errorf("Length of stocks in request and response are not equal")
			}
			if len(resp.Result) > 0 {
				if resp.Result[0].Offerid != test.params.Stocks[0].OfferId {
					t.Errorf("Offer ids in request and response are not equal")
				}
				if resp.Result[0].ProductId != test.params.Stocks[0].ProductId {
					t.Errorf("Product ids in request and response are not equal")
				}
				if resp.Result[0].WarehouseId != test.params.Stocks[0].WarehouseId {
					t.Errorf("Warehouse ids in request and response are not equal")
				}
			}
		}
	}
}
func TestCreateOrUpdateProduct(t *testing.T) {
	t.Parallel()
	tests := []struct {
		statusCode int
		headers    map[string]string
		params     *CreateOrUpdateProductParams
		response   string
	}{
		// Test Ok
		{
			http.StatusOK,
			map[string]string{"Client-Id": "my-client-id", "Api-Key": "my-api-key"},
			&CreateOrUpdateProductParams{
				Items: []CreateOrUpdateProductItem{
					{
						Attributes: []CreateOrUpdateAttribute{
							{
								ComplexId: 0,
								Id:        5076,
								Values: []CreateOrUpdateAttributeValue{
									{
										DictionaryValueId: 971082156,
										Value:             "Стойка для акустической системы",
									},
								},
							},
							{
								ComplexId: 0,
								Id:        9048,
								Values: []CreateOrUpdateAttributeValue{
									{
										Value: "Комплект защитных плёнок для X3 NFC. Темный хлопок",
									},
								},
							},
							{
								ComplexId: 0,
								Id:        8229,
								Values: []CreateOrUpdateAttributeValue{
									{
										DictionaryValueId: 95911,
										Value:             "Комплект защитных плёнок для X3 NFC. Темный хлопок",
									},
								},
							},
							{
								ComplexId: 0,
								Id:        85,
								Values: []CreateOrUpdateAttributeValue{
									{
										DictionaryValueId: 5060050,
										Value:             "Samsung",
									},
								},
							},
							{
								ComplexId: 0,
								Id:        10096,
								Values: []CreateOrUpdateAttributeValue{
									{
										DictionaryValueId: 61576,
										Value:             "серый",
									},
								},
							},
						},
						Barcode:               "112772873170",
						DescriptionCategoryId: 17033876,
						CurrencyCode:          "RUB",
						Depth:                 10,
						DimensionUnit:         "mm",
						Height:                250,
						Name:                  "Комплект защитных плёнок для X3 NFC. Темный хлопок",
						OfferId:               "143210608",
						OldPrice:              "1100",
						PremiumPrice:          "900",
						Price:                 "1000",
						VAT:                   "0.1",
						Weight:                100,
						WeightUnit:            "g",
						Width:                 150,
					},
				},
			},
			`{
				"result": {
				  "task_id": 172549793
				}
			}`,
		},
		// Test No Client-Id or Api-Key
		{
			http.StatusUnauthorized,
			map[string]string{},
			&CreateOrUpdateProductParams{},
			`{
				"code": 16,
				"message": "Client-Id and Api-Key headers are required"
			}`,
		},
	}
	for _, test := range tests {
		c := NewMockClient(core.NewMockHttpHandler(test.statusCode, test.response, test.headers))
		ctx, _ := context.WithTimeout(context.Background(), testTimeout)
		resp, err := c.Products().CreateOrUpdateProduct(ctx, test.params)
		if err != nil {
			t.Error(err)
		}
		if resp.StatusCode != test.statusCode {
			t.Errorf("got wrong status code: got: %d, expected: %d", resp.StatusCode, test.statusCode)
		}
		if resp.StatusCode == http.StatusOK {
			if resp.Result.TaskId == 0 {
				t.Errorf("Task id cannot be 0")
			}
		}
	}
}
func TestGetListOfProducts(t *testing.T) {
	t.Parallel()
	tests := []struct {
		statusCode int
		headers    map[string]string
		params     *GetListOfProductsParams
		response   string
	}{
		// Test Ok
		{
			http.StatusOK,
			map[string]string{"Client-Id": "my-client-id", "Api-Key": "my-api-key"},
			&GetListOfProductsParams{
				Filter: GetListOfProductsFilter{
					OfferId:    []string{"136748"},
					ProductId:  []int64{223681945},
					Visibility: "ALL",
				},
				LastId: "",
				Limit:  100,
			},
			`{
				"result": {
				  "items": [
					{
					  "product_id": 223681945,
					  "offer_id": "136748"
					}
				  ],
				  "total": 1,
				  "last_id": "bnVсbA=="
				}
			}`,
		},
		// Test No Client-Id or Api-Key
		{
			http.StatusUnauthorized,
			map[string]string{},
			&GetListOfProductsParams{},
			`{
				"code": 16,
				"message": "Client-Id and Api-Key headers are required"
			}`,
		},
	}
	for _, test := range tests {
		c := NewMockClient(core.NewMockHttpHandler(test.statusCode, test.response, test.headers))
		ctx, _ := context.WithTimeout(context.Background(), testTimeout)
		resp, err := c.Products().GetListOfProducts(ctx, test.params)
		if err != nil {
			t.Error(err)
		}
		if resp.StatusCode != test.statusCode {
			t.Errorf("got wrong status code: got: %d, expected: %d", resp.StatusCode, test.statusCode)
		}
		if resp.StatusCode == http.StatusOK {
			if len(resp.Result.Items) != int(resp.Result.Total) {
				t.Errorf("Length of items is not equal total")
			}
			if resp.Result.Total > int32(test.params.Limit) {
				t.Errorf("Length of items is bigger than limit")
			}
			if len(resp.Result.Items) > 0 {
				if resp.Result.Items[0].OfferId == "" {
					t.Errorf("Offer id cannot be empty")
				}
				if resp.Result.Items[0].ProductId == 0 {
					t.Errorf("Product id cannot be 0")
				}
			}
		}
	}
}
func TestGetProductsRatingBySKU(t *testing.T) {
	t.Parallel()
	tests := []struct {
		statusCode int
		headers    map[string]string
		params     *GetProductsRatingBySKUParams
		response   string
	}{
		// Test Ok
		{
			http.StatusOK,
			map[string]string{"Client-Id": "my-client-id", "Api-Key": "my-api-key"},
			&GetProductsRatingBySKUParams{
				SKUs: []int64{179737222},
			},
			`{
				"products": [
				  {
					"sku": 179737222,
					"rating": 42.5,
					"groups": [
					  {
						"key": "media",
						"name": "Медиа",
						"rating": 70,
						"weight": 25,
						"conditions": [
						  {
							"key": "media_images_2",
							"description": "Добавлено 2 изображения",
							"fulfilled": true,
							"cost": 50
						  },
						  {
							"key": "media_images_3",
							"description": "Добавлено 3 изображения и более",
							"fulfilled": true,
							"cost": 20
						  },
						  {
							"key": "media_image_3d",
							"description": "Добавлено 3D-изображение",
							"fulfilled": false,
							"cost": 15
						  },
						  {
							"key": "media_video",
							"description": "Добавлено видео",
							"fulfilled": false,
							"cost": 15
						  }
						],
						"improve_attributes": [
						  {
							"id": 4074,
							"name": "Код ролика на YouTube"
						  },
						  {
							"id": 4080,
							"name": "3D-изображение"
						  }
						],
						"improve_at_least": 2
					  },
					  {
						"key": "important_attributes",
						"name": "Важные атрибуты",
						"rating": 50,
						"weight": 30,
						"conditions": [
						  {
							"key": "important_2",
							"description": "Заполнено 2 атрибута и более",
							"fulfilled": true,
							"cost": 50
						  },
						  {
							"key": "important_50_percent",
							"description": "Заполнено более 50% атрибутов",
							"fulfilled": false,
							"cost": 25
						  },
						  {
							"key": "important_70_percent",
							"description": "Заполнено более 70% атрибутов",
							"fulfilled": false,
							"cost": 25
						  }
						],
						"improve_attributes": [
						  {
							"id": 4385,
							"name": "Гарантийный срок"
						  },
						  {
							"id": 4389,
							"name": "Страна-изготовитель"
						  },
						  {
							"id": 8513,
							"name": "Количество в упаковке, шт"
						  },
						  {
							"id": 8590,
							"name": "Макс. диагональ, дюймы"
						  },
						  {
							"id": 8591,
							"name": "Мин. диагональ, дюймы"
						  },
						  {
							"id": 9336,
							"name": "Модель браслета/умных часов"
						  },
						  {
							"id": 11046,
							"name": "Покрытие"
						  },
						  {
							"id": 11047,
							"name": "Прозрачность покрытия"
						  },
						  {
							"id": 11048,
							"name": "Дополнительные свойства покрытия"
						  },
						  {
							"id": 11049,
							"name": "Вид стекла"
						  },
						  {
							"id": 11603,
							"name": "Размер циферблата"
						  }
						],
						"improve_at_least": 6
					  },
					  {
						"key": "other_attributes",
						"name": "Остальные атрибуты",
						"rating": 0,
						"weight": 25,
						"conditions": [
						  {
							"key": "other_2",
							"description": "Заполнено 2 атрибута и более",
							"fulfilled": false,
							"cost": 50
						  },
						  {
							"key": "other_50_percent",
							"description": "Заполнено более 50% атрибутов",
							"fulfilled": false,
							"cost": 50
						  }
						],
						"improve_attributes": [
						  {
							"id": 4382,
							"name": "Размеры, мм"
						  }
						],
						"improve_at_least": 1
					  },
					  {
						"key": "text",
						"name": "Текстовое описание",
						"rating": 50,
						"weight": 20,
						"conditions": [
						  {
							"key": "text_annotation_100_chars",
							"description": "Аннотация более 100 знаков",
							"fulfilled": true,
							"cost": 25
						  },
						  {
							"key": "text_annotation_500_chars",
							"description": "Аннотация более 500 знаков",
							"fulfilled": true,
							"cost": 25
						  },
						  {
							"key": "text_rich",
							"description": "Заполнен Rich-контент",
							"fulfilled": false,
							"cost": 100
						  }
						],
						"improve_attributes": [
						  {
							"id": 11254,
							"name": "Rich-контент JSON"
						  }
						],
						"improve_at_least": 1
					  }
					]
				  }
				]
			}`,
		},
		// Test No Client-Id or Api-Key
		{
			http.StatusUnauthorized,
			map[string]string{},
			&GetProductsRatingBySKUParams{},
			`{
				"code": 16,
				"message": "Client-Id and Api-Key headers are required"
			}`,
		},
	}
	for _, test := range tests {
		c := NewMockClient(core.NewMockHttpHandler(test.statusCode, test.response, test.headers))
		ctx, _ := context.WithTimeout(context.Background(), testTimeout)
		resp, err := c.Products().GetProductsRatingBySKU(ctx, test.params)
		if err != nil {
			t.Error(err)
		}
		if resp.StatusCode != test.statusCode {
			t.Errorf("got wrong status code: got: %d, expected: %d", resp.StatusCode, test.statusCode)
		}
		if resp.StatusCode == http.StatusOK {
			if len(resp.Products) != len(test.params.SKUs) {
				t.Errorf("Length of products in response is not equal length of skus in request")
			}
			if len(resp.Products) > 0 {
				if resp.Products[0].SKU != test.params.SKUs[0] {
					t.Errorf("SKU in request and response are not equal")
				}
			}
		}
	}
}
func TestGetProductImportStatus(t *testing.T) {
	t.Parallel()
	tests := []struct {
		statusCode int
		headers    map[string]string
		params     *GetProductImportStatusParams
		response   string
	}{
		// Test Ok
		{
			http.StatusOK,
			map[string]string{"Client-Id": "my-client-id", "Api-Key": "my-api-key"},
			&GetProductImportStatusParams{
				TaskId: 172549793,
			},
			`{
				"result": {
				  "items": [
					{
					  "offer_id": "143210608",
					  "product_id": 137285792,
					  "status": "imported",
					  "errors": []
					}
				  ],
				  "total": 1
				}
			}`,
		},
		// Test No Client-Id or Api-Key
		{
			http.StatusUnauthorized,
			map[string]string{},
			&GetProductImportStatusParams{},
			`{
				"code": 16,
				"message": "Client-Id and Api-Key headers are required"
			}`,
		},
	}
	for _, test := range tests {
		c := NewMockClient(core.NewMockHttpHandler(test.statusCode, test.response, test.headers))
		ctx, _ := context.WithTimeout(context.Background(), testTimeout)
		resp, err := c.Products().GetProductImportStatus(ctx, test.params)
		if err != nil {
			t.Error(err)
		}
		if resp.StatusCode != test.statusCode {
			t.Errorf("got wrong status code: got: %d, expected: %d", resp.StatusCode, test.statusCode)
		}
		if resp.StatusCode == http.StatusOK {
			if len(resp.Result.Items) > 0 {
				if resp.Result.Items[0].ProductId == 0 {
					t.Errorf("Product id cannot be 0")
				}
				if resp.Result.Items[0].OfferId == "" {
					t.Errorf("Offer id cannot be empty")
				}
			}
		}
	}
}
func TestCreateProductByOzonID(t *testing.T) {
	t.Parallel()
	tests := []struct {
		statusCode int
		headers    map[string]string
		params     *CreateProductByOzonIDParams
		response   string
	}{
		// Test Ok
		{
			http.StatusOK,
			map[string]string{"Client-Id": "my-client-id", "Api-Key": "my-api-key"},
			&CreateProductByOzonIDParams{
				Items: []CreateProductsByOzonIDItem{
					{
						Name:         "string",
						OfferId:      "91132",
						OldPrice:     "2590",
						Price:        "2300",
						PremiumPrice: "2200",
						CurrencyCode: "RUB",
						SKU:          298789742,
						VAT:          "0.1",
					},
				},
			},
			`{
				"result": {
				  "task_id": 176594213,
				  "unmatched_sku_list": []
				}
			}`,
		},
		// Test No Client-Id or Api-Key
		{
			http.StatusUnauthorized,
			map[string]string{},
			&CreateProductByOzonIDParams{},
			`{
				"code": 16,
				"message": "Client-Id and Api-Key headers are required"
			}`,
		},
	}
	for _, test := range tests {
		c := NewMockClient(core.NewMockHttpHandler(test.statusCode, test.response, test.headers))
		ctx, _ := context.WithTimeout(context.Background(), testTimeout)
		resp, err := c.Products().CreateProductByOzonID(ctx, test.params)
		if err != nil {
			t.Error(err)
		}
		if resp.StatusCode != test.statusCode {
			t.Errorf("got wrong status code: got: %d, expected: %d", resp.StatusCode, test.statusCode)
		}
	}
}
func TestUpdateProductImages(t *testing.T) {
	t.Parallel()
	tests := []struct {
		statusCode int
		headers    map[string]string
		params     *UpdateProductImagesParams
		response   string
	}{
		// Test Ok
		{
			http.StatusOK,
			map[string]string{"Client-Id": "my-client-id", "Api-Key": "my-api-key"},
			&UpdateProductImagesParams{
				ColorImage: "string",
				Images:     []string{"string"},
				Images360:  []string{"string"},
				ProductId:  12345,
			},
			`{
				"result": {
				  "pictures": [
					{
					  "is_360": true,
					  "is_color": true,
					  "is_primary": true,
					  "product_id": 12345,
					  "state": "string",
					  "url": "string"
					},
					{
						"is_360": false,
						"is_color": true,
						"is_primary": true,
						"product_id": 12345,
						"state": "string",
						"url": "string"
					  }
				  ]
				}
			}`,
		},
		// Test No Client-Id or Api-Key
		{
			http.StatusUnauthorized,
			map[string]string{},
			&UpdateProductImagesParams{},
			`{
				"code": 16,
				"message": "Client-Id and Api-Key headers are required"
			}`,
		},
	}
	for _, test := range tests {
		c := NewMockClient(core.NewMockHttpHandler(test.statusCode, test.response, test.headers))
		ctx, _ := context.WithTimeout(context.Background(), testTimeout)
		resp, err := c.Products().UpdateProductImages(ctx, test.params)
		if err != nil {
			t.Error(err)
		}
		if resp.StatusCode != test.statusCode {
			t.Errorf("got wrong status code: got: %d, expected: %d", resp.StatusCode, test.statusCode)
		}
		if resp.StatusCode == http.StatusOK {
			if len(resp.Result.Pictures) != len(test.params.Images)+len(test.params.Images360) {
				t.Errorf("Amount of pictures in request and response are not equal")
			}
			if len(resp.Result.Pictures) > 0 {
				if resp.Result.Pictures[0].ProductId != test.params.ProductId {
					t.Errorf("Product ids in request and response are not equal")
				}
			}
		}
	}
}
func TestCheckImageUploadingStatus(t *testing.T) {
	t.Parallel()
	tests := []struct {
		statusCode int
		headers    map[string]string
		params     *CheckImageUploadingStatusParams
		response   string
	}{
		// Test Ok
		{
			http.StatusOK,
			map[string]string{"Client-Id": "my-client-id", "Api-Key": "my-api-key"},
			&CheckImageUploadingStatusParams{
				ProductId: []int64{123456},
			},
			`{
				"result": {
				  "pictures": [
					{
					  "is_360": true,
					  "is_color": true,
					  "is_primary": true,
					  "product_id": 123456,
					  "state": "string",
					  "url": "string"
					}
				  ]
				}
			}`,
		},
		// Test No Client-Id or Api-Key
		{
			http.StatusUnauthorized,
			map[string]string{},
			&CheckImageUploadingStatusParams{},
			`{
				"code": 16,
				"message": "Client-Id and Api-Key headers are required"
			}`,
		},
	}
	for _, test := range tests {
		c := NewMockClient(core.NewMockHttpHandler(test.statusCode, test.response, test.headers))
		ctx, _ := context.WithTimeout(context.Background(), testTimeout)
		resp, err := c.Products().CheckImageUploadingStatus(ctx, test.params)
		if err != nil {
			t.Error(err)
		}
		if resp.StatusCode != test.statusCode {
			t.Errorf("got wrong status code: got: %d, expected: %d", resp.StatusCode, test.statusCode)
		}
		if resp.StatusCode == http.StatusOK {
			if len(resp.Result.Pictures) > 0 {
				if resp.Result.Pictures[0].ProductId != test.params.ProductId[0] {
					t.Errorf("Product ids in request and response are not equal")
				}
			}
		}
	}
}
func TestListProductsByIDs(t *testing.T) {
	t.Parallel()
	tests := []struct {
		statusCode int
		headers    map[string]string
		params     *ListProductsByIDsParams
		response   string
	}{
		// Test Ok
		{
			http.StatusOK,
			map[string]string{"Client-Id": "my-client-id", "Api-Key": "my-api-key"},
			&ListProductsByIDsParams{
				OfferId: []string{"010", "23"},
			},
			`{
				"result": {
				  "items": [
					{
					  "id": 78712196,
					  "name": "Как выбрать детские музыкальные инструменты. Ксилофон, бубен, маракасы и другие инструменты для детей до 6 лет. Мастер-класс о раннем музыкальном развитии от Монтессори-педагога",
					  "offer_id": "010",
					  "barcode": "",
					  "barcodes": [
						"2335900005",
						"7533900005"
					  ],
					  "buybox_price": "",
					  "description_category_id": 93726157,
					  "created_at": "2021-06-03T03:40:05.871465Z",
					  "images": [],
					  "has_discounted_item": true,
					  "is_discounted": true,
					  "discounted_stocks": {
						"coming": 0,
						"present": 0,
						"reserved": 0
					  },
					  "currency_code": "RUB",
					  "marketing_price": "",
					  "min_price": "",
					  "old_price": "1000.0000",
					  "premium_price": "590.0000",
					  "price": "690.0000",
					  "recommended_price": "",
					  "sources": [
						{
						  "is_enabled": true,
						  "sku": 269628393,
						  "source": "fbo"
						},
						{
						  "is_enabled": true,
						  "sku": 269628396,
						  "source": "fbs"
						}
					  ],
					  "state": "",
					  "stocks": {
						"coming": 0,
						"present": 13,
						"reserved": 0
					  },
					  "errors": [],
					  "updated_at": "2023-02-09T06:46:44.152Z",
					  "vat": "0.0",
					  "visible": true,
					  "visibility_details": {
						"has_price": false,
						"has_stock": true,
						"active_product": false,
						"reasons": {}
					  },
					  "price_index": "0.00",
					  "images360": [],
					  "is_kgt": false,
					  "color_image": "",
					  "primary_image": "https://cdn1.ozone.ru/s3/multimedia-y/6077810038.jpg",
					  "status": {
						"state": "price_sent",
						"state_failed": "",
						"moderate_status": "approved",
						"decline_reasons": [],
						"validation_state": "success",
						"state_name": "Продается",
						"state_description": "",
						"is_failed": false,
						"is_created": true,
						"state_tooltip": "",
						"item_errors": [],
						"state_updated_at": "2021-07-26T04:50:08.486697Z"
					  }
					},
					{
					  "id": 76723583,
					  "name": "Онлайн-курс по дрессировке собак \"Собака: инструкция по применению. Одинокий волк\"",
					  "offer_id": "23",
					  "barcode": "",
					  "buybox_price": "",
					  "category_id": 90635895,
					  "created_at": "2021-05-26T20:26:07.565586Z",
					  "images": [],
					  "marketing_price": "",
					  "min_price": "",
					  "old_price": "12200.0000",
					  "premium_price": "5490.0000",
					  "price": "6100.0000",
					  "recommended_price": "",
					  "sources": [
						{
						  "is_enabled": true,
						  "sku": 267684495,
						  "source": "fbo"
						},
						{
						  "is_enabled": true,
						  "sku": 267684498,
						  "source": "fbs"
						}
					  ],
					  "state": "",
					  "stocks": {
						"coming": 0,
						"present": 19,
						"reserved": 0
					  },
					  "errors": [],
					  "updated_at": "2023-02-09T06:46:44.152Z",
					  "vat": "0.0",
					  "visible": true,
					  "visibility_details": {
						"has_price": false,
						"has_stock": true,
						"active_product": false,
						"reasons": {}
					  },
					  "price_index": "0.00",
					  "images360": [],
					  "is_kgt": false,
					  "color_image": "",
					  "primary_image": "https://cdn1.ozone.ru/s3/multimedia-v/6062554531.jpg",
					  "status": {
						"state": "price_sent",
						"state_failed": "",
						"moderate_status": "approved",
						"decline_reasons": [],
						"validation_state": "success",
						"state_name": "Продается",
						"state_description": "",
						"is_failed": false,
						"is_created": true,
						"state_tooltip": "",
						"item_errors": [],
						"state_updated_at": "2021-05-31T12:35:09.714641Z"
					  }
					}
				  ]
				}
			}`,
		},
		// Test No Client-Id or Api-Key
		{
			http.StatusUnauthorized,
			map[string]string{},
			&ListProductsByIDsParams{},
			`{
				"code": 16,
				"message": "Client-Id and Api-Key headers are required"
			}`,
		},
	}
	for _, test := range tests {
		c := NewMockClient(core.NewMockHttpHandler(test.statusCode, test.response, test.headers))
		ctx, _ := context.WithTimeout(context.Background(), testTimeout)
		resp, err := c.Products().ListProductsByIDs(ctx, test.params)
		if err != nil {
			t.Error(err)
		}
		if resp.StatusCode != test.statusCode {
			t.Errorf("got wrong status code: got: %d, expected: %d", resp.StatusCode, test.statusCode)
		}
		if resp.StatusCode == http.StatusOK {
			if len(resp.Result.Items) != len(test.params.OfferId) {
				t.Errorf("Amount of offer ids in request and response are not equal")
			}
			if len(resp.Result.Items) > 0 {
				if resp.Result.Items[0].OfferId != test.params.OfferId[0] {
					t.Errorf("Offer ids in request and response are not equal")
				}
			}
		}
	}
}
func TestGetDescriptionOfProduct(t *testing.T) {
	t.Parallel()
	tests := []struct {
		statusCode int
		headers    map[string]string
		params     *GetDescriptionOfProductParams
		response   string
	}{
		// Test Ok
		{
			http.StatusOK,
			map[string]string{"Client-Id": "my-client-id", "Api-Key": "my-api-key"},
			&GetDescriptionOfProductParams{
				Filter: GetDescriptionOfProductFilter{
					ProductId:  []int64{213761435},
					Visibility: "ALL",
				},
				LastId: "okVsfA==«",
				SortBy: "ASC",
				Limit:  100,
			},
			`{
				"result": [
				  {
					"id": 213761435,
					"barcode": "",
					"description_category_id": 17038062,
					"name": "Пленка защитная для Xiaomi Redmi Note 10 Pro 5G",
					"offer_id": "21470",
					"height": 10,
					"depth": 210,
					"width": 140,
					"dimension_unit": "mm",
					"weight": 50,
					"weight_unit": "g",
					"images": [
					  {
						"file_name": "https://cdn1.ozone.ru/s3/multimedia-f/6190456071.jpg",
						"default": true,
						"index": 0
					  },
					  {
						"file_name": "https://cdn1.ozone.ru/s3/multimedia-7/6190456099.jpg",
						"default": false,
						"index": 1
					  },
					  {
						"file_name": "https://cdn1.ozone.ru/s3/multimedia-9/6190456065.jpg",
						"default": false,
						"index": 2
					  }
					],
					"image_group_id": "",
					"images360": [],
					"pdf_list": [],
					"attributes": [
					  {
						"attribute_id": 5219,
						"complex_id": 0,
						"values": [
						  {
							"dictionary_value_id": 970718176,
							"value": "универсальный"
						  }
						]
					  },
					  {
						"attribute_id": 11051,
						"complex_id": 0,
						"values": [
						  {
							"dictionary_value_id": 970736931,
							"value": "Прозрачный"
						  }
						]
					  },
					  {
						"attribute_id": 10100,
						"complex_id": 0,
						"values": [
						  {
							"dictionary_value_id": 0,
							"value": "false"
						  }
						]
					  },
					  {
						"attribute_id": 11794,
						"complex_id": 0,
						"values": [
						  {
							"dictionary_value_id": 970860783,
							"value": "safe"
						  }
						]
					  },
					  {
						"attribute_id": 9048,
						"complex_id": 0,
						"values": [
						  {
							"dictionary_value_id": 0,
							"value": "Пленка защитная для Xiaomi Redmi Note 10 Pro 5G"
						  }
						]
					  },
					  {
						"attribute_id": 5076,
						"complex_id": 0,
						"values": [
						  {
							"dictionary_value_id": 39638,
							"value": "Xiaomi"
						  }
						]
					  },
					  {
						"attribute_id": 9024,
						"complex_id": 0,
						"values": [
						  {
							"dictionary_value_id": 0,
							"value": "21470"
						  }
						]
					  },
					  {
						"attribute_id": 10015,
						"complex_id": 0,
						"values": [
						  {
							"dictionary_value_id": 0,
							"value": "false"
						  }
						]
					  },
					  {
						"attribute_id": 85,
						"complex_id": 0,
						"values": [
						  {
							"dictionary_value_id": 971034861,
							"value": "Brand"
						  }
						]
					  },
					  {
						"attribute_id": 9461,
						"complex_id": 0,
						"values": [
						  {
							"dictionary_value_id": 349824787,
							"value": "Защитная пленка для смартфона"
						  }
						]
					  },
					  {
						"attribute_id": 4180,
						"complex_id": 0,
						"values": [
						  {
							"dictionary_value_id": 0,
							"value": "Пленка защитная для Xiaomi Redmi Note 10 Pro 5G"
						  }
						]
					  },
					  {
						"attribute_id": 4191,
						"complex_id": 0,
						"values": [
						  {
							"dictionary_value_id": 0,
							"value": "Пленка предназначена для модели Xiaomi Redmi Note 10 Pro 5G. Защитная гидрогелевая пленка обеспечит защиту вашего смартфона от царапин, пыли, сколов и потертостей."
						  }
						]
					  },
					  {
						"attribute_id": 8229,
						"complex_id": 0,
						"values": [
						  {
							"dictionary_value_id": 91521,
							"value": "Защитная пленка"
						  }
						]
					  }
					],
					"complex_attributes": [],
					"color_image": "",
					"last_id": ""
				  }
				],
				"total": 1,
				"last_id": "onVsfA=="
			}`,
		},
		// Test No Client-Id or Api-Key
		{
			http.StatusUnauthorized,
			map[string]string{},
			&GetDescriptionOfProductParams{},
			`{
				"code": 16,
				"message": "Client-Id and Api-Key headers are required"
			}`,
		},
	}
	for _, test := range tests {
		c := NewMockClient(core.NewMockHttpHandler(test.statusCode, test.response, test.headers))
		ctx, _ := context.WithTimeout(context.Background(), testTimeout)
		resp, err := c.Products().GetDescriptionOfProduct(ctx, test.params)
		if err != nil {
			t.Error(err)
		}
		if resp.StatusCode != test.statusCode {
			t.Errorf("got wrong status code: got: %d, expected: %d", resp.StatusCode, test.statusCode)
		}
		if resp.StatusCode == http.StatusOK {
			if len(resp.Result) != len(test.params.Filter.ProductId) && len(resp.Result) != len(test.params.Filter.OfferId) {
				t.Errorf("Amount of products in request and response are not equal")
			}
		}
	}
}
func TestGetProductDescription(t *testing.T) {
	t.Parallel()
	tests := []struct {
		statusCode int
		headers    map[string]string
		params     *GetProductDescriptionParams
		response   string
	}{
		// Test Ok
		{
			http.StatusOK,
			map[string]string{"Client-Id": "my-client-id", "Api-Key": "my-api-key"},
			&GetProductDescriptionParams{
				OfferId:   "5",
				ProductId: 73453843,
			},
			`{
				"result": {
				  "id": 73453843,
				  "offer_id": "5",
				  "name": "Онлайн курс по дрессировке собак \"Воспитанная собака за 3 недели\"",
				  "description": "Экспресс-курс - это сокращённый вариант курса \"Собака: инструкция по применению\", дающий базовый минимум знаний, навыков, умений. Это оптимальный вариант для совершения первых шагов по воспитанию!
Что дает Экспресс-курс: