// Copyright 2022 The Casdoor Authors. All Rights Reserved. // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. package object import ( "fmt" "github.com/casdoor/casdoor/pp" "github.com/casdoor/casdoor/util" "github.com/xorm-io/core" ) type Product struct { Owner string `xorm:"varchar(100) notnull pk" json:"owner"` Name string `xorm:"varchar(100) notnull pk" json:"name"` CreatedTime string `xorm:"varchar(100)" json:"createdTime"` DisplayName string `xorm:"varchar(100)" json:"displayName"` Image string `xorm:"varchar(100)" json:"image"` Detail string `xorm:"varchar(255)" json:"detail"` Description string `xorm:"varchar(100)" json:"description"` Tag string `xorm:"varchar(100)" json:"tag"` Currency string `xorm:"varchar(100)" json:"currency"` Price float64 `json:"price"` Quantity int `json:"quantity"` Sold int `json:"sold"` Providers []string `xorm:"varchar(100)" json:"providers"` ReturnUrl string `xorm:"varchar(1000)" json:"returnUrl"` State string `xorm:"varchar(100)" json:"state"` ProviderObjs []*Provider `xorm:"-" json:"providerObjs"` } func GetProductCount(owner, field, value string) (int64, error) { session := GetSession(owner, -1, -1, field, value, "", "") return session.Count(&Product{}) } func GetProducts(owner string) ([]*Product, error) { products := []*Product{} err := ormer.Engine.Desc("created_time").Find(&products, &Product{Owner: owner}) if err != nil { return products, err } return products, nil } func GetPaginationProducts(owner string, offset, limit int, field, value, sortField, sortOrder string) ([]*Product, error) { products := []*Product{} session := GetSession(owner, offset, limit, field, value, sortField, sortOrder) err := session.Find(&products) if err != nil { return products, err } return products, nil } func getProduct(owner string, name string) (*Product, error) { if owner == "" || name == "" { return nil, nil } product := Product{Owner: owner, Name: name} existed, err := ormer.Engine.Get(&product) if err != nil { return &product, nil } if existed { return &product, nil } else { return nil, nil } } func GetProduct(id string) (*Product, error) { owner, name := util.GetOwnerAndNameFromId(id) return getProduct(owner, name) } func UpdateProduct(id string, product *Product) (bool, error) { owner, name := util.GetOwnerAndNameFromId(id) if p, err := getProduct(owner, name); err != nil { return false, err } else if p == nil { return false, nil } affected, err := ormer.Engine.ID(core.PK{owner, name}).AllCols().Update(product) if err != nil { return false, err } return affected != 0, nil } func AddProduct(product *Product) (bool, error) { affected, err := ormer.Engine.Insert(product) if err != nil { return false, err } return affected != 0, nil } func DeleteProduct(product *Product) (bool, error) { affected, err := ormer.Engine.ID(core.PK{product.Owner, product.Name}).Delete(&Product{}) if err != nil { return false, err } return affected != 0, nil } func (product *Product) GetId() string { return fmt.Sprintf("%s/%s", product.Owner, product.Name) } func (product *Product) isValidProvider(provider *Provider) bool { for _, providerName := range product.Providers { if providerName == provider.Name { return true } } return false } func (product *Product) getProvider(providerId string) (*Provider, error) { provider, err := getProvider(product.Owner, providerId) if err != nil { return nil, err } if provider == nil { return nil, fmt.Errorf("the payment provider: %s does not exist", providerId) } if !product.isValidProvider(provider) { return nil, fmt.Errorf("the payment provider: %s is not valid for the product: %s", providerId, product.Name) } return provider, nil } func BuyProduct(id string, providerName string, user *User, host string) (string, string, error) { product, err := GetProduct(id) if err != nil { return "", "", err } if product == nil { return "", "", fmt.Errorf("the product: %s does not exist", id) } provider, err := product.getProvider(providerName) if err != nil { return "", "", err } pProvider, _, err := provider.getPaymentProvider() if err != nil { return "", "", err } owner := product.Owner productName := product.Name payerName := fmt.Sprintf("%s | %s", user.Name, user.DisplayName) paymentName := util.GenerateTimeId() productDisplayName := product.DisplayName originFrontend, originBackend := getOriginFromHost(host) returnUrl := fmt.Sprintf("%s/payments/%s/%s/result", originFrontend, owner, paymentName) notifyUrl := fmt.Sprintf("%s/api/notify-payment/%s/%s", originBackend, owner, paymentName) // Create an Order and get the payUrl payUrl, orderId, err := pProvider.Pay(providerName, productName, payerName, paymentName, productDisplayName, product.Price, product.Currency, returnUrl, notifyUrl) if err != nil { return "", "", err } // Create a Payment linked with Product and Order payment := Payment{ Owner: product.Owner, Name: paymentName, CreatedTime: util.GetCurrentTime(), DisplayName: paymentName, Provider: provider.Name, Type: provider.Type, ProductName: productName, ProductDisplayName: productDisplayName, Detail: product.Detail, Tag: product.Tag, Currency: product.Currency, Price: product.Price, ReturnUrl: product.ReturnUrl, User: user.Name, PayUrl: payUrl, State: pp.PaymentStateCreated, OutOrderId: orderId, } if provider.Type == "Dummy" { payment.State = pp.PaymentStatePaid } affected, err := AddPayment(&payment) if err != nil { return "", "", err } if !affected { return "", "", fmt.Errorf("failed to add payment: %s", util.StructToJson(payment)) } return payUrl, orderId, err } func ExtendProductWithProviders(product *Product) error { if product == nil { return nil } product.ProviderObjs = []*Provider{} m, err := getProviderMap(product.Owner) if err != nil { return err } for _, providerItem := range product.Providers { if provider, ok := m[providerItem]; ok { product.ProviderObjs = append(product.ProviderObjs, provider) } } return nil }