Skip to content

Commit

Permalink
feat: add online-boutique example (#3382)
Browse files Browse the repository at this point in the history
Moves the online-boutique example out of the examples repo and get it
compiling. I have not looked at the web or mobile directories, they have
just been copied as-is.
  • Loading branch information
stuartwdouglas authored Nov 13, 2024
1 parent 6b8f7f1 commit 666085c
Show file tree
Hide file tree
Showing 189 changed files with 9,825 additions and 0 deletions.
13 changes: 13 additions & 0 deletions examples/online-boutique/Justfile
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
# Start the FTL backend with hot-reloading
dev:
ftl dev --recreate --allow-origins '*'

gen-mobile:
ftl schema generate mobile/templates/ online-boutique/mobile/lib/api --watch=5s

gen-web:
ftl schema generate web/templates/ online-boutique/web/src/api --watch=5s

# Start the example web frontend for online-boutique
web:
cd web && npm install && npm run dev
3 changes: 3 additions & 0 deletions examples/online-boutique/backend/README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
# Online Boutique MicroServices Demo

Based on the [GCP original](https://github.com/GoogleCloudPlatform/microservices-demo).
13 changes: 13 additions & 0 deletions examples/online-boutique/backend/common/database.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
package common

import "encoding/json"

// LoadDatabase loads a "database" from JSON.
func LoadDatabase[T any](data []byte) T {
var out T
err := json.Unmarshal(data, &out)
if err != nil {
panic(err)
}
return out
}
5 changes: 5 additions & 0 deletions examples/online-boutique/backend/common/go.mod
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
module github.com/TBD54566975/ftl/examples/online-boutique/backend/common

go 1.23.0

replace github.com/TBD54566975/ftl => ../../../..
Empty file.
133 changes: 133 additions & 0 deletions examples/online-boutique/backend/common/money/money.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,133 @@
// Copyright 2022 Google LLC
//
// 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 money

import (
"errors"

"ftl/currency"
)

const (
nanosMin = -999999999
nanosMax = +999999999
nanosMod = 1000000000
)

var (
ErrInvalidValue = errors.New("one of the specified money values is invalid")
ErrMismatchingCurrency = errors.New("mismatching currency codes")
)

// IsValid checks if specified value has a valid units/nanos signs and ranges.
func IsValid(m currency.Money) bool {
return signMatches(m) && validNanos(m.Nanos)
}

func signMatches(m currency.Money) bool {
return m.Nanos == 0 || m.Units == 0 || (m.Nanos < 0) == (m.Units < 0)
}

func validNanos(nanos int) bool { return nanosMin <= nanos && nanos <= nanosMax }

// IsZero returns true if the specified money value is equal to zero.
func IsZero(m currency.Money) bool { return m.Units == 0 && m.Nanos == 0 }

// IsPositive returns true if the specified money value is valid and is
// positive.
func IsPositive(m currency.Money) bool {
return IsValid(m) && m.Units > 0 || (m.Units == 0 && m.Nanos > 0)
}

// IsNegative returns true if the specified money value is valid and is
// negative.
func IsNegative(m currency.Money) bool {
return IsValid(m) && m.Units < 0 || (m.Units == 0 && m.Nanos < 0)
}

// AreSameCurrency returns true if values l and r have a currency code and
// they are the same values.
func AreSameCurrency(l, r currency.Money) bool {
return l.CurrencyCode == r.CurrencyCode && l.CurrencyCode != ""
}

// AreEquals returns true if values l and r are the equal, including the
//
// This does not check validity of the provided values.
func AreEquals(l, r currency.Money) bool {
return l.CurrencyCode == r.CurrencyCode &&
l.Units == r.Units && l.Nanos == r.Nanos
}

// Negate returns the same amount with the sign negated.
func Negate(m currency.Money) currency.Money {
return currency.Money{
Units: -m.Units,
Nanos: -m.Nanos,
CurrencyCode: m.CurrencyCode}
}

// Must panics if the given error is not nil. This can be used with other
// functions like: "m := Must(Sum(a,b))".
func Must(v currency.Money, err error) currency.Money {
if err != nil {
panic(err)
}
return v
}

// Sum adds two values. Returns an error if one of the values are invalid or
// currency codes are not matching (unless currency code is unspecified for
// both).
func Sum(l, r currency.Money) (currency.Money, error) {
if !IsValid(l) || !IsValid(r) {
return currency.Money{}, ErrInvalidValue
} else if l.CurrencyCode != r.CurrencyCode {
return currency.Money{}, ErrMismatchingCurrency
}
units := l.Units + r.Units
nanos := l.Nanos + r.Nanos

if (units == 0 && nanos == 0) || (units > 0 && nanos >= 0) || (units < 0 && nanos <= 0) {
// same sign <units, nanos>
units += nanos / nanosMod
nanos = nanos % nanosMod
} else {
// different sign. nanos guaranteed to not to go over the limit
if units > 0 {
units--
nanos += nanosMod
} else {
units++
nanos -= nanosMod
}
}

return currency.Money{
Units: units,
Nanos: nanos,
CurrencyCode: l.CurrencyCode}, nil
}

// MultiplySlow is a slow multiplication operation done through adding the value
// to itself n-1 times.
func MultiplySlow(m currency.Money, n uint32) currency.Money {
out := m
for n > 1 {
out = Must(Sum(out, m))
n--
}
return out
}
Loading

0 comments on commit 666085c

Please sign in to comment.