2023-02-04 10:32:39 +00:00
package pkg
import (
"context"
Convert from cents to “price” and back
I do not want to use floats in the Go lang application, because it is
not supposed to do anything with these values other than to print and
retrieve them from the user; all computations will be performed by
PostgreSQL in cents.
That means i have to “convert” from the price format that users expect
to see (e.g., 1.234,56) to cents (e.g., 123456) and back when passing
data between Go and PostgreSQL, and that conversion depends on the
currency’s decimal places.
At first i did everything in Go, but saw that i would need to do it in
a loop when retrieving the list of products, and immediately knew it was
a mistake—i needed a PL/pgSQL function for that.
I still need to convert from string to float, however, when printing the
value to the user. Because the string representation is in C, but i
need to format it according to the locale with golang/x/text. That
package has the information of how to correctly format numbers, but it
is in an internal package that i can not use, and numbers.Digit only
accepts numeric types, not a string.
2023-02-05 12:55:12 +00:00
"fmt"
2023-02-04 10:32:39 +00:00
"github.com/jackc/pgx/v4"
"github.com/julienschmidt/httprouter"
"html/template"
"math"
"net/http"
)
type ProductEntry struct {
Slug string
Name string
Convert from cents to “price” and back
I do not want to use floats in the Go lang application, because it is
not supposed to do anything with these values other than to print and
retrieve them from the user; all computations will be performed by
PostgreSQL in cents.
That means i have to “convert” from the price format that users expect
to see (e.g., 1.234,56) to cents (e.g., 123456) and back when passing
data between Go and PostgreSQL, and that conversion depends on the
currency’s decimal places.
At first i did everything in Go, but saw that i would need to do it in
a loop when retrieving the list of products, and immediately knew it was
a mistake—i needed a PL/pgSQL function for that.
I still need to convert from string to float, however, when printing the
value to the user. Because the string representation is in C, but i
need to format it according to the locale with golang/x/text. That
package has the information of how to correctly format numbers, but it
is in an internal package that i can not use, and numbers.Digit only
accepts numeric types, not a string.
2023-02-05 12:55:12 +00:00
Price string
2023-02-04 10:32:39 +00:00
}
type productsIndexPage struct {
Products [ ] * ProductEntry
}
func IndexProducts ( w http . ResponseWriter , r * http . Request , _ httprouter . Params ) {
conn := getConn ( r )
company := mustGetCompany ( r )
page := & productsIndexPage {
Products : mustGetProductEntries ( r . Context ( ) , conn , company ) ,
}
mustRenderAppTemplate ( w , r , "products/index.gohtml" , page )
}
func GetProductForm ( w http . ResponseWriter , r * http . Request , params httprouter . Params ) {
locale := getLocale ( r )
conn := getConn ( r )
company := mustGetCompany ( r )
form := newProductForm ( r . Context ( ) , conn , locale , company )
slug := params [ 0 ] . Value
if slug == "new" {
w . WriteHeader ( http . StatusOK )
mustRenderNewProductForm ( w , r , form )
return
}
2023-02-08 12:47:36 +00:00
var productId int
err := conn . QueryRow ( r . Context ( ) , "select product_id, product.name, product.description, to_price(price, decimal_digits) from product join company using (company_id) join currency using (currency_code) where product.slug = $1" , slug ) . Scan ( & productId , form . Name , form . Description , form . Price )
2023-02-04 10:32:39 +00:00
if err != nil {
if err == pgx . ErrNoRows {
http . NotFound ( w , r )
return
} else {
panic ( err )
}
}
2023-02-08 12:47:36 +00:00
rows , err := conn . Query ( r . Context ( ) , "select tax_id from product_tax where product_id = $1" , productId )
if err != nil {
panic ( err )
}
defer rows . Close ( )
for rows . Next ( ) {
if err := rows . Scan ( form . Tax ) ; err != nil {
panic ( err )
}
}
2023-02-04 10:32:39 +00:00
w . WriteHeader ( http . StatusOK )
mustRenderEditProductForm ( w , r , form )
}
func mustRenderNewProductForm ( w http . ResponseWriter , r * http . Request , form * productForm ) {
mustRenderAppTemplate ( w , r , "products/new.gohtml" , form )
}
func mustRenderEditProductForm ( w http . ResponseWriter , r * http . Request , form * productForm ) {
mustRenderAppTemplate ( w , r , "products/edit.gohtml" , form )
}
func HandleAddProduct ( w http . ResponseWriter , r * http . Request , _ httprouter . Params ) {
conn := getConn ( r )
locale := getLocale ( r )
company := mustGetCompany ( r )
form := newProductForm ( r . Context ( ) , conn , locale , company )
if err := form . Parse ( r ) ; err != nil {
http . Error ( w , err . Error ( ) , http . StatusBadRequest )
return
}
if err := verifyCsrfTokenValid ( r ) ; err != nil {
http . Error ( w , err . Error ( ) , http . StatusForbidden )
return
}
if ! form . Validate ( ) {
mustRenderNewProductForm ( w , r , form )
return
}
2023-02-08 12:47:36 +00:00
tx := conn . MustBegin ( r . Context ( ) )
productId := tx . MustGetInteger ( r . Context ( ) , "insert into product (company_id, name, description, price) select company_id, $2, $3, parse_price($4, decimal_digits) from company join currency using (currency_code) where company_id = $1 returning product_id" , company . Id , form . Name , form . Description , form . Price )
if len ( form . Tax . Selected ) > 0 {
batch := & pgx . Batch { }
for _ , tax := range form . Tax . Selected {
batch . Queue ( "insert into product_tax(product_id, tax_id) values ($1, $2)" , productId , tax )
}
br := tx . SendBatch ( r . Context ( ) , batch )
for range form . Tax . Selected {
if _ , err := br . Exec ( ) ; err != nil {
panic ( err )
}
}
if err := br . Close ( ) ; err != nil {
panic ( err )
}
}
tx . MustCommit ( r . Context ( ) )
2023-02-04 10:32:39 +00:00
http . Redirect ( w , r , companyURI ( company , "/products" ) , http . StatusSeeOther )
}
func HandleUpdateProduct ( w http . ResponseWriter , r * http . Request , params httprouter . Params ) {
conn := getConn ( r )
locale := getLocale ( r )
company := mustGetCompany ( r )
form := newProductForm ( r . Context ( ) , conn , locale , company )
if err := form . Parse ( r ) ; err != nil {
http . Error ( w , err . Error ( ) , http . StatusBadRequest )
return
}
if err := verifyCsrfTokenValid ( r ) ; err != nil {
http . Error ( w , err . Error ( ) , http . StatusForbidden )
return
}
if ! form . Validate ( ) {
2023-02-12 20:03:46 +00:00
w . WriteHeader ( http . StatusUnprocessableEntity )
2023-02-04 10:32:39 +00:00
mustRenderEditProductForm ( w , r , form )
return
}
2023-02-08 12:47:36 +00:00
tx := conn . MustBegin ( r . Context ( ) )
slug := params [ 0 ] . Value
productId := tx . MustGetIntegerOrDefault ( r . Context ( ) , 0 , "update product set name = $1, description = $2, price = parse_price($3, decimal_digits) from company join currency using (currency_code) where product.company_id = company.company_id and product.slug = $4 returning product_id" , form . Name , form . Description , form . Price , slug )
if productId == 0 {
tx . MustRollback ( r . Context ( ) )
2023-02-04 10:32:39 +00:00
http . NotFound ( w , r )
}
2023-02-08 12:47:36 +00:00
batch := & pgx . Batch { }
batch . Queue ( "delete from product_tax where product_id = $1" , productId )
for _ , tax := range form . Tax . Selected {
batch . Queue ( "insert into product_tax(product_id, tax_id) values ($1, $2)" , productId , tax )
}
br := tx . SendBatch ( r . Context ( ) , batch )
for i := 0 ; i < batch . Len ( ) ; i ++ {
if _ , err := br . Exec ( ) ; err != nil {
panic ( err )
}
}
if err := br . Close ( ) ; err != nil {
panic ( err )
}
tx . MustCommit ( r . Context ( ) )
2023-02-04 10:32:39 +00:00
http . Redirect ( w , r , companyURI ( company , "/products/" + slug ) , http . StatusSeeOther )
}
func mustGetProductEntries ( ctx context . Context , conn * Conn , company * Company ) [ ] * ProductEntry {
Convert from cents to “price” and back
I do not want to use floats in the Go lang application, because it is
not supposed to do anything with these values other than to print and
retrieve them from the user; all computations will be performed by
PostgreSQL in cents.
That means i have to “convert” from the price format that users expect
to see (e.g., 1.234,56) to cents (e.g., 123456) and back when passing
data between Go and PostgreSQL, and that conversion depends on the
currency’s decimal places.
At first i did everything in Go, but saw that i would need to do it in
a loop when retrieving the list of products, and immediately knew it was
a mistake—i needed a PL/pgSQL function for that.
I still need to convert from string to float, however, when printing the
value to the user. Because the string representation is in C, but i
need to format it according to the locale with golang/x/text. That
package has the information of how to correctly format numbers, but it
is in an internal package that i can not use, and numbers.Digit only
accepts numeric types, not a string.
2023-02-05 12:55:12 +00:00
rows , err := conn . Query ( ctx , "select product.slug, product.name, to_price(price, decimal_digits) from product join company using (company_id) join currency using (currency_code) where company_id = $1 order by name" , company . Id )
2023-02-04 10:32:39 +00:00
if err != nil {
panic ( err )
}
defer rows . Close ( )
var entries [ ] * ProductEntry
for rows . Next ( ) {
entry := & ProductEntry { }
err = rows . Scan ( & entry . Slug , & entry . Name , & entry . Price )
if err != nil {
panic ( err )
}
entries = append ( entries , entry )
}
if rows . Err ( ) != nil {
panic ( rows . Err ( ) )
}
return entries
}
type productForm struct {
locale * Locale
Convert from cents to “price” and back
I do not want to use floats in the Go lang application, because it is
not supposed to do anything with these values other than to print and
retrieve them from the user; all computations will be performed by
PostgreSQL in cents.
That means i have to “convert” from the price format that users expect
to see (e.g., 1.234,56) to cents (e.g., 123456) and back when passing
data between Go and PostgreSQL, and that conversion depends on the
currency’s decimal places.
At first i did everything in Go, but saw that i would need to do it in
a loop when retrieving the list of products, and immediately knew it was
a mistake—i needed a PL/pgSQL function for that.
I still need to convert from string to float, however, when printing the
value to the user. Because the string representation is in C, but i
need to format it according to the locale with golang/x/text. That
package has the information of how to correctly format numbers, but it
is in an internal package that i can not use, and numbers.Digit only
accepts numeric types, not a string.
2023-02-05 12:55:12 +00:00
company * Company
2023-02-04 10:32:39 +00:00
Name * InputField
Description * InputField
Price * InputField
Tax * SelectField
}
func newProductForm ( ctx context . Context , conn * Conn , locale * Locale , company * Company ) * productForm {
return & productForm {
Convert from cents to “price” and back
I do not want to use floats in the Go lang application, because it is
not supposed to do anything with these values other than to print and
retrieve them from the user; all computations will be performed by
PostgreSQL in cents.
That means i have to “convert” from the price format that users expect
to see (e.g., 1.234,56) to cents (e.g., 123456) and back when passing
data between Go and PostgreSQL, and that conversion depends on the
currency’s decimal places.
At first i did everything in Go, but saw that i would need to do it in
a loop when retrieving the list of products, and immediately knew it was
a mistake—i needed a PL/pgSQL function for that.
I still need to convert from string to float, however, when printing the
value to the user. Because the string representation is in C, but i
need to format it according to the locale with golang/x/text. That
package has the information of how to correctly format numbers, but it
is in an internal package that i can not use, and numbers.Digit only
accepts numeric types, not a string.
2023-02-05 12:55:12 +00:00
locale : locale ,
company : company ,
2023-02-04 10:32:39 +00:00
Name : & InputField {
Name : "name" ,
Label : pgettext ( "input" , "Name" , locale ) ,
Type : "text" ,
Required : true ,
} ,
Description : & InputField {
Name : "description" ,
Label : pgettext ( "input" , "Description" , locale ) ,
2023-02-07 14:28:22 +00:00
Type : "textarea" ,
2023-02-04 10:32:39 +00:00
} ,
Price : & InputField {
Name : "price" ,
Label : pgettext ( "input" , "Price" , locale ) ,
Type : "number" ,
Required : true ,
Attributes : [ ] template . HTMLAttr {
` min="0" ` ,
Convert from cents to “price” and back
I do not want to use floats in the Go lang application, because it is
not supposed to do anything with these values other than to print and
retrieve them from the user; all computations will be performed by
PostgreSQL in cents.
That means i have to “convert” from the price format that users expect
to see (e.g., 1.234,56) to cents (e.g., 123456) and back when passing
data between Go and PostgreSQL, and that conversion depends on the
currency’s decimal places.
At first i did everything in Go, but saw that i would need to do it in
a loop when retrieving the list of products, and immediately knew it was
a mistake—i needed a PL/pgSQL function for that.
I still need to convert from string to float, however, when printing the
value to the user. Because the string representation is in C, but i
need to format it according to the locale with golang/x/text. That
package has the information of how to correctly format numbers, but it
is in an internal package that i can not use, and numbers.Digit only
accepts numeric types, not a string.
2023-02-05 12:55:12 +00:00
template . HTMLAttr ( fmt . Sprintf ( ` step="%v" ` , company . MinCents ( ) ) ) ,
2023-02-04 10:32:39 +00:00
} ,
} ,
Tax : & SelectField {
2023-02-05 13:06:33 +00:00
Name : "tax" ,
2023-02-08 12:47:36 +00:00
Label : pgettext ( "input" , "Taxes" , locale ) ,
Multiple : true ,
2023-02-12 20:06:48 +00:00
Options : mustGetTaxOptions ( ctx , conn , company ) ,
2023-02-04 10:32:39 +00:00
} ,
}
}
func ( form * productForm ) Parse ( r * http . Request ) error {
if err := r . ParseForm ( ) ; err != nil {
return err
}
form . Name . FillValue ( r )
form . Description . FillValue ( r )
form . Price . FillValue ( r )
form . Tax . FillValue ( r )
return nil
}
func ( form * productForm ) Validate ( ) bool {
validator := newFormValidator ( )
validator . CheckRequiredInput ( form . Name , gettext ( "Name can not be empty." , form . locale ) )
if validator . CheckRequiredInput ( form . Price , gettext ( "Price can not be empty." , form . locale ) ) {
Convert from cents to “price” and back
I do not want to use floats in the Go lang application, because it is
not supposed to do anything with these values other than to print and
retrieve them from the user; all computations will be performed by
PostgreSQL in cents.
That means i have to “convert” from the price format that users expect
to see (e.g., 1.234,56) to cents (e.g., 123456) and back when passing
data between Go and PostgreSQL, and that conversion depends on the
currency’s decimal places.
At first i did everything in Go, but saw that i would need to do it in
a loop when retrieving the list of products, and immediately knew it was
a mistake—i needed a PL/pgSQL function for that.
I still need to convert from string to float, however, when printing the
value to the user. Because the string representation is in C, but i
need to format it according to the locale with golang/x/text. That
package has the information of how to correctly format numbers, but it
is in an internal package that i can not use, and numbers.Digit only
accepts numeric types, not a string.
2023-02-05 12:55:12 +00:00
validator . CheckValidDecimal ( form . Price , form . company . MinCents ( ) , math . MaxFloat64 , gettext ( "Price must be a number greater than zero." , form . locale ) )
2023-02-04 10:32:39 +00:00
}
validator . CheckValidSelectOption ( form . Tax , gettext ( "Selected tax is not valid." , form . locale ) )
return validator . AllOK ( )
}