Minimum steps
- https://github.com/AmazeeLabs/silverback-template >
Use this template
>Create a new repository
- In the newly created repo
- Settings > Manage access > Collaborators and teams
- add
Tech Team
withAdmin
role - remove yourself
- add
- Settings > General > Pull Requests
- Disable
Allow merge commits
- Enable
Automatically delete head branches
- Disable
- Settings > Manage access > Collaborators and teams
- Clone the newly create repo
- Run
pnpx @amazeelabs/mzx run INIT.md
from the project root - Answer its questions
- Review the changes in the repo
- Commit and push
Other steps
- Create a new Lagoon project
- Create a new Netlify project
- Check the Environment overrides section below
- Check the Statistics and Estimations section below
- Check the Choose a CMS section below
- Create
dev
andprod
branches (and optionallystage
) fromrelease
There are Github workflows that can connect to the
Amazeelabs Dashboard to log complexity
statistics and retrieve automatic estimations. To use that, provide a
JIRA_PROJECT_ID
environment variable in the Github repository variables.
The project should then show up on Estimator page.
The template comes with Drupal and Decap CMS enabled by default. To disable either (or both), follow these two steps:
- Remove the dependencies to
@custom/cms
/@custom/decap
fromapps/website/package.json
- Remove the
@custom/cms
/@custom/decap
plugins fromapps/website/gatsby-config.mjs
Branch name | Connected environment | Purpose | |
---|---|---|---|
Pre Go Live | Post Go Live | ||
release |
(none) | Contains everything that is approved for PROD deployment | |
prod |
PROD | The production environment | |
dev |
DEV | Sandbox/playground, no client data, anyone can merge anything | Sandbox/playground, with client data, main testing environment |
stage |
STAGE | Second sandbox with client data and automated merge from dev to stage , regular data sync |
Second sandbox for bigger features that need a clean set up or would prevent other normal tasks from being performed while working on it. With client data. No automated merges from dev . |
lagoon-* |
(same as branch name) | Can be created for big, long-term feature developments. Use wisely as it creates additional costs. |
- Create a new branch from
release
and commit your work in - Create a PR against
release
- Merge your branch to
dev
for testing - Testing feedback is committed to the branch and merged back to
dev
for retesting - When the PR is approved and Jira ticket gets to the Deploy state, the branch
is merged to
release
- Please note, this does not trigger an actual PROD deployment
- PROD deployment can be done by merging
release
branch intoprod
Tip: The easiest way to set up a working environment for the project is devbox.
Install dependencies and prepare packages:
pnpm i
pnpm turbo:prep
Tip: Run pnpm turbo:prep:force
after switching branches to avoid issues.
Navigate to an app/package folder and run pnpm dev
.
When working on integration tasks, it may be required to re-run
pnpm turbo:prep
from the repo root.
How it works in general
Turborepo allows to cache results for scripts from package.json
files.
Minimal example:
- The
build
script compiles files fromsrc
folder and puts the result intodist
folder - We can setup a Turborepo pipeline
"build": { "inputs": ["src/**"], "outputs": ["dist/**"] }
- On the first
turbo build
run Turborepo will- calculate hashes for files from
src
folder, and save them into cache, - save
dist
folder into cache.
- calculate hashes for files from
- On the second
turbo build
run Turborepo will comparesrc
hashes with cache. If hashes do match, it will restoredist
folder from the cache without running thebuild
script.
More in docs: https://turbo.build/repo/docs/core-concepts/caching
Turborepo setup: local vs CI
Locally, Turborepo stores caches under node_modules/.cache/turbo
folder.
In CI, the caches are saved in Github artifacts.
Debug Turborepo issues in CI
It can happen that some script fails in CI because of a misconfigured Turborepo pipeline. The following can be used in order to debug this locally:
-
Setup https://github.com/ducktors/turborepo-remote-cache locally
-
Run
turborepo-remote-cache
withTURBO_TOKEN=local pnpm dev
-
Run tests in the target repo with
# Clean the repo rm -rf node_modules && git clean -dxff -e '/.turbo' -e '_local' -e '/.idea' && find . -type d -empty -delete && \ # Install dependencies pnpm i && \ # Run tests with # - local Turborepo server # - Turborepo debug info # - simulated CI TURBO_API=http://0.0.0.0:3000 TURBO_TOKEN=local TURBO_TEAM=local TURBO_RUN_SUMMARY=true CI=true pnpm turbo:test
Running pnpm turbo:prep
works conditionally for Drupal. If database exists, it
clears Drupal cache. Otherwise, it re-installs Drupal completely.
If you wish Drupal to be re-installed, run pnpm turbo:prep:force
.
The application is tailored to run locally out of the box. In a production or hosting environment, you will need to override some of the environment variables.
- DRUPAL_HASH_SALT (lagoon): Drupal's hash salt. Should be different per environment for security reasons.
- PUBLISHER_URL (lagoon): If publisher is set to a custom domain, this variable has to be defined.
- NETLIFY: To publish the project to netlify, provide the following
environment variables:
- NETLIFY_SITE_ID (lagoon): The ID of the netlify project the
- NETLIFY_AUTH_TOKEN (lagoon): The auth token for the netlify project.
- NETLIFY_URL (lagoon): The URL of the netlify project.
- NETLIFY_STORYBOOK_ID (github): If this is set, the UI packages storybook build will be published to netlify.
- CLOUDINARY: To use cloudinary for image processing, provide the following
environment variables:
- CLOUDINARY_CLOUDNAME (lagoon): The cloud name of the cloudinary project.
- CLOUDINARY_API_KEY (lagoon): The API key of the cloudinary project.
- CLOUDINARY_API_SECRET (lagoon): The API secret of the cloudinary project.
- DRUPAL_INTERNAL_URL (lagoon): The internal URL of the Drupal instance. This is used for the GraphQL build queries.
- DRUPAL_EXTERNAL_URL (lagoon): The external URL of the Drupal instance. This is used for the GraphQL client queries.
On lagoon for example, this should happen in .lagoon.env
files, or directly as
lagoon runtime configuration.
lagoon add variable -p [project name] -e dev -N NETLIFY_SITE_ID -V [netlify site id]
Publisher can require to authenticate with Drupal based on OAuth2. It is only used on Lagoon environments.
How it works
Per environment, keys are gitignored and are auto-generated via a Lagoon post-rollout task.
To generate keys manually
via Drush: cd in the cms directory then
drush simple-oauth:generate-keys ./keys
or via the UI
- Go to
/admin/config/people/simple_oauth
- Click on "Generate keys", the directory should be set to
./sites/default/files/private/keys
Per environment, Consumers are content entities.
- Go to
/admin/config/services/consumer
- Create a Consumer
- Label:
Publisher
- Client ID:
publisher
- Secret: a random string
- Redirect URI:
[publisher-url]/oauth/callback
- Label:
- Optional: the default Consumer can be safely deleted
- Create a Consumer
Troubleshooting:
- make sure that the
DRUPAL_HASH_SALT
environment variable is >= 32 chars. - if enabled on local development, use
127.0.0.1:8888
for the cms and127.0.0.1:8000
for Publisher
Edit website environment variables
PUBLISHER_SKIP_AUTHENTICATION=false
PUBLISHER_OAUTH2_CLIENT_SECRET="[secret used in the Drupal Consumer]"
PUBLISHER_OAUTH2_SESSION_SECRET="[another random string]"
Optional: add this permission to relevant roles.
How to disable it
In website .lagoon.env
set PUBLISHER_SKIP_AUTHENTICATION=true
If a CHROMATIC_PROJECT_TOKEN
environment variable is set, the Storybook build
will be published to Chromatic. Additionally
setting the NETLIFY_STORYBOOK_ID
environment variable will deploy storybook to
netlify, which provides less features but is easier to access.
These are images that are part of the design and are therefore not uploaded by
the user. They have to be put into the static/public
directory which is also
shared with the website application (Gatsby). Examples are logos, icons, etc. In
a component they should be rendered with a regular <img>
tag and the src
relative to the static/public
directory.
<img src="/logo.svg" alt="Logo" />
These are images that are uploaded by the user, or on some other way injected
from the outside. In production, images are handled by Cloudinary. In
development, basic cropping is simulated in the browser. The location to store
these images is the static/stories
directory, which is used for Storybook
only.
In the GraphQL schema, these images are represented by the ImageSource
type.
If the component requires ones of these, one should use the image
helper from
src/helpers/image
, which produces exactly this type. The image itself can be
imported from the static/stories
directory using the @stories/
alias. The
import is handled by
vite-imagetools
and has to end with as=metadata
. It is also possible to apply transformations.
import Teaser from './Teaser';
import TeaserImage from '@stories/teaser.jpg?as=metadata';
export const WithImage = {
args: {
title: 'Lorem ipsum dolor sit amet',
image: image(TeaserImage),
},
} satisfied StoryObj<typeof Teaser>
In this case, the image will retain its intrinsinc dimensions. To simulate
scaling, pass a width
property.
import Teaser from './Teaser';
import TeaserImage from '@stories/teaser.jpg?as=metadata';
export const WithImage = {
args: {
title: 'Lorem ipsum dolor sit amet',
image: image(TeaserImage, { width: 400 }),
},
} satisfied StoryObj<typeof Teaser>
The image will retain its aspect ratio. To actually crop the image, also add a
height
.
import Teaser from './Teaser';
import TeaserImage from '@stories/teaser.jpg?as=metadata';
export const WithImage = {
args: {
title: 'Lorem ipsum dolor sit amet',
image: image(TeaserImage, { width: 400, height: 300 }),
},
} satisfied StoryObj<typeof Teaser>
In GraphQL fragments, it is possible to request responsive image sources.
fragment Teaser on Page {
title
image(width: 400, height: 300, sizes: [[1200, 800]])
}
The output of image
is also of type ImageSource
. To simulate this in
Storybook, add the same sizes
property to the image
helper.
export const WithImage = {
args: {
title: 'Lorem ipsum dolor sit amet',
image: image(TeaserImage, { width: 400, height: 300, sizes: [[1200, 800]]}),
},
} satisfied StoryObj<typeof Teaser>
IMPORTANT: If embedded this way, these images will not be visible in Storybook immediately. Instead, a placeholder that indicates the loaded images dimensions will be shown.
An approximation of the image that would be delivered by Cloudinary is embedded when:
- On "demo" storybook builds deployed to netlify
- In Gatsby when built on Lagoon.
These Cloudinary approximations are not real images and will fail integration tests. Therefore they are not used in regular development and testing scenarios.
The template includes a Netlify Edge Function
(apps/website/netlify/edge-functions/strangler.ts
) that allows to proxy
unknown requests selectively to other systems. This can be used to replace only
specific pages of a legacy system or incorporate existing business logic.
Refer to the
Strangler Pattern
blog post if you wonder where the name comes from, to
Edge functions documentation
for technical details and to strangler.ts
for how to add new legacy systems.