github.com/weaviate/weaviate@v1.24.6/usecases/objects/validation/properties_validation_test.go (about)

     1  //                           _       _
     2  // __      _____  __ ___   ___  __ _| |_ ___
     3  // \ \ /\ / / _ \/ _` \ \ / / |/ _` | __/ _ \
     4  //  \ V  V /  __/ (_| |\ V /| | (_| | ||  __/
     5  //   \_/\_/ \___|\__,_| \_/ |_|\__,_|\__\___|
     6  //
     7  //  Copyright © 2016 - 2024 Weaviate B.V. All rights reserved.
     8  //
     9  //  CONTACT: hello@weaviate.io
    10  //
    11  
    12  package validation
    13  
    14  import (
    15  	"context"
    16  	"encoding/json"
    17  	"fmt"
    18  	"reflect"
    19  	"testing"
    20  	"time"
    21  
    22  	"github.com/go-openapi/strfmt"
    23  	"github.com/google/uuid"
    24  	"github.com/stretchr/testify/assert"
    25  	"github.com/stretchr/testify/require"
    26  	"github.com/weaviate/weaviate/entities/additional"
    27  	"github.com/weaviate/weaviate/entities/models"
    28  	"github.com/weaviate/weaviate/entities/schema"
    29  	"github.com/weaviate/weaviate/usecases/config"
    30  )
    31  
    32  func TestValidator_extractAndValidateProperty(t *testing.T) {
    33  	type fields struct {
    34  		schema schema.Schema
    35  		exists exists
    36  		config *config.WeaviateConfig
    37  	}
    38  	type args struct {
    39  		ctx          context.Context
    40  		propertyName string
    41  		pv           interface{}
    42  		className    string
    43  		dataType     *schema.DataType
    44  	}
    45  	validatorFields := fields{
    46  		schema: schema.Schema{
    47  			Objects: &models.Schema{
    48  				Classes: []*models.Class{
    49  					{
    50  						Class: "BlobClass",
    51  						Properties: []*models.Property{
    52  							{
    53  								DataType: []string{"blob"},
    54  								Name:     "blobProperty",
    55  							},
    56  						},
    57  					},
    58  				},
    59  			},
    60  		},
    61  	}
    62  	tests := []struct {
    63  		name    string
    64  		fields  fields
    65  		args    args
    66  		want    interface{}
    67  		wantErr bool
    68  	}{
    69  		{
    70  			name:   "Validate blob - empty base64 encoded image string",
    71  			fields: validatorFields,
    72  			args: args{
    73  				ctx:          context.Background(),
    74  				propertyName: "blobProperty",
    75  				pv:           "",
    76  				className:    "BlobClass",
    77  				dataType:     getDataType(schema.DataTypeBlob),
    78  			},
    79  			want:    nil,
    80  			wantErr: true,
    81  		},
    82  		{
    83  			name:   "Validate blob - invalid base64 encoded image string",
    84  			fields: validatorFields,
    85  			args: args{
    86  				ctx:          context.Background(),
    87  				propertyName: "blobProperty",
    88  				pv:           "image",
    89  				className:    "BlobClass",
    90  				dataType:     getDataType(schema.DataTypeBlob),
    91  			},
    92  			want:    nil,
    93  			wantErr: true,
    94  		},
    95  		{
    96  			name:   "Validate blob - valid base64 encoded string but with type definition before image string",
    97  			fields: validatorFields,
    98  			args: args{
    99  				ctx:          context.Background(),
   100  				propertyName: "blobProperty",
   101  				pv:           "",
   102  				className:    "BlobClass",
   103  				dataType:     getDataType(schema.DataTypeBlob),
   104  			},
   105  			want:    nil,
   106  			wantErr: true,
   107  		},
   108  		{
   109  			name:   "Validate blob - valid base64 encoded string",
   110  			fields: validatorFields,
   111  			args: args{
   112  				ctx:          context.Background(),
   113  				propertyName: "blobProperty",
   114  				pv:           "iVBORw0KGgoAAAANSUhEUgAAAGAAAAA/CAYAAAAfQM0aAAAAGXRFWHRTb2Z0d2FyZQBBZG9iZSBJbWFnZVJlYWR5ccllPAAAAyRpVFh0WE1MOmNvbS5hZG9iZS54bXAAAAAAADw/eHBhY2tldCBiZWdpbj0i77u/IiBpZD0iVzVNME1wQ2VoaUh6cmVTek5UY3prYzlkIj8+IDx4OnhtcG1ldGEgeG1sbnM6eD0iYWRvYmU6bnM6bWV0YS8iIHg6eG1wdGs9IkFkb2JlIFhNUCBDb3JlIDUuMy1jMDExIDY2LjE0NTY2MSwgMjAxMi8wMi8wNi0xNDo1NjoyNyAgICAgICAgIj4gPHJkZjpSREYgeG1sbnM6cmRmPSJodHRwOi8vd3d3LnczLm9yZy8xOTk5LzAyLzIyLXJkZi1zeW50YXgtbnMjIj4gPHJkZjpEZXNjcmlwdGlvbiByZGY6YWJvdXQ9IiIgeG1sbnM6eG1wPSJodHRwOi8vbnMuYWRvYmUuY29tL3hhcC8xLjAvIiB4bWxuczp4bXBNTT0iaHR0cDovL25zLmFkb2JlLmNvbS94YXAvMS4wL21tLyIgeG1sbnM6c3RSZWY9Imh0dHA6Ly9ucy5hZG9iZS5jb20veGFwLzEuMC9zVHlwZS9SZXNvdXJjZVJlZiMiIHhtcDpDcmVhdG9yVG9vbD0iQWRvYmUgUGhvdG9zaG9wIENTNiAoTWFjaW50b3NoKSIgeG1wTU06SW5zdGFuY2VJRD0ieG1wLmlpZDpCRjQ5NEM3RDI5QTkxMUUyOTc1NENCMzI4N0QwNDNCOSIgeG1wTU06RG9jdW1lbnRJRD0ieG1wLmRpZDpCRjQ5NEM3RTI5QTkxMUUyOTc1NENCMzI4N0QwNDNCOSI+IDx4bXBNTTpEZXJpdmVkRnJvbSBzdFJlZjppbnN0YW5jZUlEPSJ4bXAuaWlkOkJGNDk0QzdCMjlBOTExRTI5NzU0Q0IzMjg3RDA0M0I5IiBzdFJlZjpkb2N1bWVudElEPSJ4bXAuZGlkOkJGNDk0QzdDMjlBOTExRTI5NzU0Q0IzMjg3RDA0M0I5Ii8+IDwvcmRmOkRlc2NyaXB0aW9uPiA8L3JkZjpSREY+IDwveDp4bXBtZXRhPiA8P3hwYWNrZXQgZW5kPSJyIj8+WeGRxAAAB2hJREFUeNrUXFtslUUQ3hJCoQVEKy0k1qQgrRg0vaAJaq1tvJSgaLy8mKDF2IvxBY2Bgm8+iIoxvhB72tTUmKgPigbFKCEtxeKD9hZjAi3GJrYJtqRai7TQB+pMz/zwU/5zzsxe2u4kXwiwZ+bb/Xb/s7v/zEmrra1VTFsFeBRQCtgEuBWwkv5vHPAn4DdAB+B7wBjXcUNDQ8o2dXV1SmDzyhUtLS3tBPyxC9CdrN1ihi/swKuA7YD0BG1uJhQDngdcAnwDeJ86Ole2kLii+J2AFsA+wF9RjRalmEUHaZY8m6RDUYZtn6HPHiRfLm2hck0D7AScAdRH8UokwD2AnwA7UoiUyhaRD/S12dHg+8B1OWA/4BTgqVQCPEJL8haLBNDXEfJt03ziipYH+BJwHFAYJcAWwCeAZQ6CLyPfWyz584nrbCuj74eHwgKsddih2R1ba+jHJ65R1k6PuWNhAd4DZM/BTiWbdhwm5hPXsA0AngY8COgNP4JwSTyu4zE/P18VFhZKP7aNYuouXxFX5Ic8Nc2Ea2D/AfYCNgIORZ0DdusOfnFxcXDwUD09PZKP76alKDUR16KiIlVQUHDl7/39/Uozpg7Xac45YB0dGrQHHw07KVwJpRRbYiKuyCc8+MhXcyXocP2RnvMvJhr8QIBK08EPbGJiQuqq0mX7KD4GIohi4xVPTU0N6/BRamPwu7u7dZb3/RozkW3IB3lZEkGHayeI8FFVVdWaZAIUcD2Wl5fbHHy024XtC6QBkomA/XHIFb8X0Xamp6efASHqt27dGnkVkcNxVlFRoXJycmwOvuLGNmifVATsD/bLZezgKgKE2J+bm3sKHk3XXUWs4Mz87Oxs24OvOLEN26cUAfvFXAkrlKGBCDNXEbAajldXV1+5ijjP+KCrg855x+3nk2uy8SwDdIIIM1cRI6k+0NraqkZGRmzuKAIbFrYf0Q2UaPOA/Wpra3PBNfHhYHq6HbC5qanpGB7ETgPWc0TApTr7eyDolOaj6LRG+/W2Bn94eJg7+DpcowZ+AGb+642NjYfC3wEdXAdI1uK2Du2ksH2HrcHHfggGX4frNVcRMPh7BwcHN8ZiseuuIr4DvKXib29YX2bhmW+wEqYptsREXC2eWXS44oyfuYqYmpra19LSEnkaRgEG6Nj8gGRHESVCRkaG9Kg+IOyTiGtmZqatnZsOV/zMLnjcsF7KH5AIECVCX1+f6u3tlbg4oLmc2VyDy8HgPshg2yzmCo8aFsdAALzpw9dw23REwJkvHPwjSu92UcwVRcAnAd4LaQ6+CVe2AGivAe5WwhcdGp0aoVgmJuIqnBy2uSa18Buxs4AXAJMO401SjLOGfnziyhYg2GrtcNSxSfJ90pI/n7iyBUA7quKv/IYsxhmiZ/ZRy/x94soWAO1nwL0qnhVw2cD/ZfKBvjod9cEnrmwB0DBh9RUVfxHxhYrnUHLtEn2mlHyMOe6HT1wT7oISGSas4ntNzJmsVFczjnMBN1CbfwGD1BYPID8A/lFzbz5xZQsQnmWfExa6ecNVIsBKWuIlgA0qnjG2PLhsou0aZgF3qfil2fg89ssbrhwBNtB+GN/dLUnQ5kbCHYAnAFMAvGpsoY7OlS0krmOhxx7WLHwAeBLwVahN2uIUswgrPB5T8rRv7DxWqDwM+JaCjzue8b5wZe2C7gJ8quKVJqY599vJ1yZHffCJK0uA+wAfAtZYjIO+Gsi3TfOJK0sAfFP/jpKV+HBtKfkutOTPJ64sAVYD3qXgrmwpxVht6McnrmwBMAP4pjlYdRij3tCHT1xZAuDdermOA836gDKKqWNirob1ASZc2eeAl3QH36A+AGP+ohFWxNVSfYAuV9YKyKUTo/bgo2nUB5RQbImJuFqsD9DhyhbAuDgjMI36gFKX7S3XB5S6egSV2Bh8zYyDYjr4SGYi2yzmMIm5YnFGkFOLSQGNjY3X/BtaLBabWQF5XKcO6gOkZT950gAW6wPWuXoEZXEaOqoPyHLcPqkIwvqALFcCZHJmvqP6gEzH7VOKIKgPyHQlwIVUjRzWB1xw3H4+ubIFGE3VyGF9wKjj9ik3D4L6gFFXArCSTlEEzKe3LMIfwvYDNgcf+4P9csSVLUAXt7GD+oBuYfsuW4OvUR/Q7UoA/G2zaRvbOqEI0xRbYiKulusDTrgSYEg6sxKJIKwP6FLyjDYRV4v1ATpc2QKgNZtu6zTqA5o1ObM/h5eDyMvCtrlZObLgNhRv+jAHvkwqQjDzhYPfrvRvF0VcLdQHaHGNxWKrZv0d//hahcqr8Ccww1kRbwPuVMIXHRqd+ptimZiIq0F9gA2urEcQ2jkVf/tz0WG8ixTjnKEfn7iyBQi2WnuULLlV0qE9FrdzPnFlC4CGRQkvqyQ/MqRh6KtO2S948IkrWwC0XwHPAQ4r85z7w+TL1U8Y+8Q14S4oyjA9703AZ4AqFX8RvoTpN8i3/Bi/p+egHz5xZQsQGCasvqGuZhzj76DdpuIZx8FPuOAviWDG8e8qXl0yXxnHPnGdsf8FGAByGwC02iMZswAAAABJRU5ErkJggg==",
   115  				className:    "BlobClass",
   116  				dataType:     getDataType(schema.DataTypeBlob),
   117  			},
   118  			want:    "iVBORw0KGgoAAAANSUhEUgAAAGAAAAA/CAYAAAAfQM0aAAAAGXRFWHRTb2Z0d2FyZQBBZG9iZSBJbWFnZVJlYWR5ccllPAAAAyRpVFh0WE1MOmNvbS5hZG9iZS54bXAAAAAAADw/eHBhY2tldCBiZWdpbj0i77u/IiBpZD0iVzVNME1wQ2VoaUh6cmVTek5UY3prYzlkIj8+IDx4OnhtcG1ldGEgeG1sbnM6eD0iYWRvYmU6bnM6bWV0YS8iIHg6eG1wdGs9IkFkb2JlIFhNUCBDb3JlIDUuMy1jMDExIDY2LjE0NTY2MSwgMjAxMi8wMi8wNi0xNDo1NjoyNyAgICAgICAgIj4gPHJkZjpSREYgeG1sbnM6cmRmPSJodHRwOi8vd3d3LnczLm9yZy8xOTk5LzAyLzIyLXJkZi1zeW50YXgtbnMjIj4gPHJkZjpEZXNjcmlwdGlvbiByZGY6YWJvdXQ9IiIgeG1sbnM6eG1wPSJodHRwOi8vbnMuYWRvYmUuY29tL3hhcC8xLjAvIiB4bWxuczp4bXBNTT0iaHR0cDovL25zLmFkb2JlLmNvbS94YXAvMS4wL21tLyIgeG1sbnM6c3RSZWY9Imh0dHA6Ly9ucy5hZG9iZS5jb20veGFwLzEuMC9zVHlwZS9SZXNvdXJjZVJlZiMiIHhtcDpDcmVhdG9yVG9vbD0iQWRvYmUgUGhvdG9zaG9wIENTNiAoTWFjaW50b3NoKSIgeG1wTU06SW5zdGFuY2VJRD0ieG1wLmlpZDpCRjQ5NEM3RDI5QTkxMUUyOTc1NENCMzI4N0QwNDNCOSIgeG1wTU06RG9jdW1lbnRJRD0ieG1wLmRpZDpCRjQ5NEM3RTI5QTkxMUUyOTc1NENCMzI4N0QwNDNCOSI+IDx4bXBNTTpEZXJpdmVkRnJvbSBzdFJlZjppbnN0YW5jZUlEPSJ4bXAuaWlkOkJGNDk0QzdCMjlBOTExRTI5NzU0Q0IzMjg3RDA0M0I5IiBzdFJlZjpkb2N1bWVudElEPSJ4bXAuZGlkOkJGNDk0QzdDMjlBOTExRTI5NzU0Q0IzMjg3RDA0M0I5Ii8+IDwvcmRmOkRlc2NyaXB0aW9uPiA8L3JkZjpSREY+IDwveDp4bXBtZXRhPiA8P3hwYWNrZXQgZW5kPSJyIj8+WeGRxAAAB2hJREFUeNrUXFts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",
   119  			wantErr: false,
   120  		},
   121  		{
   122  			name:   "Validate blob - nil entry",
   123  			fields: validatorFields,
   124  			args: args{
   125  				ctx:          context.Background(),
   126  				propertyName: "blobProperty",
   127  				pv:           nil,
   128  				className:    "BlobClass",
   129  				dataType:     getDataType(schema.DataTypeBlob),
   130  			},
   131  			want:    nil,
   132  			wantErr: true,
   133  		},
   134  	}
   135  	for _, tt := range tests {
   136  		t.Run(tt.name, func(t *testing.T) {
   137  			v := &Validator{
   138  				exists: tt.fields.exists,
   139  				config: tt.fields.config,
   140  			}
   141  			got, err := v.extractAndValidateProperty(tt.args.ctx, tt.args.propertyName, tt.args.pv, tt.args.className, tt.args.dataType, "")
   142  			if (err != nil) != tt.wantErr {
   143  				t.Errorf("Validator.extractAndValidateProperty() error = %v, wantErr %v", err, tt.wantErr)
   144  				return
   145  			}
   146  			if !reflect.DeepEqual(got, tt.want) {
   147  				t.Errorf("Validator.extractAndValidateProperty() = %v, want %v", got, tt.want)
   148  			}
   149  		})
   150  	}
   151  }
   152  
   153  func getDataType(dataType schema.DataType) *schema.DataType {
   154  	return &dataType
   155  }
   156  
   157  func TestProperties(t *testing.T) {
   158  	const myBeacon = "weaviate://localhost/things/8e555f0d-8590-48c2-a9a6-70772ed14c0a"
   159  	myJournalClass := &models.Class{
   160  		Properties: []*models.Property{{Name: "inJournal", DataType: []string{"Journal"}}},
   161  	}
   162  	specs := map[string]struct {
   163  		class     *models.Class
   164  		obj       *models.Object
   165  		expErr    bool
   166  		expBeacon strfmt.URI
   167  	}{
   168  		"incorrect cref body - example from issue #1253": {
   169  			class: myJournalClass,
   170  			obj: &models.Object{
   171  				Properties: map[string]any{"inJournal": []any{map[string]any{
   172  					"beacon": map[string]any{"beacon": myBeacon},
   173  				}}},
   174  			},
   175  			expErr: true,
   176  		},
   177  		"complete beacon": {
   178  			class: myJournalClass,
   179  			obj: &models.Object{
   180  				Properties: map[string]any{"inJournal": []any{map[string]any{
   181  					"beacon": myBeacon,
   182  				}}},
   183  			},
   184  			expBeacon: myBeacon,
   185  		},
   186  		"beacon without class": {
   187  			class: myJournalClass,
   188  			obj: &models.Object{
   189  				Properties: map[string]any{"inJournal": []any{map[string]any{
   190  					"beacon": "weaviate://foo/8e555f0d-8590-48c2-a9a6-70772ed14c0a",
   191  				}}},
   192  			},
   193  			expBeacon: "weaviate://localhost/Journal/8e555f0d-8590-48c2-a9a6-70772ed14c0a",
   194  		},
   195  	}
   196  	for name, spec := range specs {
   197  		t.Run(name, func(t *testing.T) {
   198  			validator := &Validator{exists: func(_ context.Context, class string, _ strfmt.UUID, _ *additional.ReplicationProperties, _ string) (bool, error) {
   199  				return true, nil
   200  			}}
   201  			gotErr := validator.properties(context.Background(), spec.class, spec.obj, nil)
   202  			if spec.expErr {
   203  				require.Error(t, gotErr)
   204  				return
   205  			}
   206  			require.NoError(t, gotErr)
   207  			props := spec.obj.Properties
   208  			gotBeacon := extractBeacon(t, props)
   209  			assert.Equal(t, spec.expBeacon, gotBeacon)
   210  		})
   211  	}
   212  }
   213  
   214  func extractBeacon(t *testing.T, props models.PropertySchema) strfmt.URI {
   215  	require.IsType(t, map[string]any{}, props)
   216  	require.Contains(t, props.(map[string]any), "inJournal")
   217  	journalProp := props.(map[string]any)["inJournal"]
   218  	require.IsType(t, models.MultipleRef{}, journalProp)
   219  	require.Len(t, journalProp.(models.MultipleRef), 1)
   220  	gotBeacon := journalProp.(models.MultipleRef)[0].Beacon
   221  	return gotBeacon
   222  }
   223  
   224  func TestValidator_ValuesCasting(t *testing.T) {
   225  	t.Run("int(s)", func(t *testing.T) {
   226  		type testCase struct {
   227  			value         interface{}
   228  			expectedValue float64
   229  			expectedErr   bool
   230  		}
   231  
   232  		testCases := []testCase{
   233  			{
   234  				value:         json.Number("123"),
   235  				expectedValue: float64(123),
   236  				expectedErr:   false,
   237  			},
   238  			{
   239  				value:         int64(123),
   240  				expectedValue: float64(123),
   241  				expectedErr:   false,
   242  			},
   243  			{
   244  				value:         float64(123),
   245  				expectedValue: float64(123),
   246  				expectedErr:   false,
   247  			},
   248  
   249  			{
   250  				value:         json.Number("123.5"),
   251  				expectedValue: float64(0),
   252  				expectedErr:   true,
   253  			},
   254  			{
   255  				value:         float64(123.5),
   256  				expectedValue: float64(0),
   257  				expectedErr:   true,
   258  			},
   259  			{
   260  				value:         "123.5",
   261  				expectedValue: float64(0),
   262  				expectedErr:   true,
   263  			},
   264  			{
   265  				value:         "something",
   266  				expectedValue: float64(0),
   267  				expectedErr:   true,
   268  			},
   269  			{
   270  				value:         true,
   271  				expectedValue: float64(0),
   272  				expectedErr:   true,
   273  			},
   274  		}
   275  
   276  		for i, tc := range testCases {
   277  			t.Run(fmt.Sprintf("int #%d", i), func(t *testing.T) {
   278  				value, err := intVal(tc.value)
   279  
   280  				if tc.expectedErr {
   281  					assert.Error(t, err)
   282  				} else {
   283  					assert.NoError(t, err)
   284  				}
   285  				assert.Equal(t, tc.expectedValue, value)
   286  			})
   287  		}
   288  		for i, tc := range testCases {
   289  			t.Run(fmt.Sprintf("ints (single) #%d", i), func(t *testing.T) {
   290  				value, err := intArrayVal(tc.value)
   291  
   292  				assert.Error(t, err)
   293  				assert.Nil(t, value)
   294  			})
   295  			t.Run(fmt.Sprintf("ints (array) #%d", i), func(t *testing.T) {
   296  				value, err := intArrayVal([]interface{}{tc.value})
   297  
   298  				if tc.expectedErr {
   299  					assert.Error(t, err)
   300  					assert.Nil(t, value)
   301  				} else {
   302  					assert.NoError(t, err)
   303  					assert.Equal(t, []float64{tc.expectedValue}, value)
   304  				}
   305  			})
   306  		}
   307  	})
   308  
   309  	t.Run("number(s)", func(t *testing.T) {
   310  		type testCase struct {
   311  			value         interface{}
   312  			expectedValue float64
   313  			expectedErr   bool
   314  		}
   315  
   316  		testCases := []testCase{
   317  			{
   318  				value:         json.Number("123"),
   319  				expectedValue: float64(123),
   320  				expectedErr:   false,
   321  			},
   322  			{
   323  				value:         int64(123),
   324  				expectedValue: float64(123),
   325  				expectedErr:   false,
   326  			},
   327  			{
   328  				value:         float64(123),
   329  				expectedValue: float64(123),
   330  				expectedErr:   false,
   331  			},
   332  			{
   333  				value:         json.Number("123.5"),
   334  				expectedValue: float64(123.5),
   335  				expectedErr:   false,
   336  			},
   337  			{
   338  				value:         float64(123.5),
   339  				expectedValue: float64(123.5),
   340  				expectedErr:   false,
   341  			},
   342  
   343  			{
   344  				value:         "123.5",
   345  				expectedValue: float64(0),
   346  				expectedErr:   true,
   347  			},
   348  			{
   349  				value:         "something",
   350  				expectedValue: float64(0),
   351  				expectedErr:   true,
   352  			},
   353  			{
   354  				value:         true,
   355  				expectedValue: float64(0),
   356  				expectedErr:   true,
   357  			},
   358  		}
   359  
   360  		for i, tc := range testCases {
   361  			t.Run(fmt.Sprintf("number #%d", i), func(t *testing.T) {
   362  				value, err := numberVal(tc.value)
   363  
   364  				if tc.expectedErr {
   365  					assert.Error(t, err)
   366  				} else {
   367  					assert.NoError(t, err)
   368  				}
   369  				assert.Equal(t, tc.expectedValue, value)
   370  			})
   371  		}
   372  		for i, tc := range testCases {
   373  			t.Run(fmt.Sprintf("numbers (single) #%d", i), func(t *testing.T) {
   374  				value, err := numberArrayVal(tc.value)
   375  
   376  				assert.Error(t, err)
   377  				assert.Nil(t, value)
   378  			})
   379  			t.Run(fmt.Sprintf("numbers (array) #%d", i), func(t *testing.T) {
   380  				value, err := numberArrayVal([]interface{}{tc.value})
   381  
   382  				if tc.expectedErr {
   383  					assert.Error(t, err)
   384  					assert.Nil(t, value)
   385  				} else {
   386  					assert.NoError(t, err)
   387  					assert.Equal(t, []float64{tc.expectedValue}, value)
   388  				}
   389  			})
   390  		}
   391  	})
   392  
   393  	t.Run("string(s)", func(t *testing.T) {
   394  		type testCase struct {
   395  			value         interface{}
   396  			expectedValue string
   397  			expectedErr   bool
   398  		}
   399  
   400  		testCases := []testCase{
   401  			{
   402  				value:         "123.5",
   403  				expectedValue: "123.5",
   404  				expectedErr:   false,
   405  			},
   406  			{
   407  				value:         "something",
   408  				expectedValue: "something",
   409  				expectedErr:   false,
   410  			},
   411  
   412  			{
   413  				value:         json.Number("123"),
   414  				expectedValue: "",
   415  				expectedErr:   true,
   416  			},
   417  			{
   418  				value:         int64(123),
   419  				expectedValue: "",
   420  				expectedErr:   true,
   421  			},
   422  			{
   423  				value:         float64(123),
   424  				expectedValue: "",
   425  				expectedErr:   true,
   426  			},
   427  			{
   428  				value:         []byte("something"),
   429  				expectedValue: "",
   430  				expectedErr:   true,
   431  			},
   432  			{
   433  				value:         true,
   434  				expectedValue: "",
   435  				expectedErr:   true,
   436  			},
   437  		}
   438  
   439  		for i, tc := range testCases {
   440  			t.Run(fmt.Sprintf("string #%d", i), func(t *testing.T) {
   441  				value, err := stringVal(tc.value)
   442  
   443  				if tc.expectedErr {
   444  					assert.Error(t, err)
   445  				} else {
   446  					assert.NoError(t, err)
   447  				}
   448  				assert.Equal(t, tc.expectedValue, value)
   449  			})
   450  		}
   451  		for i, tc := range testCases {
   452  			t.Run(fmt.Sprintf("strings (single) #%d", i), func(t *testing.T) {
   453  				value, err := stringArrayVal(tc.value, "text")
   454  
   455  				assert.Error(t, err)
   456  				assert.Nil(t, value)
   457  			})
   458  			t.Run(fmt.Sprintf("strings (array) #%d", i), func(t *testing.T) {
   459  				value, err := stringArrayVal([]interface{}{tc.value}, "text")
   460  
   461  				if tc.expectedErr {
   462  					assert.Error(t, err)
   463  					assert.Nil(t, value)
   464  				} else {
   465  					assert.NoError(t, err)
   466  					assert.Equal(t, []string{tc.expectedValue}, value)
   467  				}
   468  			})
   469  		}
   470  	})
   471  
   472  	t.Run("bool(s)", func(t *testing.T) {
   473  		type testCase struct {
   474  			value         interface{}
   475  			expectedValue bool
   476  			expectedErr   bool
   477  		}
   478  
   479  		testCases := []testCase{
   480  			{
   481  				value:         true,
   482  				expectedValue: true,
   483  				expectedErr:   false,
   484  			},
   485  			{
   486  				value:         false,
   487  				expectedValue: false,
   488  				expectedErr:   false,
   489  			},
   490  
   491  			{
   492  				value:         float64(1),
   493  				expectedValue: false,
   494  				expectedErr:   true,
   495  			},
   496  			{
   497  				value:         int64(1),
   498  				expectedValue: false,
   499  				expectedErr:   true,
   500  			},
   501  			{
   502  				value:         "1",
   503  				expectedValue: false,
   504  				expectedErr:   true,
   505  			},
   506  			{
   507  				value:         "true",
   508  				expectedValue: false,
   509  				expectedErr:   true,
   510  			},
   511  			{
   512  				value:         "something",
   513  				expectedValue: false,
   514  				expectedErr:   true,
   515  			},
   516  		}
   517  
   518  		for i, tc := range testCases {
   519  			t.Run(fmt.Sprintf("bool #%d", i), func(t *testing.T) {
   520  				value, err := boolVal(tc.value)
   521  
   522  				if tc.expectedErr {
   523  					assert.Error(t, err)
   524  				} else {
   525  					assert.NoError(t, err)
   526  				}
   527  				assert.Equal(t, tc.expectedValue, value)
   528  			})
   529  		}
   530  		for i, tc := range testCases {
   531  			t.Run(fmt.Sprintf("bools (single) #%d", i), func(t *testing.T) {
   532  				value, err := boolArrayVal(tc.value)
   533  
   534  				assert.Error(t, err)
   535  				assert.Nil(t, value)
   536  			})
   537  			t.Run(fmt.Sprintf("bools (array) #%d", i), func(t *testing.T) {
   538  				value, err := boolArrayVal([]interface{}{tc.value})
   539  
   540  				if tc.expectedErr {
   541  					assert.Error(t, err)
   542  					assert.Nil(t, value)
   543  				} else {
   544  					assert.NoError(t, err)
   545  					assert.Equal(t, []bool{tc.expectedValue}, value)
   546  				}
   547  			})
   548  		}
   549  	})
   550  
   551  	t.Run("uuid(s)", func(t *testing.T) {
   552  		type testCase struct {
   553  			value         interface{}
   554  			expectedValue uuid.UUID
   555  			expectedErr   bool
   556  		}
   557  
   558  		testCases := []testCase{
   559  			{
   560  				value:         "e780b0a4-8d0e-4c09-898e-19d6b81e1e63",
   561  				expectedValue: uuid.MustParse("e780b0a4-8d0e-4c09-898e-19d6b81e1e63"),
   562  				expectedErr:   false,
   563  			},
   564  			{
   565  				value:         "e780b0a48d0e4c09898e19d6b81e1e63",
   566  				expectedValue: uuid.MustParse("e780b0a4-8d0e-4c09-898e-19d6b81e1e63"),
   567  				expectedErr:   false,
   568  			},
   569  
   570  			{
   571  				value:         float64(123),
   572  				expectedValue: [16]byte{},
   573  				expectedErr:   true,
   574  			},
   575  			{
   576  				value:         int64(123),
   577  				expectedValue: [16]byte{},
   578  				expectedErr:   true,
   579  			},
   580  			{
   581  				value:         "123",
   582  				expectedValue: [16]byte{},
   583  				expectedErr:   true,
   584  			},
   585  			{
   586  				value:         "something",
   587  				expectedValue: [16]byte{},
   588  				expectedErr:   true,
   589  			},
   590  			{
   591  				value:         true,
   592  				expectedValue: [16]byte{},
   593  				expectedErr:   true,
   594  			},
   595  		}
   596  
   597  		for i, tc := range testCases {
   598  			t.Run(fmt.Sprintf("uuid #%d", i), func(t *testing.T) {
   599  				value, err := uuidVal(tc.value)
   600  
   601  				if tc.expectedErr {
   602  					assert.Error(t, err)
   603  				} else {
   604  					assert.NoError(t, err)
   605  				}
   606  				assert.Equal(t, tc.expectedValue, value)
   607  			})
   608  		}
   609  		for i, tc := range testCases {
   610  			t.Run(fmt.Sprintf("uuids (single) #%d", i), func(t *testing.T) {
   611  				value, err := uuidArrayVal(tc.value)
   612  
   613  				assert.Error(t, err)
   614  				assert.Nil(t, value)
   615  			})
   616  			t.Run(fmt.Sprintf("uuids (array) #%d", i), func(t *testing.T) {
   617  				value, err := uuidArrayVal([]interface{}{tc.value})
   618  
   619  				if tc.expectedErr {
   620  					assert.Error(t, err)
   621  					assert.Nil(t, value)
   622  				} else {
   623  					assert.NoError(t, err)
   624  					assert.Equal(t, []uuid.UUID{tc.expectedValue}, value)
   625  				}
   626  			})
   627  		}
   628  	})
   629  
   630  	t.Run("date(s)", func(t *testing.T) {
   631  		type testCase struct {
   632  			value         interface{}
   633  			expectedValue time.Time
   634  			expectedErr   bool
   635  		}
   636  
   637  		testCases := []testCase{
   638  			{
   639  				value:         "2024-01-02T03:04:05.00Z",
   640  				expectedValue: time.Unix(1704164645, 0).UTC(),
   641  				expectedErr:   false,
   642  			},
   643  
   644  			{
   645  				value:         float64(123),
   646  				expectedValue: time.Time{},
   647  				expectedErr:   true,
   648  			},
   649  			{
   650  				value:         int64(123),
   651  				expectedValue: time.Time{},
   652  				expectedErr:   true,
   653  			},
   654  			{
   655  				value:         "123",
   656  				expectedValue: time.Time{},
   657  				expectedErr:   true,
   658  			},
   659  			{
   660  				value:         "something",
   661  				expectedValue: time.Time{},
   662  				expectedErr:   true,
   663  			},
   664  			{
   665  				value:         true,
   666  				expectedValue: time.Time{},
   667  				expectedErr:   true,
   668  			},
   669  		}
   670  
   671  		for i, tc := range testCases {
   672  			t.Run(fmt.Sprintf("date #%d", i), func(t *testing.T) {
   673  				value, err := dateVal(tc.value)
   674  
   675  				if tc.expectedErr {
   676  					assert.Error(t, err)
   677  				} else {
   678  					assert.NoError(t, err)
   679  				}
   680  				assert.Equal(t, tc.expectedValue, value)
   681  			})
   682  		}
   683  		for i, tc := range testCases {
   684  			t.Run(fmt.Sprintf("dates (single) #%d", i), func(t *testing.T) {
   685  				value, err := dateArrayVal(tc.value)
   686  
   687  				assert.Error(t, err)
   688  				assert.Nil(t, value)
   689  			})
   690  			t.Run(fmt.Sprintf("dates (array) #%d", i), func(t *testing.T) {
   691  				value, err := dateArrayVal([]interface{}{tc.value})
   692  
   693  				if tc.expectedErr {
   694  					assert.Error(t, err)
   695  					assert.Nil(t, value)
   696  				} else {
   697  					assert.NoError(t, err)
   698  					assert.Equal(t, []time.Time{tc.expectedValue}, value)
   699  				}
   700  			})
   701  		}
   702  	})
   703  }