aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--auth/:w182
-rw-r--r--errors/errors.go15
-rw-r--r--errors/status.go1
-rw-r--r--invoice/controller.go153
-rw-r--r--invoice/hooks.go57
-rw-r--r--invoice/invoice.go61
-rw-r--r--invoice/router.go32
-rw-r--r--invoice/service.go74
-rw-r--r--invoice/validators.go71
-rw-r--r--main.go4
10 files changed, 460 insertions, 190 deletions
diff --git a/auth/:w b/auth/:w
deleted file mode 100644
index 7e3346e..0000000
--- a/auth/:w
+++ /dev/null
@@ -1,182 +0,0 @@
-/* openbills - Server for web based Libre Billing Software
- * Copyright (C) 2023 Vidhu Kant Sharma <vidhukant@vidhukant.com>
- *
- * This program is free software: you can redistribute it and/or modify
- * it under the terms of the GNU General Public License as published by
- * the Free Software Foundation, either version 3 of the License, or
- * (at your option) any later version.
- *
- * This program is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- * GNU General Public License for more details.
- *
- * You should have received a copy of the GNU General Public License
- * along with this program. If not, see <https://www.gnu.org/licenses/>.
- */
-
-package auth
-
-import (
- "github.com/gin-gonic/gin"
- "github.com/golang-jwt/jwt/v5"
- "github.com/spf13/viper"
- "golang.org/x/crypto/bcrypt"
- "vidhukant.com/openbills/user"
- "net/http"
- "time"
- "vidhukant.com/openbills/errors"
-)
-
-var (
- COST int
- AUTH_KEY, REFRESH_KEY []byte
-)
-func init() {
- COST = viper.GetInt("cryptography.password_hashing_cost")
- AUTH_KEY = []byte(viper.GetString("cryptography.auth_key"))
- REFRESH_KEY = []byte(viper.GetString("cryptography.refresh_key"))
-}
-
-func handleSignUp (ctx *gin.Context) {
- var user user.User
- ctx.Bind(&user)
-
- var err error
-
- // hash password
- var bytes []byte
- bytes, err = bcrypt.GenerateFromPassword([]byte(user.Password), 14)
- if err != nil {
- // TODO: handle potential errors
- ctx.Error(err)
- ctx.Abort()
- return
- }
- user.Password = string(bytes)
-
- err = user.Create()
- if err != nil {
- ctx.Error(err)
- ctx.Abort()
- return
- }
-
- // remove password hash from response
- user.Password = ""
-
- ctx.JSON(http.StatusOK, gin.H{
- "message": "success",
- "data": user,
- })
-}
-
-func handleSignIn (ctx *gin.Context) {
- var req LoginReq
- ctx.Bind(&req)
-
- var err error
- var u user.User
-
- err = user.CheckPassword(&u, req.AccountName, req.Method, req.Password)
- if err != nil {
- // TODO: handle potential errors
- ctx.Error(err)
- ctx.Abort()
- return
- }
-
- authToken, err := jwt.NewWithClaims(jwt.SigningMethodHS256,
- AuthClaims {
- jwt.RegisteredClaims {
- IssuedAt: jwt.NewNumericDate(time.Now()),
- ExpiresAt: jwt.NewNumericDate(time.Now().Add(time.Minute * 2)),
- },
- u.ID,
- },
- ).SignedString(AUTH_KEY)
- if err != nil {
- // TODO: handle potential errors
- ctx.Error(err)
- ctx.Abort()
- return
- }
-
- refreshToken, err := jwt.NewWithClaims(jwt.SigningMethodHS256,
- AuthClaims {
- jwt.RegisteredClaims {
- IssuedAt: jwt.NewNumericDate(time.Now()),
- ExpiresAt: jwt.NewNumericDate(time.Now().Add(time.Hour * 6)),
- },
- u.ID,
- },
- ).SignedString(REFRESH_KEY)
- if err != nil {
- // TODO: handle potential errors
- ctx.Error(err)
- ctx.Abort()
- return
- }
-
- // remove password hash from response
- u.Password = ""
-
- ctx.JSON(http.StatusOK, gin.H{
- "auth_token": authToken,
- "refresh_token": refreshToken,
- "message": "success",
- "data": u,
- })
-}
-
-func handleRefresh (ctx *gin.Context) {
- var req RefreshReq
- ctx.Bind(&req)
-
- tk, _ := jwt.ParseWithClaims(req.RefreshToken, &AuthClaims{}, func (token *jwt.Token) (interface{}, error) {
- return []byte(REFRESH_KEY), nil
- })
-
- claims, ok := tk.Claims.(*AuthClaims)
- if !ok {
- ctx.Error(errors.ErrUnauthorized)
- ctx.Abort()
- return
- }
-
- if !tk.Valid {
- eat := claims.ExpiresAt.Unix()
- if eat != 0 && eat < time.Now().Unix() {
- ctx.Error(errors.ErrSessionExpired)
- } else {
- ctx.Error(errors.ErrUnauthorized)
- }
-
- ctx.Abort()
- return
- }
-
- // TODO: if token is valid, check if user even exists before generating authToken
-
- authToken, err := jwt.NewWithClaims(jwt.SigningMethodHS256,
- AuthClaims {
- jwt.RegisteredClaims {
- IssuedAt: jwt.NewNumericDate(time.Now()),
- ExpiresAt: jwt.NewNumericDate(time.Now().Add(time.Minute * 2)),
- },
- claims.UserID,
- },
- ).SignedString(AUTH_KEY)
- if err != nil {
- // TODO: handle potential errors
- ctx.Error(err)
- ctx.Abort()
- return
- }
-
- ctx.JSON(http.StatusOK, gin.H{
- "auth_token": authToken,
- "message": "success",
- //"data": u,
- })
-}
diff --git a/errors/errors.go b/errors/errors.go
index 9a5a215..9fed3a3 100644
--- a/errors/errors.go
+++ b/errors/errors.go
@@ -54,13 +54,14 @@ var (
ErrBrandNotFound = errors.New("This Brand Does Not Exist")
// 409
- ErrNonUniqueGSTIN = errors.New("GSTIN Must Be Unique")
- ErrNonUniquePhone = errors.New("Phone Number Is Already In Use")
- ErrNonUniqueEmail = errors.New("Email Address Is Already In Use")
- ErrNonUniqueUsername = errors.New("Username Is Already In Use")
- ErrNonUniqueWebsite = errors.New("Website Is Already In Use")
- ErrNonUniqueBrandName = errors.New("Brand Name Is Already In Use")
- ErrNonUniqueBrandItem = errors.New("Item With Same Name And Brand Already Exists")
+ ErrNonUniqueGSTIN = errors.New("GSTIN Must Be Unique")
+ ErrNonUniquePhone = errors.New("Phone Number Is Already In Use")
+ ErrNonUniqueEmail = errors.New("Email Address Is Already In Use")
+ ErrNonUniqueUsername = errors.New("Username Is Already In Use")
+ ErrNonUniqueWebsite = errors.New("Website Is Already In Use")
+ ErrNonUniqueBrandName = errors.New("Brand Name Is Already In Use")
+ ErrNonUniqueBrandItem = errors.New("Item With Same Name And Brand Already Exists")
+ ErrNonUniqueInvoiceNumber = errors.New("Invoice Number Must Be Unique")
// 500
ErrInternalServerError = errors.New("Internal Server Error")
diff --git a/errors/status.go b/errors/status.go
index c7fc2a4..c9113a9 100644
--- a/errors/status.go
+++ b/errors/status.go
@@ -71,6 +71,7 @@ func StatusCodeFromErr(err error) int {
errors.Is(err, ErrNonUniqueEmail) ||
errors.Is(err, ErrNonUniqueWebsite) ||
errors.Is(err, ErrNonUniqueBrandName) ||
+ errors.Is(err, ErrNonUniqueInvoiceNumber) ||
errors.Is(err, ErrNonUniqueBrandItem) {
return http.StatusConflict
}
diff --git a/invoice/controller.go b/invoice/controller.go
new file mode 100644
index 0000000..efcaa40
--- /dev/null
+++ b/invoice/controller.go
@@ -0,0 +1,153 @@
+/* openbills - Server for web based Libre Billing Software
+ * Copyright (C) 2023 Vidhu Kant Sharma <vidhukant@vidhukant.com>
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <https://www.gnu.org/licenses/>.
+ */
+
+package invoice
+
+import (
+ e "vidhukant.com/openbills/errors"
+ "github.com/gin-gonic/gin"
+ "net/http"
+ "strconv"
+)
+
+func handleGetSingleInvoice (ctx *gin.Context) {
+ id, err := strconv.ParseUint(ctx.Param("id"), 10, 64)
+ if err != nil {
+ ctx.Error(e.ErrInvalidID)
+ return
+ }
+
+ uId, ok := ctx.Get("UserID")
+ if !ok {
+ ctx.Error(e.ErrUnauthorized)
+ ctx.Abort()
+ return
+ }
+
+ userId := uId.(uint)
+
+ var invoice Invoice
+
+ err = getInvoice(&invoice, uint(id))
+ if err != nil {
+ ctx.Error(err)
+ ctx.Abort()
+ return
+ }
+
+ if invoice.UserID != userId {
+ ctx.Error(e.ErrForbidden)
+ ctx.Abort()
+ return
+ }
+
+ ctx.JSON(http.StatusOK, gin.H{
+ "message": "success",
+ "data": invoice,
+ })
+}
+
+func handleGetInvoices (ctx *gin.Context) {
+ var invoices []Invoice
+
+ uId, ok := ctx.Get("UserID")
+ if !ok {
+ ctx.Error(e.ErrUnauthorized)
+ ctx.Abort()
+ return
+ }
+
+ userId := uId.(uint)
+
+ err := getInvoices(&invoices, userId)
+ if err != nil {
+ ctx.Error(err)
+ ctx.Abort()
+ return
+ }
+
+ ctx.JSON(http.StatusOK, gin.H{
+ "message": "success",
+ "data": invoices,
+ })
+}
+
+func handleSaveInvoice (ctx *gin.Context) {
+ var invoice Invoice
+ ctx.Bind(&invoice)
+
+ uId, ok := ctx.Get("UserID")
+ if !ok {
+ ctx.Error(e.ErrUnauthorized)
+ ctx.Abort()
+ return
+ }
+
+ userId := uId.(uint)
+ invoice.UserID = userId
+
+ err := invoice.upsert()
+ if err != nil {
+ ctx.Error(err)
+ ctx.Abort()
+ return
+ }
+
+ ctx.JSON(http.StatusOK, gin.H{
+ "message": "success",
+ "data": invoice,
+ })
+}
+
+func handleDelInvoice (ctx *gin.Context) {
+ id, err := strconv.ParseUint(ctx.Param("id"), 10, 64)
+ if err != nil {
+ ctx.Error(e.ErrInvalidID)
+ return
+ }
+
+ var invoice Invoice
+ invoice.ID = uint(id)
+
+ uId, ok := ctx.Get("UserID")
+ if !ok {
+ ctx.Error(e.ErrUnauthorized)
+ ctx.Abort()
+ return
+ }
+
+ userId := uId.(uint)
+ invoice.UserID = userId
+
+ err = checkInvoiceOwnership(invoice.ID, invoice.UserID)
+ if err != nil {
+ ctx.Error(err)
+ ctx.Abort()
+ return
+ }
+
+ err = invoice.del()
+ if err != nil {
+ ctx.Error(err)
+ ctx.Abort()
+ return
+ }
+
+ ctx.JSON(http.StatusOK, gin.H{
+ "message": "success",
+ })
+}
diff --git a/invoice/hooks.go b/invoice/hooks.go
new file mode 100644
index 0000000..3d933fb
--- /dev/null
+++ b/invoice/hooks.go
@@ -0,0 +1,57 @@
+/* openbills - Server for web based Libre Billing Software
+ * Copyright (C) 2023 Vidhu Kant Sharma <vidhukant@vidhukant.com>
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <https://www.gnu.org/licenses/>.
+ */
+
+package invoice
+
+import (
+ "gorm.io/gorm"
+ "vidhukant.com/openbills/errors"
+)
+
+func (i *Invoice) BeforeSave(tx *gorm.DB) error {
+ var err error
+
+ err = i.validate()
+ if err != nil {
+ return err
+ }
+
+ return nil
+}
+
+func (i *Invoice) BeforeDelete(tx *gorm.DB) error {
+ // if ID is 0, invoice won't be deleted
+ if i.ID == 0 {
+ return errors.ErrNoWhereCondition
+ }
+
+ var err error
+
+ // delete billing address
+ err = db.Where("invoice_id = ?", i.ID).Delete(&InvoiceBillingAddress{}).Error
+ if err != nil {
+ return err
+ }
+
+ // delete shipping address
+ err = db.Where("invoice_id = ?", i.ID).Delete(&InvoiceShippingAddress{}).Error
+ if err != nil {
+ return err
+ }
+
+ return nil
+}
diff --git a/invoice/invoice.go b/invoice/invoice.go
new file mode 100644
index 0000000..0f4a601
--- /dev/null
+++ b/invoice/invoice.go
@@ -0,0 +1,61 @@
+/* openbills - Server for web based Libre Billing Software
+ * Copyright (C) 2023 Vidhu Kant Sharma <vidhukant@vidhukant.com>
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <https://www.gnu.org/licenses/>.
+ */
+
+package invoice
+
+import (
+ "gorm.io/gorm"
+ d "vidhukant.com/openbills/db"
+ "vidhukant.com/openbills/user"
+ c "vidhukant.com/openbills/customer"
+ "time"
+)
+
+var db *gorm.DB
+func init() {
+ db = d.DB
+
+ db.AutoMigrate(&Invoice{}, &InvoiceBillingAddress{}, &InvoiceShippingAddress{})
+}
+
+type InvoiceBillingAddress struct {
+ gorm.Model
+ c.Address
+ InvoiceID uint
+}
+
+type InvoiceShippingAddress struct {
+ gorm.Model
+ c.Address
+ InvoiceID uint
+}
+
+type Invoice struct {
+ gorm.Model
+ UserID uint `json:"-"`
+ User user.User `json:"-"`
+ InvoiceDate time.Time
+ InvoiceNumber uint
+ BillingAddress InvoiceBillingAddress
+ ShippingAddress InvoiceShippingAddress
+ Draft bool
+
+ // Transporter Transporter
+ // DueDate string
+ // TransactionID string
+ // Client idk
+}
diff --git a/invoice/router.go b/invoice/router.go
new file mode 100644
index 0000000..5536466
--- /dev/null
+++ b/invoice/router.go
@@ -0,0 +1,32 @@
+/* openbills - Server for web based Libre Billing Software
+ * Copyright (C) 2023 Vidhu Kant Sharma <vidhukant@vidhukant.com>
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <https://www.gnu.org/licenses/>.
+ */
+
+package invoice
+
+import (
+ "github.com/gin-gonic/gin"
+)
+
+func Routes(route *gin.RouterGroup) {
+ g := route.Group("/invoice")
+ {
+ g.GET("/", handleGetInvoices)
+ g.GET("/:id", handleGetSingleInvoice)
+ g.POST("/", handleSaveInvoice)
+ g.DELETE("/:id", handleDelInvoice)
+ }
+}
diff --git a/invoice/service.go b/invoice/service.go
new file mode 100644
index 0000000..6b59949
--- /dev/null
+++ b/invoice/service.go
@@ -0,0 +1,74 @@
+/* openbills - Server for web based Libre Billing Software
+ * Copyright (C) 2023 Vidhu Kant Sharma <vidhukant@vidhukant.com>
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <https://www.gnu.org/licenses/>.
+ */
+
+package invoice
+
+import (
+ e "vidhukant.com/openbills/errors"
+)
+
+func getInvoice(invoice *Invoice, id uint) error {
+ res := db.Preload("BillingAddress").Preload("ShippingAddress").Find(&invoice, id)
+
+ // TODO: handle potential errors
+ if res.Error != nil {
+ return res.Error
+ }
+
+ if res.RowsAffected == 0 {
+ return e.ErrNotFound
+ }
+
+ return nil
+}
+
+func getInvoices(invoices *[]Invoice, userId uint) error {
+ res := db.Where("user_id = ?", userId).Find(&invoices)
+
+ // TODO: handle potential errors
+ if res.Error != nil {
+ return res.Error
+ }
+
+ if res.RowsAffected == 0 {
+ return e.ErrEmptyResponse
+ }
+
+ return nil
+}
+
+func (i *Invoice) upsert() error {
+ res := db.Save(i)
+ // TODO: handle potential errors
+ return res.Error
+}
+
+func (i *Invoice) del() error {
+ res := db.Where("id = ? and user_id = ?", i.ID, i.UserID).Delete(i)
+
+ // TODO: handle potential errors
+ if res.Error != nil {
+ return res.Error
+ }
+
+ // returns 404 if either row doesn't exist or if the user doesn't own it
+ if res.RowsAffected == 0 {
+ return e.ErrNotFound
+ }
+
+ return nil
+}
diff --git a/invoice/validators.go b/invoice/validators.go
new file mode 100644
index 0000000..645bdff
--- /dev/null
+++ b/invoice/validators.go
@@ -0,0 +1,71 @@
+/* openbills - Server for web based Libre Billing Software
+ * Copyright (C) 2023 Vidhu Kant Sharma <vidhukant@vidhukant.com>
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <https://www.gnu.org/licenses/>.
+ */
+
+package invoice
+
+import (
+ //"regexp"
+ //"strings"
+ //"net/mail"
+ //"net/url"
+ "vidhukant.com/openbills/errors"
+ //e "errors"
+)
+
+func (i *Invoice) validate() error {
+ var count int64
+ err := db.Model(&Invoice{}).
+ Where("user_id = ? and invoice_number = ?", i.UserID, i.InvoiceNumber).
+ Count(&count).
+ Error
+
+ if err != nil {
+ return err
+ }
+
+ if count > 0 {
+ return errors.ErrNonUniqueInvoiceNumber
+ }
+
+ return nil
+}
+
+func checkInvoiceOwnership(invoiceId, userId uint) error {
+ var invoice Invoice
+ err := db.
+ Select("id", "user_id").
+ Where("id = ?", invoiceId).
+ Find(&invoice).
+ Error
+
+ // TODO: handle potential errors
+ if err != nil {
+ return err
+ }
+
+ // invoice doesn't exist
+ if invoice.ID == 0 {
+ return errors.ErrNotFound
+ }
+
+ // user doesn't own this invoice
+ if invoice.UserID != userId {
+ return errors.ErrForbidden
+ }
+
+ return nil
+}
diff --git a/main.go b/main.go
index 0b5c59a..91b2128 100644
--- a/main.go
+++ b/main.go
@@ -30,6 +30,7 @@ import (
"vidhukant.com/openbills/auth"
"vidhukant.com/openbills/customer"
"vidhukant.com/openbills/item"
+ "vidhukant.com/openbills/invoice"
"github.com/gin-gonic/gin"
"github.com/spf13/viper"
@@ -37,7 +38,7 @@ import (
"log"
)
-const OPENBILLS_VERSION = "v0.0.10"
+const OPENBILLS_VERSION = "v0.0.11"
func init() {
if viper.GetBool("production_mode") {
@@ -63,6 +64,7 @@ func main() {
{
customer.Routes(protected)
item.Routes(protected)
+ invoice.Routes(protected)
}
log.Printf("\x1b[46m\x1b[30m[info]\x1b[0m Running OpenBills Server %s\n", OPENBILLS_VERSION)