Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Fix GUID conversion #207

Open
wants to merge 1 commit into
base: main
Choose a base branch
from

Conversation

andrea-mgn
Copy link

GUIDs are not handled like drivers for other SQL dialects (postgres, mariaDB, ...)
Someone had already signaled an issue:
https://stackoverflow.com/questions/70119954/go-mssqldb-returns-incorrect-id-value
But seems it has not been fixed in the meantime.

I'm not sure which would be the best way to change it, but we'd need a driver behaving like other ones.

@andrea-mgn andrea-mgn force-pushed the fix-guid-conversion branch 2 times, most recently from c376f79 to 9b50b3e Compare July 24, 2024 13:22
types.go Outdated
@@ -455,6 +455,25 @@ func writeByteLenType(w io.Writer, ti typeInfo, buf []byte) (err error) {
return
}

func writeGuidType(w io.Writer, ti typeInfo, buf []byte) (err error) {
if !(ti.Size == 0x10 || ti.Size == 0x00) {
panic("Invalid size for BYTELEN_TYPE")

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

for guid type?

@andrea-mgn andrea-mgn force-pushed the fix-guid-conversion branch 5 times, most recently from 7fcf586 to 8dbf84e Compare July 24, 2024 14:24
@shueybubbles
Copy link
Collaborator

It's why the driver has its own mssql.UniqueIdentifier type. I'm worried that changing the default behavior now could be a huge breaking change for other clients.

@shueybubbles
Copy link
Collaborator

I think we could take this, as long as we document the breaking change in the readme and CHANGELOG.md.
Take a look at comments from the earlier attempt in the old repo: denisenkom#505
The prior driver owner had requested a couple additional tests to verify round tripping and string conversion.

@shueybubbles
Copy link
Collaborator

also - see if frameworks like gorm have implemented any workarounds like reordering the slice already. They'll need to undo those workarounds if they pick up a version with this fix. Filing an issue against them could remind them to do so.

@shueybubbles
Copy link
Collaborator

another option is to make the behavior dependent on a connection string property, and new apps would set it, or a Connector property, etc.

@andrea-mgn
Copy link
Author

@shueybubbles, thanks for your comments.
Yes, I think that adding a property on a connection string would be a viable option (would it be read in msdsn.Config?), but I don't know what would be the proper way to pass it to the functions in types.go. (probably adding a flag in tdsSession then passing it explicitly to every function, but maybe there's a better way).
Another thing I cannot fix is the tests involving nullable UUIDs. For some reason the conversion isn't used properly and byte order isn't correct, unlike non-nullable UUIDs, and I cannot find which is the difference. Could you please advise how to fix it?

@andrea-mgn andrea-mgn force-pushed the fix-guid-conversion branch 15 times, most recently from 35ec6b8 to 84e7b5a Compare July 25, 2024 14:50
@andrea-mgn
Copy link
Author

@shueybubbles I've implemented the management of a new parameter in the connection string, named "guid conversion", to enabled the proper conversion of GUID; I've also duplicated involved tests so that they are executed both with and without this parameter.
I still have 2 problems, for which I'd need your help:

  • I'm not sure what would be the better way to pass this flag to TVP.encode (currently PID fields in TestTVP tests is not converted correctly due to the lack of this part)
  • I have no idea why nullable unique identifiers are not converted correctly when this parameter is enabled.

types.go Outdated Show resolved Hide resolved
@shueybubbles
Copy link
Collaborator

Sorry, I've not had much time to work on these driver issues lately. I will try to make some time in the next couple weeks.

@andrea-mgn
Copy link
Author

Sorry, I've not had much time to work on these driver issues lately. I will try to make some time in the next couple weeks.

ok, thank you

@andrea-mgn andrea-mgn force-pushed the fix-guid-conversion branch 8 times, most recently from bc8ab37 to 236ed76 Compare September 17, 2024 14:54
@andrea-mgn
Copy link
Author

I updated the PR, now I've solved the handling of TVP.encode, but it still fails the one about nullable Uniqueidentifier.
I tried debugging, when enabling GuidConversion, for both parseRow and parseNbcRow, the result returned by the Reader (the same one for both fields) inside columnContent:
for p_id is [103 69 35 1 171 137 239 205 1 35 69 103 137 171 205 239], correct;
for p_idNull is [1 35 69 103 137 171 205 239 1 35 69 103 137 171 205 239], byte swapping didn't happened (or more likely it happened twice).
@bharathpalaksha , @stuartpa, @PeteBassettBet365, could you please help to understand how to fix it?

@andrea-mgn
Copy link
Author

@shueybubbles, could you please help understanding why tests for nullable unique identifier don't pass, unlike the one for unique identifiers?

@shueybubbles
Copy link
Collaborator

@andrea-mgn I think it's because the switch statement in convertAssignRows treats a &UniqueIdentifier as []byte

sql.convertAssignRows (c:\Program Files\Go\src\database\sql\convert.go:243)
sql.(*Rows).Scan (c:\Program Files\Go\src\database\sql\sql.go:3354)
go-mssqldb.testTVP_WithTag (f:\git\go-mssqldb\tvp_go19_db_test.go:957)
go-mssqldb.TestTVP_WithTagAndGuidConversion (f:\git\go-mssqldb\tvp_go19_db_test.go:1126)
testing.tRunner (c:\Program Files\Go\src\testing\testing.go:1689)
testing.(*T).Run.gowrap1 (c:\Program Files\Go\src\testing\testing.go:1742)
runtime.goexit (c:\Program Files\Go\src\runtime\asm_amd64.s:1695)

It might need a case entry for UniqueIdentifier

case []byte:
  switch d := dest.(type) {
  case *string:
   if d == nil {
    return errNilPtr
   }
   *d = string(s)
   return nil
  case *any:
   if d == nil {
    return errNilPtr
   }
   *d = bytes.Clone(s)
   return nil
  case *[]byte:
   if d == nil {
    return errNilPtr
   }
   *d = bytes.Clone(s)
   return nil
  case *RawBytes:
   if d == nil {
    return errNilPtr
   }
   *d = s
   return nil
  }

In reply to: 2249651953

@andrea-mgn
Copy link
Author

@shueybubbles, many thanks for your analysis, but that is a standard go library, hence I would not know how to fix it.
@aberezin777 FYI

In reply to: 2249651953

@shueybubbles
Copy link
Collaborator

oh you're right, I wasn't looking closely at the file paths.
It might be that to use your parameter with nullable guids, the app would have to use the NullUniqueIdentifier or []byte type instead of UniqueIdentifier

Considering the goal of your change is to enable apps to get non-SQL Server formatted guids, such apps are not likely to use the UniqueIdentifier type in the first place. Maybe the tests can exclude that type when covering the new parameter.


In reply to: 2550719670

@andrea-mgn andrea-mgn force-pushed the fix-guid-conversion branch 4 times, most recently from 677f7d5 to 8063954 Compare December 19, 2024 10:25
@codecov-commenter
Copy link

codecov-commenter commented Dec 19, 2024

Codecov Report

Attention: Patch coverage is 88.31169% with 9 lines in your changes missing coverage. Please review.

Project coverage is 74.81%. Comparing base (dad23d2) to head (7ef689a).

Files with missing lines Patch % Lines
types.go 88.70% 5 Missing and 2 partials ⚠️
mssql.go 0.00% 0 Missing and 1 partial ⚠️
token.go 83.33% 1 Missing ⚠️
Additional details and impacted files
@@            Coverage Diff             @@
##             main     #207      +/-   ##
==========================================
+ Coverage   74.71%   74.81%   +0.10%     
==========================================
  Files          32       32              
  Lines        6410     6457      +47     
==========================================
+ Hits         4789     4831      +42     
- Misses       1333     1336       +3     
- Partials      288      290       +2     

☔ View full report in Codecov by Sentry.
📢 Have feedback on the report? Share it here.

@andrea-mgn
Copy link
Author

@shueybubbles, thank you.
I agree with you, so I've disabled tests for nullable identifiers in case of guidConversions.
It passes tests, and code coverage seems similar with the rest.

@shueybubbles
Copy link
Collaborator

thx. Please update the README with appropriate documentation for how to use this new parameter and what its limitations are.


In reply to: 2553394960

@andrea-mgn andrea-mgn force-pushed the fix-guid-conversion branch 2 times, most recently from 5a68a63 to 54eeb3e Compare December 20, 2024 10:55
@andrea-mgn
Copy link
Author

@shueybubbles, thanks. I've described briefly the new parameter in README.md, please check as I'm not a native speaker.

@@ -585,6 +607,9 @@ func (p Config) URL() *url.URL {
if p.ColumnEncryption {
q.Add("columnencryption", "true")
}

q.Add(GuidConversion, strconv.FormatBool(p.Encoding.GuidConversion))
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

don't add the parameter to the url unless its value is non-default

bulkcopy_test.go Outdated
testBulkcopy(t, true /*guidConversion*/)
}

func TestBulkcopyWithoutGuidConversion(t *testing.T) {
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

let's keep the test names for the non-guid conversion cases the same as they were before.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants