Compare commits
20 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
f18e21f181 | ||
|
|
d56cf589da | ||
|
|
9e631a91b1 | ||
|
|
e575ff03a8 | ||
|
|
0ec82e6397 | ||
|
|
687a5df39f | ||
|
|
131af89f4f | ||
|
|
b93afeec2f | ||
|
|
bc56b4d16a | ||
|
|
b08b64f5b2 | ||
|
|
ad93704f41 | ||
|
|
b97ee3132b | ||
|
|
cfdde462f4 | ||
|
|
e039b583e2 | ||
|
|
26571a41fa | ||
|
|
559a100e90 | ||
|
|
6ff851823e | ||
|
|
db6e320b7f | ||
|
|
563e6baeda | ||
|
|
e1f7e99027 |
29
go.mod
29
go.mod
@@ -1,36 +1,9 @@
|
|||||||
module github.com/debyltech/go-snipcart
|
module github.com/debyltech/go-snipcart
|
||||||
|
|
||||||
go 1.20
|
go 1.19
|
||||||
|
|
||||||
require (
|
require (
|
||||||
github.com/debyltech/go-helpers v1.1.0
|
github.com/debyltech/go-helpers v1.1.0
|
||||||
github.com/debyltech/go-shippr v0.1.0
|
github.com/debyltech/go-shippr v0.1.0
|
||||||
github.com/gin-gonic/gin v1.9.0
|
|
||||||
github.com/skip2/go-qrcode v0.0.0-20200617195104-da1b6568686e
|
github.com/skip2/go-qrcode v0.0.0-20200617195104-da1b6568686e
|
||||||
)
|
)
|
||||||
|
|
||||||
require (
|
|
||||||
github.com/bytedance/sonic v1.8.0 // indirect
|
|
||||||
github.com/chenzhuoyu/base64x v0.0.0-20221115062448-fe3a3abad311 // indirect
|
|
||||||
github.com/gin-contrib/sse v0.1.0 // indirect
|
|
||||||
github.com/go-playground/locales v0.14.1 // indirect
|
|
||||||
github.com/go-playground/universal-translator v0.18.1 // indirect
|
|
||||||
github.com/go-playground/validator/v10 v10.11.2 // indirect
|
|
||||||
github.com/goccy/go-json v0.10.0 // indirect
|
|
||||||
github.com/json-iterator/go v1.1.12 // indirect
|
|
||||||
github.com/klauspost/cpuid/v2 v2.0.9 // indirect
|
|
||||||
github.com/leodido/go-urn v1.2.1 // indirect
|
|
||||||
github.com/mattn/go-isatty v0.0.17 // indirect
|
|
||||||
github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421 // indirect
|
|
||||||
github.com/modern-go/reflect2 v1.0.2 // indirect
|
|
||||||
github.com/pelletier/go-toml/v2 v2.0.6 // indirect
|
|
||||||
github.com/twitchyliquid64/golang-asm v0.15.1 // indirect
|
|
||||||
github.com/ugorji/go/codec v1.2.9 // indirect
|
|
||||||
golang.org/x/arch v0.0.0-20210923205945-b76863e36670 // indirect
|
|
||||||
golang.org/x/crypto v0.5.0 // indirect
|
|
||||||
golang.org/x/net v0.7.0 // indirect
|
|
||||||
golang.org/x/sys v0.5.0 // indirect
|
|
||||||
golang.org/x/text v0.7.0 // indirect
|
|
||||||
google.golang.org/protobuf v1.28.1 // indirect
|
|
||||||
gopkg.in/yaml.v3 v3.0.1 // indirect
|
|
||||||
)
|
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
package snipcart
|
package snipcart
|
||||||
|
|
||||||
type OrderStatus string
|
type OrderStatus string
|
||||||
|
type NotificationType string
|
||||||
|
|
||||||
const (
|
const (
|
||||||
Processed OrderStatus = "Processed"
|
Processed OrderStatus = "Processed"
|
||||||
@@ -10,4 +11,10 @@ const (
|
|||||||
Pending = "Pending"
|
Pending = "Pending"
|
||||||
Cancelled = "Cancelled"
|
Cancelled = "Cancelled"
|
||||||
Dispatched = "Dispatched"
|
Dispatched = "Dispatched"
|
||||||
|
|
||||||
|
Comment NotificationType = "Comment"
|
||||||
|
OrderStatusChanged = "OrderStatusChanged"
|
||||||
|
OrderShipped = "OrderShipped"
|
||||||
|
TrackingNumber = "TrackingNumber"
|
||||||
|
Invoice = "Invice"
|
||||||
)
|
)
|
||||||
@@ -5,6 +5,8 @@ import (
|
|||||||
"encoding/json"
|
"encoding/json"
|
||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"net/http"
|
||||||
|
"time"
|
||||||
|
|
||||||
helper "github.com/debyltech/go-helpers/json"
|
helper "github.com/debyltech/go-helpers/json"
|
||||||
"github.com/skip2/go-qrcode"
|
"github.com/skip2/go-qrcode"
|
||||||
@@ -14,10 +16,14 @@ const (
|
|||||||
defaultLimit = 50
|
defaultLimit = 50
|
||||||
apiUri = "https://app.snipcart.com"
|
apiUri = "https://app.snipcart.com"
|
||||||
ordersPath = "/api/orders"
|
ordersPath = "/api/orders"
|
||||||
|
productsPath = "/api/products"
|
||||||
|
validationPath = "/api/requestvalidation/"
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
orderUri = apiUri + ordersPath
|
orderUri = apiUri + ordersPath
|
||||||
|
productsUri = apiUri + productsPath
|
||||||
|
validationUri = apiUri + validationPath
|
||||||
)
|
)
|
||||||
|
|
||||||
type Client struct {
|
type Client struct {
|
||||||
@@ -48,26 +54,33 @@ type SnipcartItem struct {
|
|||||||
|
|
||||||
type SnipcartOrder struct {
|
type SnipcartOrder struct {
|
||||||
Token string `json:"token"`
|
Token string `json:"token"`
|
||||||
|
Created time.Time `json:"creationDate"`
|
||||||
|
Modified time.Time `json:"modificationDate"`
|
||||||
Invoice string `json:"invoiceNumber"`
|
Invoice string `json:"invoiceNumber"`
|
||||||
Subtotal float64 `json:"subtotal,omitempty"`
|
Subtotal float64 `json:"subtotal,omitempty"`
|
||||||
Currency string `json:"currency,omitempty"`
|
Currency string `json:"currency,omitempty"`
|
||||||
Total float64 `json:"grandTotal,omitempty"`
|
Total float64 `json:"grandTotal,omitempty"`
|
||||||
Status string `json:"status"`
|
Status string `json:"status"`
|
||||||
TotalWeight float64 `json:"totalWeight"`
|
TotalWeight float64 `json:"totalWeight"`
|
||||||
Name string `json:"shippingAddressName"`
|
ShippingAddress SnipcartShippingAddress `json:"shippingAddress,omitempty"`
|
||||||
Company string `json:"shippingAddressCompanyName"`
|
Name string `json:"shippingAddressName,omitempty"`
|
||||||
Address1 string `json:"shippingAddressAddress1"`
|
Company string `json:"shippingAddressCompanyName,omitempty"`
|
||||||
Address2 string `json:"shippingAddressAddress2"`
|
Address1 string `json:"shippingAddressAddress1,omitempty"`
|
||||||
City string `json:"shippingAddressCity"`
|
Address2 string `json:"shippingAddressAddress2,omitempty"`
|
||||||
Province string `json:"shippingAddressProvince"`
|
City string `json:"shippingAddressCity,omitempty"`
|
||||||
Country string `json:"shippingAddressCountry"`
|
Province string `json:"shippingAddressProvince,omitempty"`
|
||||||
PostalCode string `json:"shippingAddressPostalCode"`
|
Country string `json:"shippingAddressCountry,omitempty"`
|
||||||
|
PostalCode string `json:"shippingAddressPostalCode,omitempty"`
|
||||||
Phone string `json:"shippingAddressPhone,omitempty"`
|
Phone string `json:"shippingAddressPhone,omitempty"`
|
||||||
Email string `json:"email,omitempty"`
|
Email string `json:"email,omitempty"`
|
||||||
TrackingNumber string `json:"trackingNumber"`
|
TrackingNumber string `json:"trackingNumber"`
|
||||||
TrackingUrl string `json:"trackingUrl"`
|
TrackingUrl string `json:"trackingUrl"`
|
||||||
ShippingCost float64 `json:"shippingFees"`
|
ShippingCost float64 `json:"shippingFees"`
|
||||||
|
ShippingProvider string `json:"shippingProvider,omitempty"`
|
||||||
|
ShippingMethod string `json:"shippingMethod,omitempty"`
|
||||||
|
ShippingRateId string `json:"shippingRateUserDefinedId,omitempty"`
|
||||||
Items []SnipcartItem `json:"items"`
|
Items []SnipcartItem `json:"items"`
|
||||||
|
Metadata any `json:"metadata"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type SnipcartOrderUpdate struct {
|
type SnipcartOrderUpdate struct {
|
||||||
@@ -75,6 +88,7 @@ type SnipcartOrderUpdate struct {
|
|||||||
PaymentStatus string `json:"paymentStatus,omitempty"`
|
PaymentStatus string `json:"paymentStatus,omitempty"`
|
||||||
TrackingNumber string `json:"trackingNumber,omitempty"`
|
TrackingNumber string `json:"trackingNumber,omitempty"`
|
||||||
TrackingUrl string `json:"trackingUrl,omitempty"`
|
TrackingUrl string `json:"trackingUrl,omitempty"`
|
||||||
|
ShippingRateId string `json:"shippingRateUserDefinedId,omitempty"`
|
||||||
Metadata any `json:"metadata,omitempty"`
|
Metadata any `json:"metadata,omitempty"`
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -83,8 +97,63 @@ type SnipcartOrders struct {
|
|||||||
Items []SnipcartOrder
|
Items []SnipcartOrder
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewClient(snipcartApiKey string) Client {
|
type SnipcartTax struct {
|
||||||
return Client{
|
Name string `json:"name"`
|
||||||
|
Amount float64 `json:"amount"`
|
||||||
|
NumberForInvoice string `json:"numberForInvoice"`
|
||||||
|
Rate float64 `json:"rate"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type SnipcartNotification struct {
|
||||||
|
Type NotificationType `json:"type"`
|
||||||
|
DeliveryMethod string `json:"deliveryMethod"`
|
||||||
|
Message string `json:"message,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type SnipcartNotificationResponse struct {
|
||||||
|
Id string `json:"id"`
|
||||||
|
Created time.Time `json:"creationDate"`
|
||||||
|
Type NotificationType `json:"type"`
|
||||||
|
DeliveryMethod string `json:"deliveryMethod"`
|
||||||
|
Body string `json:"body"`
|
||||||
|
Message string `json:"message"`
|
||||||
|
Subject string `json:"subject"`
|
||||||
|
SentOn time.Time `json:"sentOn"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type SnipcartProductVariant struct {
|
||||||
|
Stock int `json:"stock"`
|
||||||
|
Variation []any `json:"variation"`
|
||||||
|
AllowBackorder bool `json:"allowOutOfStockPurchases"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type SnipcartProduct struct {
|
||||||
|
Token string `json:"id"`
|
||||||
|
Id string `json:"userDefinedId"`
|
||||||
|
Name string `json:"name"`
|
||||||
|
Stock int `json:"stock"`
|
||||||
|
TotalStock int `json:"totalStock"`
|
||||||
|
AllowBackorder bool `json:"allowOutOfStockPurchases"`
|
||||||
|
Variants []SnipcartProductVariant `json:"variants"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type SnipcartProductsResponse struct {
|
||||||
|
Keywords string `json:"keywords"`
|
||||||
|
UserDefinedId string `json:"userDefinedId"`
|
||||||
|
Archived bool `json:"archived"`
|
||||||
|
From time.Time `json:"from"`
|
||||||
|
To time.Time `json:"to"`
|
||||||
|
OrderBy string `json:"orderBy"`
|
||||||
|
Paginated bool `json:"hasMoreResults"`
|
||||||
|
TotalItems int `json:"totalItems"`
|
||||||
|
Offset int `json:"offset"`
|
||||||
|
Limit int `json:"limit"`
|
||||||
|
Sort []any `json:"sort"`
|
||||||
|
Items []SnipcartProduct `json:"items"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewClient(snipcartApiKey string) *Client {
|
||||||
|
return &Client{
|
||||||
SnipcartKey: snipcartApiKey,
|
SnipcartKey: snipcartApiKey,
|
||||||
AuthBase64: base64.StdEncoding.EncodeToString([]byte(snipcartApiKey + ":")),
|
AuthBase64: base64.StdEncoding.EncodeToString([]byte(snipcartApiKey + ":")),
|
||||||
}
|
}
|
||||||
@@ -155,7 +224,6 @@ func (s *Client) UpdateOrder(token string, orderUpdate *SnipcartOrderUpdate) (*S
|
|||||||
if response.StatusCode < 200 && response.StatusCode >= 300 {
|
if response.StatusCode < 200 && response.StatusCode >= 300 {
|
||||||
return nil, fmt.Errorf("unexpected response received: %s", response.Status)
|
return nil, fmt.Errorf("unexpected response received: %s", response.Status)
|
||||||
}
|
}
|
||||||
fmt.Println(response.Status)
|
|
||||||
|
|
||||||
defer response.Body.Close()
|
defer response.Body.Close()
|
||||||
|
|
||||||
@@ -167,3 +235,88 @@ func (s *Client) UpdateOrder(token string, orderUpdate *SnipcartOrderUpdate) (*S
|
|||||||
|
|
||||||
return &responseOrder, nil
|
return &responseOrder, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (s *Client) SendNotification(token string, notification *SnipcartNotification) (*SnipcartNotificationResponse, error) {
|
||||||
|
response, err := helper.Post(orderUri+"/"+token+"/notifications", "Basic", s.AuthBase64, notification)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
defer response.Body.Close()
|
||||||
|
|
||||||
|
var responseNotification SnipcartNotificationResponse
|
||||||
|
err = json.NewDecoder(response.Body).Decode(&responseNotification)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return &responseNotification, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *Client) ValidateWebhook(token string) error {
|
||||||
|
validateRequest, err := http.NewRequest("GET", validationUri+token, nil)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
client := &http.Client{}
|
||||||
|
|
||||||
|
auth := base64.StdEncoding.EncodeToString([]byte(s.SnipcartKey + ":"))
|
||||||
|
validateRequest.Header.Set("Authorization", fmt.Sprintf("Basic %s", auth))
|
||||||
|
validateRequest.Header.Set("Accept", "application/json")
|
||||||
|
|
||||||
|
validateResponse, err := client.Do(validateRequest)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("error validating webhook: %s", err.Error())
|
||||||
|
}
|
||||||
|
|
||||||
|
if validateResponse.StatusCode < 200 || validateResponse.StatusCode >= 300 {
|
||||||
|
return fmt.Errorf("non-2XX status code for validating webhook: %d", validateResponse.StatusCode)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *Client) GetProducts(queries map[string]string) (*SnipcartProductsResponse, error) {
|
||||||
|
response, err := helper.Get(productsUri, "Basic", s.AuthBase64, queries)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
if response.StatusCode < 200 && response.StatusCode >= 300 {
|
||||||
|
return nil, fmt.Errorf("unexpected response received: %s", response.Status)
|
||||||
|
}
|
||||||
|
|
||||||
|
defer response.Body.Close()
|
||||||
|
|
||||||
|
var products SnipcartProductsResponse
|
||||||
|
err = json.NewDecoder(response.Body).Decode(&products)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return &products, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *Client) GetProductById(id string) (*SnipcartProduct, error) {
|
||||||
|
response, err := helper.Get(productsUri, "Basic", s.AuthBase64, map[string]string{"userDefinedId": id})
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
if response.StatusCode < 200 && response.StatusCode >= 300 {
|
||||||
|
return nil, fmt.Errorf("unexpected response received: %s", response.Status)
|
||||||
|
}
|
||||||
|
|
||||||
|
defer response.Body.Close()
|
||||||
|
|
||||||
|
var products SnipcartProductsResponse
|
||||||
|
err = json.NewDecoder(response.Body).Decode(&products)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(products.Items) < 1 {
|
||||||
|
return nil, fmt.Errorf("no products with id '%s'", id)
|
||||||
|
}
|
||||||
|
|
||||||
|
return &products.Items[0], nil
|
||||||
|
}
|
||||||
|
|||||||
21
snipcart/webhook.go
Normal file
21
snipcart/webhook.go
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
package snipcart
|
||||||
|
|
||||||
|
type SnipcartShippingAddress struct {
|
||||||
|
FullName string `json:"fullName"`
|
||||||
|
FirstName string `json:"firstName"`
|
||||||
|
Name string `json:"name"`
|
||||||
|
Company string `json:"company"`
|
||||||
|
Address1 string `json:"address1"`
|
||||||
|
Address2 string `json:"address2"`
|
||||||
|
FullAddress string `json:"fullAddress"`
|
||||||
|
City string `json:"city"`
|
||||||
|
Country string `json:"country"`
|
||||||
|
PostalCode string `json:"postalCode"`
|
||||||
|
Province string `json:"province"`
|
||||||
|
Phone string `json:"phone"`
|
||||||
|
VatNumber string `json:"vatNumber,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type SnipcartWebhookTaxResponse struct {
|
||||||
|
Taxes []SnipcartTax `json:"taxes"`
|
||||||
|
}
|
||||||
@@ -1,33 +0,0 @@
|
|||||||
package webhook
|
|
||||||
|
|
||||||
import (
|
|
||||||
"bytes"
|
|
||||||
"encoding/json"
|
|
||||||
"io/ioutil"
|
|
||||||
|
|
||||||
"github.com/debyltech/go-shippr/shippo"
|
|
||||||
)
|
|
||||||
|
|
||||||
type Config struct {
|
|
||||||
ShippoApiKey string `json:"shippo_api_key"`
|
|
||||||
WeightUnit string `json:"weight_unit"`
|
|
||||||
DimensionUnit string `json:"dimension_unit"`
|
|
||||||
ManufactureCountry string `json:"manufacture_country"`
|
|
||||||
SenderAddress shippo.Address `json:"sender_address"`
|
|
||||||
DefaultParcel shippo.Parcel `json:"default_parcel"`
|
|
||||||
}
|
|
||||||
|
|
||||||
func NewConfigFromFile(filePath string) (*Config, error) {
|
|
||||||
configBytes, err := ioutil.ReadFile(filePath)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
var config Config
|
|
||||||
err = json.NewDecoder(bytes.NewBuffer(configBytes)).Decode(&config)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
return &config, nil
|
|
||||||
}
|
|
||||||
162
webhook.go
162
webhook.go
@@ -1,162 +0,0 @@
|
|||||||
package main
|
|
||||||
|
|
||||||
import (
|
|
||||||
"encoding/json"
|
|
||||||
"errors"
|
|
||||||
"flag"
|
|
||||||
"fmt"
|
|
||||||
"log"
|
|
||||||
"net/http"
|
|
||||||
"strconv"
|
|
||||||
"time"
|
|
||||||
|
|
||||||
"github.com/debyltech/go-shippr/shippo"
|
|
||||||
"github.com/debyltech/go-snipcart/snipcart"
|
|
||||||
"github.com/debyltech/go-snipcart/snipcart/webhook"
|
|
||||||
"github.com/gin-gonic/gin"
|
|
||||||
)
|
|
||||||
|
|
||||||
const (
|
|
||||||
ValidateUrl string = "https://app.snipcart.com/api/requestvalidation/"
|
|
||||||
)
|
|
||||||
|
|
||||||
type ShippingWebhookEvent struct {
|
|
||||||
EventName string `json:"eventName"`
|
|
||||||
CreatedOn time.Time `json:"createdOn"`
|
|
||||||
Order snipcart.SnipcartOrder `json:"content"`
|
|
||||||
}
|
|
||||||
|
|
||||||
type ShippingRatesResponse struct {
|
|
||||||
Cost float64 `json:"cost"`
|
|
||||||
Description string `json:"description"`
|
|
||||||
DeliveryDays int `json:"guaranteedDaysToDelivery"`
|
|
||||||
}
|
|
||||||
|
|
||||||
func ValidateWebhook(token string) error {
|
|
||||||
validateRequest, err := http.Get(ValidateUrl + token)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
if validateRequest.StatusCode < 200 || validateRequest.StatusCode >= 300 {
|
|
||||||
return errors.New("non-2XX response received")
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func HandleShippingRates(config *webhook.Config, shippoClient *shippo.Client) gin.HandlerFunc {
|
|
||||||
fn := func(c *gin.Context) {
|
|
||||||
err := ValidateWebhook(c.GetHeader("X-Snipcart-RequestToken"))
|
|
||||||
if err != nil {
|
|
||||||
c.AbortWithError(http.StatusBadRequest, err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
var event ShippingWebhookEvent
|
|
||||||
err = json.NewDecoder(c.Request.Body).Decode(&event)
|
|
||||||
if err != nil {
|
|
||||||
c.AbortWithError(http.StatusInternalServerError, err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
defer c.Request.Body.Close()
|
|
||||||
|
|
||||||
var lineItems []shippo.LineItem
|
|
||||||
for _, v := range event.Order.Items {
|
|
||||||
lineItems = append(lineItems, shippo.LineItem{
|
|
||||||
Quantity: v.Quantity,
|
|
||||||
TotalPrice: fmt.Sprintf("%.2f", v.TotalPrice),
|
|
||||||
Currency: event.Order.Currency,
|
|
||||||
Weight: fmt.Sprintf("%.2f", v.Weight),
|
|
||||||
WeightUnit: config.WeightUnit,
|
|
||||||
Title: v.Name,
|
|
||||||
ManufactureCountry: config.ManufactureCountry,
|
|
||||||
Sku: v.ID,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
parcel := config.DefaultParcel
|
|
||||||
parcel.WeightUnit = config.WeightUnit
|
|
||||||
parcel.DistanceUnit = config.DimensionUnit
|
|
||||||
parcel.Weight = fmt.Sprintf("%.2f", event.Order.TotalWeight)
|
|
||||||
|
|
||||||
rateRequest := shippo.RateRequest{
|
|
||||||
AddressFrom: shippo.Address{
|
|
||||||
Name: config.SenderAddress.Name,
|
|
||||||
Address1: config.SenderAddress.Address1,
|
|
||||||
Address2: config.SenderAddress.Address2,
|
|
||||||
City: config.SenderAddress.City,
|
|
||||||
State: config.SenderAddress.State,
|
|
||||||
Country: config.SenderAddress.Country,
|
|
||||||
PostalCode: config.SenderAddress.PostalCode,
|
|
||||||
},
|
|
||||||
AddressTo: shippo.Address{
|
|
||||||
Name: event.Order.Name,
|
|
||||||
Company: event.Order.Company,
|
|
||||||
Address1: event.Order.Address1,
|
|
||||||
Address2: event.Order.Address2,
|
|
||||||
City: event.Order.City,
|
|
||||||
Country: event.Order.Country,
|
|
||||||
State: event.Order.Province,
|
|
||||||
PostalCode: event.Order.PostalCode,
|
|
||||||
Phone: event.Order.Phone,
|
|
||||||
Email: event.Order.Email,
|
|
||||||
},
|
|
||||||
LineItems: lineItems,
|
|
||||||
Parcel: parcel,
|
|
||||||
}
|
|
||||||
|
|
||||||
rateResponse, err := shippoClient.GenerateRates(rateRequest)
|
|
||||||
if err != nil {
|
|
||||||
c.AbortWithError(http.StatusBadRequest, err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
var rates []ShippingRatesResponse
|
|
||||||
for _, v := range rateResponse.Rates {
|
|
||||||
cost, err := strconv.ParseFloat(v.Amount, 64)
|
|
||||||
if err != nil {
|
|
||||||
c.AbortWithError(http.StatusInternalServerError, err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
rates = append(rates, ShippingRatesResponse{
|
|
||||||
Cost: cost,
|
|
||||||
Description: v.Title,
|
|
||||||
DeliveryDays: v.EstimatedDays,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
c.JSON(http.StatusOK, rates)
|
|
||||||
}
|
|
||||||
|
|
||||||
return fn
|
|
||||||
}
|
|
||||||
|
|
||||||
func main() {
|
|
||||||
configPath := flag.String("config", "", "path to config.json")
|
|
||||||
flag.Parse()
|
|
||||||
|
|
||||||
if *configPath == "" {
|
|
||||||
log.Fatal("config path not defined")
|
|
||||||
}
|
|
||||||
|
|
||||||
config, err := webhook.NewConfigFromFile(*configPath)
|
|
||||||
if err != nil {
|
|
||||||
log.Fatal(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
shippoClient := shippo.NewClient(config.ShippoApiKey)
|
|
||||||
|
|
||||||
r := gin.Default()
|
|
||||||
|
|
||||||
api := r.Group("/api")
|
|
||||||
{
|
|
||||||
v1 := api.Group("/v1")
|
|
||||||
{
|
|
||||||
v1.POST("/shipping", HandleShippingRates(config, &shippoClient))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
r.Run("localhost:8081")
|
|
||||||
}
|
|
||||||
Reference in New Issue
Block a user