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

Please add a PGX adapter #381

Open
veqryn opened this issue Sep 4, 2024 · 61 comments
Open

Please add a PGX adapter #381

veqryn opened this issue Sep 4, 2024 · 61 comments

Comments

@veqryn
Copy link

veqryn commented Sep 4, 2024

Is your feature request related to a problem? Please describe.
Right now the default for Jet is to let postgres arrays get generated into golang strings.
You can override the generated types, but then you'd also have to implement your own scan/value functions too.
Using PGX directly (github.com/jackc/pgx/v5) lets us do things like scanning postgres arrays directly to a slice ([]int64), and many other great features.
Unfortunately, Jet only accepts the standard library database/sql interface, which forces us to use github.com/jackc/pgx/v5/stdlib

Describe the solution you'd like
Other scanning libraries such as scany and ksql provide an adapter that lets their library use both the standard library interface and also PGX directly.
It would be great if Jet provided similar.

@go-jet
Copy link
Owner

go-jet commented Sep 7, 2024

Right now the default for Jet is to let postgres arrays get generated into golang strings.
You can override the generated types, but then you'd also have to implement your own scan/value functions too.

Check #380, this PR should add full array support.

Other scanning libraries such as scany and ksql provide an adapter that lets their library use both the standard library interface and also PGX directly.

I don't see how it is possible to have adapter that converts pgx to standard database/sql calls. For instance standard database connection has this method:

QueryContext(ctx context.Context, query string, args ...interface{}) (*sql.Rows, error)

All sql.Rows fields are private and there is no public constructor. Thus no adapter is possible. The only way is to add a new set of methods just for pgx.

Also, note that pgx does provide standard database interface - "github.com/jackc/pgx/v4/stdlib".

@veqryn
Copy link
Author

veqryn commented Sep 8, 2024

You'd have to look at scany and ksql to see how they do it. It might not be an adapter, it might be a copy of the implementation or something else.

@safaci2000
Copy link
Contributor

At least for scany, they simply re-implement the entire service, so it's duplicated to some degree. They implement their own interface Rows{} and then have an implementation using sql.Rows and another one using pgx.Rows.

The interface itself is generic and has no reference to sql.Rows, which I don't believe is the case for Jet.

qrm.Queryable return sql.Rows, which would have to be modified to be a more generic object/iterator as a response rather than sql.db

@DevX86
Copy link

DevX86 commented Sep 15, 2024

@go-jet Is this possible in any capacity? We need it to get rid of database/sql.

@DevX86
Copy link

DevX86 commented Sep 15, 2024

Scany seem the way to go but I love this library and want to see it flourish.

@houtn11
Copy link

houtn11 commented Sep 16, 2024

@DevX86 Why do you want to get rid of database/sql?

@DevX86
Copy link

DevX86 commented Sep 16, 2024

Massive speed improvements, even selecting 15,000 records I saw a 3x improvement due to the binary transfer and native bindings.

@DevX86
Copy link

DevX86 commented Sep 16, 2024

I'm trying to refactor into scany, my friends we were spoiled by go-jet. I can't live without it, especially when it comes to complex queries. @go-jet @houten11 Is there honestly nothing we can do for native pgx? (no database/sql)

@body20002
Copy link

You actually can use them together, use go-jet to generate the sql and pgx to execute it
Here's a simple example

selectStmt := SELECT(Table.AllColumns).FROM(Table).WHERE(Table.ID.EQ(1))).LIMIT(1)

func query[T any](db *pgxpool.Pool, ctx context.Context, stmt Statement) (*T, error) {
	query, args := stmt.Sql()

	row, _ := db.Query(ctx, query, args...)
	data, err := pgx.CollectOneRow(row, pgx.RowToAddrOfStructByPos[T])

	return data, err
}

# and just use it like this
data, err := query[Table](pool, ctx, selectStmt)

@DevX86
Copy link

DevX86 commented Sep 17, 2024

@body20002

I need something for complex structs like

type A struct {
    B struct {
        C
        D E `db:"F"`
        G H `db:"I"`
        J []struct {
            K
            L struct {
                M
                N struct {
                    O `alias:"P"`
                    Q []struct {
                        R `alias:"S"`
                    }
                }
            } `db:"T"`
            U []V `db:"W"`
        } `db:"X"`
        Y Z  `db:"AA"`
        AB AC `db:"AD"`
    }
}

Worked on something like that for hours last night combining scany with go jet, I don't like the results.
I flatten the struct then parse it like this.

var resultViews []views.ResultView
for _, tempGroup := range groupedTemps {
    firstTemp := tempGroup[0]

    // Extract embedded structs
    modelA := utils.ExtractEmbeddedStruct[models.ModelA](firstTemp)
    modelB := utils.ExtractEmbeddedStruct[models.ModelB](firstTemp)
    modelC := utils.ExtractEmbeddedStruct[models.ModelC](firstTemp)
    modelD := utils.ExtractEmbeddedStruct[models.ModelD](firstTemp)
    modelE := utils.ExtractEmbeddedStruct[models.ModelE](firstTemp).Resolve()

    // Collect items and build views
    modelFItems := utils.ExtractEmbeddedSlice[Temp, models.ModelF](tempGroup)
    modelGItems := utils.ExtractEmbeddedSlice[Temp, models.ModelG](tempGroup)
    modelHItems := utils.ExtractEmbeddedSlice[Temp, models.ModelH](tempGroup)
    modelIItems := utils.ExtractEmbeddedSlice[Temp, models.ModelI](tempGroup)

    // Continue processing with the extracted data
    // For example, building a result view:
    resultView := views.NewResultView(
        &modelA,
        modelFItems,
        &modelB,
        &modelC,
        modelE,
        // ... other parameters
    )

    resultViews = append(resultViews, *resultView)
}

And this which has a struct plus slice, but we have structs into slices into structs etc. So this wont scale.

type ProjectData struct {
    models.Project       `db:"projects"`
    ProjectRecords       []models.ProjectRecord `db:"project_records" json:"project_records"`
}
func (r *ResearchRepository) GetFilteredProjectData(ctx context.Context, researcherID uuid.UUID, projectID *uuid.UUID) ([]views.ProjectData, error) {
    ctx, span := researchTracer.Start(ctx, "ResearchRepository.GetFilteredProjectData")
    defer span.End()

    // SQL statement builder
    stmt := sqlbuilder.
        SELECT(
            projectTable.AllColumns,
            projectRecordTable.AllColumns).
        FROM(researcherTable.
            INNER_JOIN(projectTable, projectTable.ResearcherID.EQ(researcherTable.ID)).
            INNER_JOIN(projectDetailTable, projectDetailTable.ProjectID.EQ(projectTable.ID)).
            INNER_JOIN(projectTable, projectDetailTable.ProjectID.EQ(projectTable.ID)).
            INNER_JOIN(projectRecordTable, projectRecordTable.ProjectID.EQ(projectTable.ID))).
        WHERE(researcherTable.ID.EQ(sqlbuilder.UUID(researcherID)))

    fmt.Println(stmt.DebugSql())

    if projectID != nil {
        stmt = stmt.WHERE(projectTable.ID.EQ(sqlbuilder.UUID(*projectID)))
    }

    // Define a temporary structure to hold results
    type Temp struct {
        models.Project       `db:"projects"`
        models.ProjectRecord `db:"project_records"`
    }

    var temps []Temp
    if err := pgxscan.Select(ctx, r.db, &temps, stmt.DebugSql()); err != nil {
        return nil, utils.SpanErr(span, err)
    }

    groupedTemps, err := utils.GroupByID(temps)
    if err != nil {
        return nil, err
    }

    var projectDataList []views.ProjectData
    for _, tempGroup := range groupedTemps {
        firstTemp := tempGroup[0]

        projectData := views.ProjectData{
            Project:        utils.ExtractEmbeddedStruct[models.Project](firstTemp),
            ProjectRecords: utils.ExtractEmbeddedSlice[Temp, models.ProjectRecord](tempGroup),
        }

        projectDataList = append(projectDataList, projectData)
    }

    return projectDataList, nil
}

@DevX86
Copy link

DevX86 commented Sep 17, 2024

@go-jet For reference, scany can't handle maps, or nested structs, only flat structs and flat slices.

@dropwhile
Copy link

@DevX86 Do you really need scany? As @body20002 pointed out in their example, since v5 pgx has nice support for scanning on its own (CollectRows, CollectOneRow, RowToStructByName, RowToStructByNameLax, etc).
reference: https://ectobit.com/blog/pgx-v5-3/

I used to use scany, and just converted to using plain pgx after v5 came out.

@body20002
Copy link

@DevX86 I haven't tried it on complex queries and certainty not a query that complex, but It should work with nested structs given the struct is correct or corresponds to the query return values

I think pgx docs states so

@DevX86
Copy link

DevX86 commented Sep 17, 2024

@dropwhile @body20002 Hate to ping you guys but I'm in the middle of a major refactor of our database layer. As long as it's pgx native I don't mind which I use (or something new) my main concern are these complex queries. Can you post an example of a nested struct with slices on it?

@go-jet
Copy link
Owner

go-jet commented Sep 18, 2024

@DevX86 I'll look into `pgx' when I find some time. My intuition tells me that even if pgx is 3x faster than pq with database/sql in a real production environment, it will never make a significant difference. Because latency between user and server and server and database is measured in ms, and scanning in go structures is measured in ns. For example, if a scan takes 200 ns or 600 ns, it doesn't matter that much when the entire response to the user takes more than 10 ms (~200,000 times the difference).

Now, this is assuming that the database returns a reasonable amount of rows. What is a reasonable amount of rows depends on each setup, but a rough estimate is no more than 1000, preferably less than 100, for no more than 10 columns. If your queries return 15000 rows, it will take time to group that into the result regardless of the database driver used. By joining multiple tables in a single query, you can easily end up with a huge number of rows, in which case you'll need to consider pagination or even split the query into two or more.

@DevX86
Copy link

DevX86 commented Sep 18, 2024

@go-jet Thanks for the response, my normal queries including server response time are 2-3x faster. (Some around 10-30% but those add up) I used a 15k rows as an initial test. The joins are for our deeply nested data types, otherwise it ends up being a ton of requsets to the backend and constructing nested object out of many flat ones.

@DevX86
Copy link

DevX86 commented Sep 18, 2024

Also we get some cool things like better connection pooling. I can't decide if I should wait on the ideas ahead or continue with the refactor using the odd bridge I built, it's going to take a lot of man hours and I'm not sure about the maintainablity for other developers. For integration simplicity I also had to take the fields directly off the generated struct sso that's another pitfall to make this work 'seamlessly' with scany. I'm sure there's a way to patch it but I've put much headache and many hours into this, I'm just glad it's working at all.

We have ~1,000 queries in go-jet. Since day 1 of discovery I've loved this library and it's one of the few allowed into the codebase. If it's feasible please make this happen if you can @go-jet

@DevX86
Copy link

DevX86 commented Sep 18, 2024

I guess the bridge isn't horrendous with some renaming.

And changes like this for usage consistency.

// ExtractStruct extracts the first struct of type K from a slice of T.
func ExtractStruct[T any, K any](items []T) K {
	if len(items) == 0 {
		var zero K
		return zero
	}

	// Extract the first item
	firstItem := items[0]

	// Use reflection to extract the embedded struct of type K
	return ExtractEmbeddedStruct[K](firstItem)
}

The main thing I'm seeing in terms of the bridge implementation difficulty, is the mental coalescing of flat entities into a struct rather than thinking in terms of the end goal (here's the struct I need) . If anyone is interested I could do more battle testing on this and create a utility library for bridging jet-go and scany.

@DevX86
Copy link

DevX86 commented Sep 18, 2024

Here's the solution for scany implementation + works with encapsulated structs, if needed, modify your generator to include this.

// Tables.
UseTable(func(table metadata.Table) template.TableModel {
	return template.DefaultTableModel(table).
		UseField(func(columnMetaData metadata.Column) template.TableModelField {
			defaultTableModelField := template.DefaultTableModelField(columnMetaData)

			dbTag := columnMetaData.Name
			return defaultTableModelField.UseTags(
				fmt.Sprintf(`db:"%s"`, dbTag),
			)
		})
}).

@DevX86
Copy link

DevX86 commented Sep 18, 2024

Due to the issue below, a bridge is less ideal as we would have to make duplicate structs with nullable fields and some sort of Resolve() function with jet as an embedded struct, that either returns the normal struct or nil. Currently doing that but this at scale won't be fun. I'm considering having go-jet generate all tables again with all fields as pointers as NullTableName but that's a lot of duplication.

Scany Left Join Issue

@veqryn
Copy link
Author

veqryn commented Sep 19, 2024

As nice as Scany and Ksql are, it would be much better if Jet supported PGX natively.
Scany and Ksql have their own expectations of what the struct is supposed to look like.
Jet creates its own struct table models, and it would be best placed to be the library scanning rows into them.
It sounds like internally using your own interface, then accepting both sql.Rows and pgx.Rows, etc, would make them both work.

@DevX86
Copy link

DevX86 commented Sep 20, 2024

@veqryn Yeah scany's concept of scanning just isn't built for complexity like go-jet is. I just found a solution for the hopefully meantime which isn't so bad once you migrate a few queries. Native support would be awesome for this project and it honestly may attract some new people who are looking for an alternative solution or need to handle more complex structs.

@safaci2000
Copy link
Contributor

So, I'm digging into the pgx vs db.sql API differences, there's a few stdlib packages that pgx exposes, I'm assuming that both:

stdlib.OpenDBFromPool(*pgxpool.Pool)
sqlx.NewDb(stdlib.OpenDBFromPool(instance.dbPool), "pgx") returns *DB which includes extends sql.DB.

are not sufficient for your usecase? I think both of these return a sql.DB equivalent that can be used with Jet.

@DevX86
Copy link

DevX86 commented Sep 28, 2024

@safaci2000 Admittedly I'm not smart, are you saying the native version of pgx can be used with go-jet? They have a database/sql version and a native version which is much more performant.

@safaci2000
Copy link
Contributor

@safaci2000 Admittedly I'm not smart, are you saying the native version of pgx can be used with jetgo? They have a database/sql version and a native version which is much more performant.

I'm not sure. My point that it seems like there's a pattern you can use that may use the pgx driver. Since you seen very motivated I'm using that driver I was pointing out something to try out to see if you get the performance you need.

@kblomster
Copy link
Contributor

kblomster commented Sep 30, 2024

If you just need arrays, then this hack below might work for now. Note that this hinges on using pgx under the hood, but doing it via the database/sql interface.

// Array represents a postgres array containing elements of type T.
// We need this Scan implementation below because pgx is kinda awkward when used via database/sql; see e.g.
// https://github.com/jackc/pgx/issues/1458
// https://github.com/jackc/pgx/issues/1662
// https://github.com/jackc/pgx/issues/1779
type Array[T any] []T

func (a *Array[T]) Scan(src any) error {
	// implementation from https://github.com/jackc/pgx/issues/1779#issuecomment-1787125955
	m := pgtype.NewMap()

	v := (*[]T)(a)

	t, ok := m.TypeForValue(v)
	if !ok {
		return fmt.Errorf("cannot convert to sql.Scanner: cannot find registered type for %T", a)
	}

	var bufSrc []byte
	if src != nil {
		switch src := src.(type) {
		case string:
			bufSrc = []byte(src)
		case []byte:
			bufSrc = src
		default:
			bufSrc = []byte(fmt.Sprint(bufSrc))
		}
	}

	return m.Scan(t.OID, pgtype.TextFormatCode, bufSrc, v)
}

In future versions of golang, this should no longer be necessary because this proposal was accepted (although the implementation hasn't been merged yet): golang/go#67546

@DevX86
Copy link

DevX86 commented Oct 3, 2024

@go-jet Is there anything I can do to help with this feature even if it's just research?

@safaci2000
Copy link
Contributor

@go-jet Is there anything I can do to help with this feature even if it's just research?

Have you tried the pattern I gave you last week. You seem concerned about performance, try using jet with those patterns and see if the performance is comparable to your expectations for pure PGX

@DevX86
Copy link

DevX86 commented Oct 4, 2024

@go-jet Is there anything I can do to help with this feature even if it's just research?

Have you tried the pattern I gave you last week. You seem concerned about performance, try using jet with those patterns and see if the performance is comparable to your expectations for pure PGX

Not sure how to approach that, can you explain in more detail?

@DevX86
Copy link

DevX86 commented Oct 27, 2024

@go-jet Any findings on this?

@go-jet
Copy link
Owner

go-jet commented Oct 28, 2024

No, I haven’t had the opportunity to look into this yet.

@go-jet
Copy link
Owner

go-jet commented Oct 29, 2024

@DevX86 In the meantime you can check Prepared Statements Caching, it might give you some performance improvements.

@DevX86
Copy link

DevX86 commented Nov 4, 2024

I never saw this, this is pretty cool. I've migrated over 100 queries to go-jet + pgx + scany. Using this for anything unmigrated, thank you! @go-jet

@DevX86
Copy link

DevX86 commented Dec 1, 2024

@go-jet 100% independent of this issue but do you have a donation link? I've gotten so much out of this library it's unreal :)

@go-jet
Copy link
Owner

go-jet commented Dec 4, 2024

@go-jet 100% independent of this issue but do you have a donation link? I've gotten so much out of this library it's unreal :)

Thanks, I'm glad you find the library useful. I haven't set up a sponsorship link yet, it didn't seem worth the hassle before. I might set one up now.

Regarding this issue, I did a quick test with pgx and QRM, but the initial results weren't that satisfactory. Scanning took roughly the same. I'll give it another closer look next year.

@csg33k
Copy link

csg33k commented Dec 5, 2024

This might be relevant as well to the current issue. golang/go#67546

@DevX86
Copy link

DevX86 commented Dec 6, 2024

@csg33k Looks like it's making progress too, would this be a transparent fix for go-jet to use the native interface?

@DevX86
Copy link

DevX86 commented Dec 6, 2024

@go-jet Can you send me your reproduction? I've gotten solid improvements across the board.

@csg33k
Copy link

csg33k commented Dec 6, 2024

@csg33k Looks like it's making progress too, would this be a transparent fix for go-jet to use the native interface?

I wouldn't say transparent but it would allow the use of sql.DB irrelevant of the driver and it should make the transition much easier. At least that's my read on it.

@go-jet
Copy link
Owner

go-jet commented Dec 9, 2024

@DevX86 Check pgx branch and test benchmark.

@go-jet
Copy link
Owner

go-jet commented Dec 9, 2024

This might be relevant as well to the current issue. golang/go#67546

Thanks for the info. However, this seems like a change to the sql driver interface. Since the driver layer is a layer bellow database/sql, I don't think it will help much with this issue.

@DevX86
Copy link

DevX86 commented Dec 11, 2024

@go-jet Excited to try this! Doing now!

@DevX86
Copy link

DevX86 commented Dec 11, 2024

@go-jet

This is fetching 15,000 rows with 4/5 columns.

Had to replace uuid.UUID with []uint16 as the PGX version does not support uuid.UUID yet.

WithoutID

  • PQ Jet: 32ms
  • PGX Jet: 27ms
  • Scany: 15ms

With ID:

  • PQ Jet uuid.UUID : 34ms
  • PQ Jet []uint16 34ms
  • PGX Jet[]uint16: 30ms
  • Scany uuid.UUID : 20ms (won't scan with []uint16)

It's faster but I wonder why scany is much faster. It may have todo with their slice scanning?
I can try a more complex setup if you need.

@go-jet
Copy link
Owner

go-jet commented Dec 11, 2024

It's faster but I wonder why scany is much faster. It may have todo with their slice scanning?

Whats your destination?
Statement Query methods relies on reflection and a map to scan and group result, which adds latency.
If your destination is flat, you can eliminate both reflection and latency by using Rows method.

@DevX86
Copy link

DevX86 commented Dec 12, 2024

It's faster but I wonder why scany is much faster. It may have todo with their slice scanning?

Whats your destination? Statement Query methods relies on reflection and a map to scan and group result, which adds latency. If your destination is flat, you can eliminate both reflection and latency by using Rows method.

Destination is directly into the slice. Outside of strong tying, it's #2 of what makes jet nice.
Especially if there are a lot of fields or if a field in added / removed, we can hit regenerate and no code changes are needed on the repository layer.

@DevX86
Copy link

DevX86 commented Dec 14, 2024

@go-jet I may use this in the meantime anyway, is there an easy way to add uuid.UUID or would a lot outside of that need to be done to add robust support.

@safaci2000
Copy link
Contributor

@DevX86 not sure what you need for uuid.UUID but it should already be supported. There is a few uuid libs to choose from. They all have their ups and downs but if you use the custom code generator you can customize the output easily enough.

https://github.com/go-jet/jet/wiki/Generator#generator-customization

In my case I just added these lines:

import 	"github.com/gofrs/uuid" // or github.com/google/uuid  if you prefer.
//
if column.DataType.Kind == "base" {
	if column.DataType.Name == "uuid" {
		defaultTableModelField.Type = template.NewType(uuid.UUID{})
	}
        //add any other data type overrides 
}

@DevX86
Copy link

DevX86 commented Dec 14, 2024

PGX Wrapper
processRows

Looks like they're using rows.Next under the hood and
scanSliceElements

Is there any difference between how we're doing things for scanning into slices?

@DevX86
Copy link

DevX86 commented Dec 14, 2024

@DevX86 not sure what you need for uuid.UUID but it should already be supported. There is a few uuid libs to choose from. They all have their ups and downs but if you use the custom code generator you can customize the output easily enough.

https://github.com/go-jet/jet/wiki/Generator#generator-customization

In my case I just added these lines:

import 	"github.com/gofrs/uuid" // or github.com/google/uuid  if you prefer.
//
if column.DataType.Kind == "base" {
	if column.DataType.Name == "uuid" {
		defaultTableModelField.Type = template.NewType(uuid.UUID{})
	}
        //add any other data type overrides 
}

UUID is already generated in pq jet, are you sure we need this?
Default pq jet is "github.com/google/uuid"

@safaci2000
Copy link
Contributor

Well, I don't like the API of google, but I was also answering your question about adding UUID. Maybe I missed the context a bit. If you already know how to use UUID, then disregard my feedback.

@DevX86
Copy link

DevX86 commented Dec 14, 2024

PQ QRM
Looks like here is where uuid is being generated from (maybe not generated but we probably need it for scanning)

PGX QRM
I was going to add it and do a cool PR but we don't have a mapRowToStruct in pgx.

@go-jet
Copy link
Owner

go-jet commented Dec 16, 2024

@go-jet I may use this in the meantime anyway, is there an easy way to add uuid.UUID or would a lot outside of that need to be done to add robust support.

Yeah, you're right there is an issue with uuids on this branch. The problem is that pgx now returns UUIDs as a [16]byte array and no UUID type expects array(not even pgtype which is weird).

Anyway the fix is simple, just wrap existing UUID type and implement a new Scan method. For instance for google uuid:

type UUIDPGX struct {
	uuid.UUID
}

// Scan implements the database/sql Scanner interface.
func (dst *UUIDPGX) Scan(src any) error {
	if src == nil {
		*dst = UUIDPGX{}
		return nil
	}

	switch src := src.(type) {
	case [16]byte:
		dst.UUID = src
		return nil
	}

	return dst.UUID.Scan(src)
}

You'll also have to customize generator to use new UUID type.

@safaci2000
Copy link
Contributor

@go-jet if you check out: github.com/gofrs/uuid that's exactly what it does.

const Size = 16
type UUID [Size]byte

Don't have to use it, but data point.

@DevX86
Copy link

DevX86 commented Dec 16, 2024

@go-jet Is there a way around this? Ideally using google's uuid? We embed the models so having multiple UUID types will muddy up the codebase.

@go-jet
Copy link
Owner

go-jet commented Dec 17, 2024

@go-jet if you check out: github.com/gofrs/uuid that's exactly what it does.

This UUID type doesn't work either because the Scan method doesn't handle src of type [16]byte.

@go-jet
Copy link
Owner

go-jet commented Dec 17, 2024

@go-jet Is there a way around this? Ideally using google's uuid? We embed the models so having multiple UUID types will muddy up the codebase.

Another approach is to fork google's UUID library and add the missing case statements to handle [16]byte. You can then use your forked version in go.mod. Although you'll still need to customize the generator, no other changes is necessary.

@DevX86
Copy link

DevX86 commented Dec 17, 2024

@go-jet Is there a way around this? Ideally using google's uuid? We embed the models so having multiple UUID types will muddy up the codebase.

Another approach is to fork google's UUID library and add the missing case statements to handle [16]byte. You can then use your forked version in go.mod. Although you'll still need to customize the generator, no other changes is necessary.

Very interesting approach. This may be viable.
Scany doesn't have the odd conversion, but UUIDs can be supported as normal, do you know how they accomplish this?

@DevX86
Copy link

DevX86 commented Jan 22, 2025

@go-jet Any ideas?

@go-jet
Copy link
Owner

go-jet commented Jan 23, 2025

Scany doesn't have the odd conversion, but UUIDs can be supported as normal, do you know how they accomplish this?

Something intercepts the call before Scan and converts []byte to string. Haven't dig in deeper why or how?

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

No branches or pull requests

9 participants