8

我想在 BYTEA 列中插入一些二进制数据

我将如何将 somefile.tar.gz 的内容插入到具有 BYTEA 列的表中?

  1. 一个例子会有所帮助
  2. 是否可以使用 BYTEA 的 HEX 格式(postgres 9.2)?
  3. 是否可以从/到 golang 向/从 postgres 流式传输?
4

2 回答 2

0

使用 sqlx 和 lib/pq 你只需使用一个字节片[]byte

    // data is []byte
    data, err := os.ReadFile("somefile.tar.gz")
    if err != nil {
       panic(err)
    }

    _, err = db.Exec("INSERT INTO test (id, data) values ($1, $2)", 1, data)
    if err != nil {
       panic(err)
    }

注意:如果您使用字符串文字对此进行测试,请记住 Go 字符串(除非另有说明)以 UTF-8 编码,因此您将存储在BYTEA列中的字节将是该字符串逐字的 UTF-8 编码. 例如[]byte("你好")商店e4 bd a0 e5 a5 bd

是否可以使用 BYTEA 的 HEX 格式(postgres 9.2)?

是的,您可以在 SQL 字符串中使用十六进制格式。除非字符串包含在反引号中`,否则必须转义斜杠。

    _, err = db.Exec("INSERT INTO test (id, data) values ($1, '\\xBAADC0DE')", 2)
    if err != nil {
      panic(err)
    }

与将 Go 字符串转换为字节切片不同,使用 Postgres 十六进制格式存储文字字节,例如ba ad c0 de. 这不会是 UTF-8 编码的。

是否可以从/到 golang 向/从 postgres 流式传输?

来自 Postgres,有点,带有sql.Rows. 更多信息在这里

于 2021-11-21T21:09:41.303 回答
0

如果愿意切换到github.com/jackc/pgx,可以流式传输大对象(PostgreSQL文档)。pgx.LargeObject类型实现:

io.Writer
io.Reader
io.Seeker
io.Closer

大对象存储在系统表中,没有可以在表的列中使用的大对象类型。大对象由它们的对象标识符引用。所以需要用文件元数据和oid映射来维护一个单独的表。

示例程序:

package main

import (
    "context"
    "io"
    "log"
    "os"
    "time"

    "github.com/jackc/pgx/v4"
)

const (
    // files table maps Large Object oid to file names
    createFileTable = `CREATE TABLE files (
        id oid primary key,
        name varchar,
        unique(name)
    );`
)

func main() {
    ctx, cancel := context.WithTimeout(context.TODO(), time.Minute)
    defer cancel()

    conn, err := pgx.Connect(ctx, "user=postgres host=/run/postgresql dbname=postgres")
    if err != nil {
        panic(err)
    }
    defer conn.Close(ctx)

    if _, err = conn.Exec(ctx, createFileTable); err != nil {
        panic(err)
    }

    written, err := storeFile(ctx, conn, "somefile.bin")
    log.Printf("storeFile written: %d", written)
    if err != nil {
        panic(err)
    }

    read, err := loadFile(ctx, conn, "somefile.bin")
    log.Printf("loadFile read: %d", read)
    if err != nil {
        panic(err)
    }
}

// storeFile as Large Object in the database.
// The resulting object identifier is stored along with the file name in the files table.
// The amount of written bytes and an erorr is returned, if one occured.
func storeFile(ctx context.Context, conn *pgx.Conn, name string) (written int64, err error) {
    file, err := os.Open(name)
    if err != nil {
        return 0, err
    }
    defer file.Close()

    // LargeObjects can only operate on an active TX
    tx, err := conn.Begin(ctx)
    if err != nil {
        return 0, err
    }
    defer tx.Rollback(ctx)

    lobs := tx.LargeObjects()

    // Create a new Large Object.
    // We pass 0, so the DB can pick an available oid for us.
    oid, err := lobs.Create(ctx, 0)
    if err != nil {
        return 0, err
    }

    // record the oid and filename in the files table
    _, err = tx.Exec(ctx, "INSERT INTO files (id, name) VALUES ($1, $2)", oid, name)
    if err != nil {
        return 0, err
    }

    // Open the new Object for writing.
    obj, err := lobs.Open(ctx, oid, pgx.LargeObjectModeWrite)
    if err != nil {
        return 0, err
    }

    // Copy the file stream to the Large Object stream
    written, err = io.Copy(obj, file)
    if err != nil {
        return written, err
    }

    err = tx.Commit(ctx)
    return written, err
}

// loadFile loads the file identified by name as Large Object
// and writes the contents to a local file by the same name.
// The amount of bytes read or an error is returned.
func loadFile(ctx context.Context, conn *pgx.Conn, name string) (read int64, err error) {
    tx, err := conn.Begin(ctx)
    if err != nil {
        return 0, err
    }
    defer tx.Rollback(ctx)

    var oid uint32
    err = conn.QueryRow(ctx, "SELECT id FROM files WHERE name = $1", name).Scan(&oid)
    if err != nil {
        return 0, err
    }

    file, err := os.Create(name)
    if err != nil {
        return 0, err
    }

    lobs := tx.LargeObjects()
    obj, err := lobs.Open(ctx, oid, pgx.LargeObjectModeRead)
    if err != nil {
        return 0, err
    }

    return io.Copy(file, obj)
}
于 2022-02-12T11:26:03.347 回答