cosmos-sdk/indexer/postgres/enum.go

69 lines
1.7 KiB
Go

package postgres
import (
"context"
"database/sql"
"fmt"
"io"
"strings"
"cosmossdk.io/schema"
)
// CreateEnumType creates an enum type in the database.
func (m *ModuleIndexer) CreateEnumType(ctx context.Context, conn DBConn, enum schema.EnumType) error {
typeName := enumTypeName(m.moduleName, enum)
row := conn.QueryRowContext(ctx, "SELECT 1 FROM pg_type WHERE typname = $1", typeName)
var res interface{}
if err := row.Scan(&res); err != nil {
if err != sql.ErrNoRows {
return fmt.Errorf("failed to check if enum type %q exists: %v", typeName, err) //nolint:errorlint // using %v for go 1.12 compat
}
} else {
// the enum type already exists
return nil
}
buf := new(strings.Builder)
err := CreateEnumTypeSql(buf, m.moduleName, enum)
if err != nil {
return err
}
sqlStr := buf.String()
if m.options.Logger != nil {
m.options.Logger("Creating enum type", sqlStr)
}
_, err = conn.ExecContext(ctx, sqlStr)
return err
}
// CreateEnumTypeSql generates a CREATE TYPE statement for the enum definition.
func CreateEnumTypeSql(writer io.Writer, moduleName string, enum schema.EnumType) error {
_, err := fmt.Fprintf(writer, "CREATE TYPE %q AS ENUM (", enumTypeName(moduleName, enum))
if err != nil {
return err
}
for i, value := range enum.Values {
if i > 0 {
_, err = fmt.Fprintf(writer, ", ")
if err != nil {
return err
}
}
_, err = fmt.Fprintf(writer, "'%s'", value)
if err != nil {
return err
}
}
_, err = fmt.Fprintf(writer, ");")
return err
}
// enumTypeName returns the name of the enum type scoped to the module.
func enumTypeName(moduleName string, enum schema.EnumType) string {
return fmt.Sprintf("%s_%s", moduleName, enum.Name)
}