From 1ad845b3fa5228e3bad6dc86244cbdf8bbfac18d Mon Sep 17 00:00:00 2001 From: Keir Finlow-Bates Date: Sun, 13 Aug 2023 12:17:17 +0300 Subject: [PATCH] First commit --- .editorconfig | 10 + .env.defaults | 19 + .env.example | 4 + .gitignore | 148 +- README.md | 123 +- api/db/schema.prisma | 18 + api/jest.config.js | 8 + api/package.json | 9 + api/server.config.js | 52 + .../requireAuth/requireAuth.test.ts | 18 + api/src/directives/requireAuth/requireAuth.ts | 27 + api/src/directives/skipAuth/skipAuth.test.ts | 10 + api/src/directives/skipAuth/skipAuth.ts | 16 + api/src/functions/graphql.ts | 19 + api/src/graphql/.keep | 0 api/src/lib/auth.ts | 25 + api/src/lib/db.ts | 21 + api/src/lib/logger.ts | 17 + api/src/services/.keep | 0 api/tsconfig.json | 35 + colorbar.png | Bin 0 -> 965 bytes colorbarGenerator.js | 22 + graphql.config.js | 5 + jest.config.js | 8 + package.json | 35 + prettier.config.js | 18 + production.sh | 7 + redwood.toml | 22 + scripts/.keep | 0 scripts/seed.ts | 63 + scripts/tsconfig.json | 42 + web/config/chakra.config.js | 4 + web/config/storybook.preview.js | 23 + web/config/webpack.config.js | 40 + web/jest.config.js | 8 + web/package.json | 28 + web/public/README.md | 35 + web/public/favicon.png | Bin 0 -> 1681 bytes web/public/img/base.png | Bin 0 -> 3864 bytes web/public/img/bottom.png | Bin 0 -> 5650 bytes web/public/img/colorbar.png | Bin 0 -> 329 bytes web/public/img/middle.png | Bin 0 -> 3904 bytes web/public/img/snowglobs.png | Bin 0 -> 8825 bytes web/public/img/snowglobs.xcf | Bin 0 -> 40011 bytes web/public/img/top.png | Bin 0 -> 5696 bytes web/public/robots.txt | 2 + web/src/App.tsx | 27 + web/src/Routes.tsx | 21 + web/src/components/.keep | 0 web/src/entry.client.tsx | 17 + web/src/index.css | 0 web/src/index.html | 15 + web/src/layouts/.keep | 0 .../pages/FatalErrorPage/FatalErrorPage.tsx | 57 + web/src/pages/HomePage/HomePage.stories.tsx | 13 + web/src/pages/HomePage/HomePage.test.tsx | 14 + web/src/pages/HomePage/HomePage.tsx | 258 + web/src/pages/NotFoundPage/NotFoundPage.tsx | 44 + web/tsconfig.json | 39 + web/vite.config.ts | 15 + yarn.lock | 23961 ++++++++++++++++ 61 files changed, 25291 insertions(+), 131 deletions(-) create mode 100644 .editorconfig create mode 100644 .env.defaults create mode 100644 .env.example create mode 100644 api/db/schema.prisma create mode 100644 api/jest.config.js create mode 100644 api/package.json create mode 100644 api/server.config.js create mode 100644 api/src/directives/requireAuth/requireAuth.test.ts create mode 100644 api/src/directives/requireAuth/requireAuth.ts create mode 100644 api/src/directives/skipAuth/skipAuth.test.ts create mode 100644 api/src/directives/skipAuth/skipAuth.ts create mode 100644 api/src/functions/graphql.ts create mode 100644 api/src/graphql/.keep create mode 100644 api/src/lib/auth.ts create mode 100644 api/src/lib/db.ts create mode 100644 api/src/lib/logger.ts create mode 100644 api/src/services/.keep create mode 100644 api/tsconfig.json create mode 100644 colorbar.png create mode 100644 colorbarGenerator.js create mode 100644 graphql.config.js create mode 100644 jest.config.js create mode 100644 package.json create mode 100644 prettier.config.js create mode 100755 production.sh create mode 100644 redwood.toml create mode 100644 scripts/.keep create mode 100644 scripts/seed.ts create mode 100644 scripts/tsconfig.json create mode 100644 web/config/chakra.config.js create mode 100644 web/config/storybook.preview.js create mode 100644 web/config/webpack.config.js create mode 100644 web/jest.config.js create mode 100644 web/package.json create mode 100644 web/public/README.md create mode 100644 web/public/favicon.png create mode 100644 web/public/img/base.png create mode 100644 web/public/img/bottom.png create mode 100644 web/public/img/colorbar.png create mode 100644 web/public/img/middle.png create mode 100644 web/public/img/snowglobs.png create mode 100644 web/public/img/snowglobs.xcf create mode 100644 web/public/img/top.png create mode 100644 web/public/robots.txt create mode 100644 web/src/App.tsx create mode 100644 web/src/Routes.tsx create mode 100644 web/src/components/.keep create mode 100644 web/src/entry.client.tsx create mode 100644 web/src/index.css create mode 100644 web/src/index.html create mode 100644 web/src/layouts/.keep create mode 100644 web/src/pages/FatalErrorPage/FatalErrorPage.tsx create mode 100644 web/src/pages/HomePage/HomePage.stories.tsx create mode 100644 web/src/pages/HomePage/HomePage.test.tsx create mode 100644 web/src/pages/HomePage/HomePage.tsx create mode 100644 web/src/pages/NotFoundPage/NotFoundPage.tsx create mode 100644 web/tsconfig.json create mode 100644 web/vite.config.ts create mode 100644 yarn.lock diff --git a/.editorconfig b/.editorconfig new file mode 100644 index 0000000..ae10a5c --- /dev/null +++ b/.editorconfig @@ -0,0 +1,10 @@ +# editorconfig.org +root = true + +[*] +charset = utf-8 +end_of_line = lf +indent_size = 2 +indent_style = space +insert_final_newline = true +trim_trailing_whitespace = true diff --git a/.env.defaults b/.env.defaults new file mode 100644 index 0000000..fb88fb3 --- /dev/null +++ b/.env.defaults @@ -0,0 +1,19 @@ +# These environment variables will be used by default if you do not create any +# yourself in .env. This file should be safe to check into your version control +# system. Any custom values should go in .env and .env should *not* be checked +# into version control. + +# schema.prisma defaults +DATABASE_URL=file:./dev.db + +# location of the test database for api service scenarios (defaults to ./.redwood/test.db if not set) +# TEST_DATABASE_URL=file:./.redwood/test.db + +# disables Prisma CLI update notifier +PRISMA_HIDE_UPDATE_MESSAGE=true + +# Option to override the current environment's default api-side log level +# See: https://redwoodjs.com/docs/logger for level options, defaults to "trace" otherwise. +# Most applications want "debug" or "info" during dev, "trace" when you have issues and "warn" in production. +# Ordered by how verbose they are: trace | debug | info | warn | error | silent +# LOG_LEVEL=debug diff --git a/.env.example b/.env.example new file mode 100644 index 0000000..2a2de6c --- /dev/null +++ b/.env.example @@ -0,0 +1,4 @@ +# DATABASE_URL=file:./dev.db +# TEST_DATABASE_URL=file:./.redwood/test.db +# PRISMA_HIDE_UPDATE_MESSAGE=true +# LOG_LEVEL=trace diff --git a/.gitignore b/.gitignore index ceaea36..9b81495 100644 --- a/.gitignore +++ b/.gitignore @@ -1,132 +1,22 @@ -# ---> Node -# Logs -logs -*.log -npm-debug.log* -yarn-debug.log* -yarn-error.log* -lerna-debug.log* -.pnpm-debug.log* - -# Diagnostic reports (https://nodejs.org/api/report.html) -report.[0-9]*.[0-9]*.[0-9]*.[0-9]*.json - -# Runtime data -pids -*.pid -*.seed -*.pid.lock - -# Directory for instrumented libs generated by jscoverage/JSCover -lib-cov - -# Coverage directory used by tools like istanbul -coverage -*.lcov - -# nyc test coverage -.nyc_output - -# Grunt intermediate storage (https://gruntjs.com/creating-plugins#storing-task-files) -.grunt - -# Bower dependency directory (https://bower.io/) -bower_components - -# node-waf configuration -.lock-wscript - -# Compiled binary addons (https://nodejs.org/api/addons.html) -build/Release - -# Dependency directories -node_modules/ -jspm_packages/ - -# Snowpack dependency directory (https://snowpack.dev/) -web_modules/ - -# TypeScript cache -*.tsbuildinfo - -# Optional npm cache directory -.npm - -# Optional eslint cache -.eslintcache - -# Optional stylelint cache -.stylelintcache - -# Microbundle cache -.rpt2_cache/ -.rts2_cache_cjs/ -.rts2_cache_es/ -.rts2_cache_umd/ - -# Optional REPL history -.node_repl_history - -# Output of 'npm pack' -*.tgz - -# Yarn Integrity file -.yarn-integrity - -# dotenv environment variable files +.idea +.DS_Store .env -.env.development.local -.env.test.local -.env.production.local -.env.local - -# parcel-bundler cache (https://parceljs.org/) -.cache -.parcel-cache - -# Next.js build output -.next -out - -# Nuxt.js build / generate output -.nuxt +.netlify +.redwood/* +!.redwood/README.md +dev.db* dist - -# Gatsby files -.cache/ -# Comment in the public line in if your project uses Gatsby and not Next.js -# https://nextjs.org/blog/next-9-1#public-directory-support -# public - -# vuepress build output -.vuepress/dist - -# vuepress v2.x temp and cache directory -.temp -.cache - -# Docusaurus cache and generated files -.docusaurus - -# Serverless directories -.serverless/ - -# FuseBox cache -.fusebox/ - -# DynamoDB Local files -.dynamodb/ - -# TernJS port file -.tern-port - -# Stores VSCode versions used for testing VSCode extensions -.vscode-test - -# yarn v2 -.yarn/cache -.yarn/unplugged -.yarn/build-state.yml -.yarn/install-state.gz +dist-babel +node_modules +yarn-error.log +web/public/mockServiceWorker.js +web/types/graphql.d.ts +api/types/graphql.d.ts +api/src/lib/generateGraphiQLHeader.* .pnp.* - +.yarn/* +!.yarn/patches +!.yarn/plugins +!.yarn/releases +!.yarn/sdks +!.yarn/versions diff --git a/README.md b/README.md index f703ecc..08f5886 100644 --- a/README.md +++ b/README.md @@ -1,3 +1,122 @@ -# snowglobs +# README -Generate and download a snowglob \ No newline at end of file +Welcome to [RedwoodJS](https://redwoodjs.com)! + +> **Prerequisites** +> +> - Redwood requires [Node.js](https://nodejs.org/en/) (=18.x) and [Yarn](https://yarnpkg.com/) (>=1.15) +> - Are you on Windows? For best results, follow our [Windows development setup](https://redwoodjs.com/docs/how-to/windows-development-setup) guide + +Start by installing dependencies: + +``` +yarn install +``` + +Then start the development server: + +``` +yarn redwood dev +``` + +Your browser should automatically open to [http://localhost:8910](http://localhost:8910) where you'll see the Welcome Page, which links out to many great resources. + +> **The Redwood CLI** +> +> Congratulations on running your first Redwood CLI command! From dev to deploy, the CLI is with you the whole way. And there's quite a few commands at your disposal: +> +> ``` +> yarn redwood --help +> ``` +> +> For all the details, see the [CLI reference](https://redwoodjs.com/docs/cli-commands). + +## Prisma and the database + +Redwood wouldn't be a full-stack framework without a database. It all starts with the schema. Open the [`schema.prisma`](api/db/schema.prisma) file in `api/db` and replace the `UserExample` model with the following `Post` model: + +```prisma +model Post { + id Int @id @default(autoincrement()) + title String + body String + createdAt DateTime @default(now()) +} +``` + +Redwood uses [Prisma](https://www.prisma.io/), a next-gen Node.js and TypeScript ORM, to talk to the database. Prisma's schema offers a declarative way of defining your app's data models. And Prisma [Migrate](https://www.prisma.io/migrate) uses that schema to make database migrations hassle-free: + +``` +yarn rw prisma migrate dev + +# ... + +? Enter a name for the new migration: › create posts +``` + +> `rw` is short for `redwood` + +You'll be prompted for the name of your migration. `create posts` will do. + +Now let's generate everything we need to perform all the CRUD (Create, Retrieve, Update, Delete) actions on our `Post` model: + +``` +yarn redwood generate scaffold post +``` + +Navigate to [http://localhost:8910/posts/new](http://localhost:8910/posts/new), fill in the title and body, and click "Save". + +Did we just create a post in the database? Yup! With `yarn rw generate scaffold `, Redwood created all the pages, components, and services necessary to perform all CRUD actions on our posts table. + +## Frontend first with Storybook + +Don't know what your data models look like? That's more than ok—Redwood integrates Storybook so that you can work on design without worrying about data. Mockup, build, and verify your React components, even in complete isolation from the backend: + +``` +yarn rw storybook +``` + +Seeing "Couldn't find any stories"? That's because you need a `*.stories.{tsx,jsx}` file. The Redwood CLI makes getting one easy enough—try generating a [Cell](https://redwoodjs.com/docs/cells), Redwood's data-fetching abstraction: + +``` +yarn rw generate cell examplePosts +``` + +The Storybook server should hot reload and now you'll have four stories to work with. They'll probably look a little bland since there's no styling. See if the Redwood CLI's `setup ui` command has your favorite styling library: + +``` +yarn rw setup ui --help +``` + +## Testing with Jest + +It'd be hard to scale from side project to startup without a few tests. Redwood fully integrates Jest with both the front- and back-ends, and makes it easy to keep your whole app covered by generating test files with all your components and services: + +``` +yarn rw test +``` + +To make the integration even more seamless, Redwood augments Jest with database [scenarios](https://redwoodjs.com/docs/testing#scenarios) and [GraphQL mocking](https://redwoodjs.com/docs/testing#mocking-graphql-calls). + +## Ship it + +Redwood is designed for both serverless deploy targets like Netlify and Vercel and serverful deploy targets like Render and AWS: + +``` +yarn rw setup deploy --help +``` + +Don't go live without auth! Lock down your app with Redwood's built-in, database-backed authentication system ([dbAuth](https://redwoodjs.com/docs/authentication#self-hosted-auth-installation-and-setup)), or integrate with nearly a dozen third-party auth providers: + +``` +yarn rw setup auth --help +``` + +## Next Steps + +The best way to learn Redwood is by going through the comprehensive [tutorial](https://redwoodjs.com/docs/tutorial/foreword) and joining the community (via the [Discourse forum](https://community.redwoodjs.com) or the [Discord server](https://discord.gg/redwoodjs)). + +## Quick Links + +- Stay updated: read [Forum announcements](https://community.redwoodjs.com/c/announcements/5), follow us on [Twitter](https://twitter.com/redwoodjs), and subscribe to the [newsletter](https://redwoodjs.com/newsletter) +- [Learn how to contribute](https://redwoodjs.com/docs/contributing) diff --git a/api/db/schema.prisma b/api/db/schema.prisma new file mode 100644 index 0000000..3dea71a --- /dev/null +++ b/api/db/schema.prisma @@ -0,0 +1,18 @@ +datasource db { + provider = "sqlite" + url = env("DATABASE_URL") +} + +generator client { + provider = "prisma-client-js" + binaryTargets = "native" +} + +// Define your own datamodels here and run `yarn redwood prisma migrate dev` +// to create migrations for them and apply to your dev DB. +// TODO: Please remove the following example: +model UserExample { + id Int @id @default(autoincrement()) + email String @unique + name String? +} diff --git a/api/jest.config.js b/api/jest.config.js new file mode 100644 index 0000000..932fc82 --- /dev/null +++ b/api/jest.config.js @@ -0,0 +1,8 @@ +// More info at https://redwoodjs.com/docs/project-configuration-dev-test-build + +const config = { + rootDir: '../', + preset: '@redwoodjs/testing/config/jest/api', +} + +module.exports = config diff --git a/api/package.json b/api/package.json new file mode 100644 index 0000000..29e22bf --- /dev/null +++ b/api/package.json @@ -0,0 +1,9 @@ +{ + "name": "api", + "version": "0.0.0", + "private": true, + "dependencies": { + "@redwoodjs/api": "6.0.6", + "@redwoodjs/graphql-server": "6.0.6" + } +} diff --git a/api/server.config.js b/api/server.config.js new file mode 100644 index 0000000..e82b04e --- /dev/null +++ b/api/server.config.js @@ -0,0 +1,52 @@ +/** + * This file allows you to configure the Fastify Server settings + * used by the RedwoodJS dev server. + * + * It also applies when running RedwoodJS with `yarn rw serve`. + * + * For the Fastify server options that you can set, see: + * https://www.fastify.io/docs/latest/Reference/Server/#factory + * + * Examples include: logger settings, timeouts, maximum payload limits, and more. + * + * Note: This configuration does not apply in a serverless deploy. + */ + +/** @type {import('fastify').FastifyServerOptions} */ +const config = { + requestTimeout: 15_000, + logger: { + // Note: If running locally using `yarn rw serve` you may want to adjust + // the default non-development level to `info` + level: process.env.NODE_ENV === 'development' ? 'debug' : 'warn', + }, +} + +/** + * You can also register Fastify plugins and additional routes for the API and Web sides + * in the configureFastify function. + * + * This function has access to the Fastify instance and options, such as the side + * (web, api, or proxy) that is being configured and other settings like the apiRootPath + * of the functions endpoint. + * + * Note: This configuration does not apply in a serverless deploy. + */ + +/** @type {import('@redwoodjs/api-server/dist/fastify').FastifySideConfigFn} */ +const configureFastify = async (fastify, options) => { + if (options.side === 'api') { + fastify.log.trace({ custom: { options } }, 'Configuring api side') + } + + if (options.side === 'web') { + fastify.log.trace({ custom: { options } }, 'Configuring web side') + } + + return fastify +} + +module.exports = { + config, + configureFastify, +} diff --git a/api/src/directives/requireAuth/requireAuth.test.ts b/api/src/directives/requireAuth/requireAuth.test.ts new file mode 100644 index 0000000..0f01aa3 --- /dev/null +++ b/api/src/directives/requireAuth/requireAuth.test.ts @@ -0,0 +1,18 @@ +import { mockRedwoodDirective, getDirectiveName } from '@redwoodjs/testing/api' + +import requireAuth from './requireAuth' + +describe('requireAuth directive', () => { + it('declares the directive sdl as schema, with the correct name', () => { + expect(requireAuth.schema).toBeTruthy() + expect(getDirectiveName(requireAuth.schema)).toBe('requireAuth') + }) + + it('requireAuth has stub implementation. Should not throw when current user', () => { + // If you want to set values in context, pass it through e.g. + // mockRedwoodDirective(requireAuth, { context: { currentUser: { id: 1, name: 'Lebron McGretzky' } }}) + const mockExecution = mockRedwoodDirective(requireAuth, { context: {} }) + + expect(mockExecution).not.toThrowError() + }) +}) diff --git a/api/src/directives/requireAuth/requireAuth.ts b/api/src/directives/requireAuth/requireAuth.ts new file mode 100644 index 0000000..59daccb --- /dev/null +++ b/api/src/directives/requireAuth/requireAuth.ts @@ -0,0 +1,27 @@ +import gql from 'graphql-tag' + +import { + createValidatorDirective, + ValidatorDirectiveFunc, +} from '@redwoodjs/graphql-server' + +import { requireAuth as applicationRequireAuth } from 'src/lib/auth' + +export const schema = gql` + """ + Use to check whether or not a user is authenticated and is associated + with an optional set of roles. + """ + directive @requireAuth(roles: [String]) on FIELD_DEFINITION +` + +type RequireAuthValidate = ValidatorDirectiveFunc<{ roles?: string[] }> + +const validate: RequireAuthValidate = ({ directiveArgs }) => { + const { roles } = directiveArgs + applicationRequireAuth({ roles }) +} + +const requireAuth = createValidatorDirective(schema, validate) + +export default requireAuth diff --git a/api/src/directives/skipAuth/skipAuth.test.ts b/api/src/directives/skipAuth/skipAuth.test.ts new file mode 100644 index 0000000..88d99a5 --- /dev/null +++ b/api/src/directives/skipAuth/skipAuth.test.ts @@ -0,0 +1,10 @@ +import { getDirectiveName } from '@redwoodjs/testing/api' + +import skipAuth from './skipAuth' + +describe('skipAuth directive', () => { + it('declares the directive sdl as schema, with the correct name', () => { + expect(skipAuth.schema).toBeTruthy() + expect(getDirectiveName(skipAuth.schema)).toBe('skipAuth') + }) +}) diff --git a/api/src/directives/skipAuth/skipAuth.ts b/api/src/directives/skipAuth/skipAuth.ts new file mode 100644 index 0000000..e85b94a --- /dev/null +++ b/api/src/directives/skipAuth/skipAuth.ts @@ -0,0 +1,16 @@ +import gql from 'graphql-tag' + +import { createValidatorDirective } from '@redwoodjs/graphql-server' + +export const schema = gql` + """ + Use to skip authentication checks and allow public access. + """ + directive @skipAuth on FIELD_DEFINITION +` + +const skipAuth = createValidatorDirective(schema, () => { + return +}) + +export default skipAuth diff --git a/api/src/functions/graphql.ts b/api/src/functions/graphql.ts new file mode 100644 index 0000000..f395c3b --- /dev/null +++ b/api/src/functions/graphql.ts @@ -0,0 +1,19 @@ +import { createGraphQLHandler } from '@redwoodjs/graphql-server' + +import directives from 'src/directives/**/*.{js,ts}' +import sdls from 'src/graphql/**/*.sdl.{js,ts}' +import services from 'src/services/**/*.{js,ts}' + +import { db } from 'src/lib/db' +import { logger } from 'src/lib/logger' + +export const handler = createGraphQLHandler({ + loggerConfig: { logger, options: {} }, + directives, + sdls, + services, + onException: () => { + // Disconnect from your database with an unhandled exception. + db.$disconnect() + }, +}) diff --git a/api/src/graphql/.keep b/api/src/graphql/.keep new file mode 100644 index 0000000..e69de29 diff --git a/api/src/lib/auth.ts b/api/src/lib/auth.ts new file mode 100644 index 0000000..f98fe93 --- /dev/null +++ b/api/src/lib/auth.ts @@ -0,0 +1,25 @@ +/** + * Once you are ready to add authentication to your application + * you'll build out requireAuth() with real functionality. For + * now we just return `true` so that the calls in services + * have something to check against, simulating a logged + * in user that is allowed to access that service. + * + * See https://redwoodjs.com/docs/authentication for more info. + */ +export const isAuthenticated = () => { + return true +} + +export const hasRole = ({ roles }) => { + return roles !== undefined +} + +// This is used by the redwood directive +// in ./api/src/directives/requireAuth + +// Roles are passed in by the requireAuth directive if you have auth setup +// eslint-disable-next-line no-unused-vars, @typescript-eslint/no-unused-vars +export const requireAuth = ({ roles }) => { + return isAuthenticated() +} diff --git a/api/src/lib/db.ts b/api/src/lib/db.ts new file mode 100644 index 0000000..5006d00 --- /dev/null +++ b/api/src/lib/db.ts @@ -0,0 +1,21 @@ +// See https://www.prisma.io/docs/reference/tools-and-interfaces/prisma-client/constructor +// for options. + +import { PrismaClient } from '@prisma/client' + +import { emitLogLevels, handlePrismaLogging } from '@redwoodjs/api/logger' + +import { logger } from './logger' + +/* + * Instance of the Prisma Client + */ +export const db = new PrismaClient({ + log: emitLogLevels(['info', 'warn', 'error']), +}) + +handlePrismaLogging({ + db, + logger, + logLevels: ['info', 'warn', 'error'], +}) diff --git a/api/src/lib/logger.ts b/api/src/lib/logger.ts new file mode 100644 index 0000000..150a309 --- /dev/null +++ b/api/src/lib/logger.ts @@ -0,0 +1,17 @@ +import { createLogger } from '@redwoodjs/api/logger' + +/** + * Creates a logger with RedwoodLoggerOptions + * + * These extend and override default LoggerOptions, + * can define a destination like a file or other supported pino log transport stream, + * and sets whether or not to show the logger configuration settings (defaults to false) + * + * @param RedwoodLoggerOptions + * + * RedwoodLoggerOptions have + * @param {options} LoggerOptions - defines how to log, such as redaction and format + * @param {string | DestinationStream} destination - defines where to log, such as a transport stream or file + * @param {boolean} showConfig - whether to display logger configuration on initialization + */ +export const logger = createLogger({}) diff --git a/api/src/services/.keep b/api/src/services/.keep new file mode 100644 index 0000000..e69de29 diff --git a/api/tsconfig.json b/api/tsconfig.json new file mode 100644 index 0000000..65c666c --- /dev/null +++ b/api/tsconfig.json @@ -0,0 +1,35 @@ +{ + "compilerOptions": { + "noEmit": true, + "allowJs": true, + "esModuleInterop": true, + "target": "esnext", + "module": "esnext", + "moduleResolution": "node", + "skipLibCheck": false, + "baseUrl": "./", + "rootDirs": [ + "./src", + "../.redwood/types/mirror/api/src" + ], + "paths": { + "src/*": [ + "./src/*", + "../.redwood/types/mirror/api/src/*" + ], + "types/*": ["./types/*", "../types/*"], + "@redwoodjs/testing": ["../node_modules/@redwoodjs/testing/api"] + }, + "typeRoots": [ + "../node_modules/@types", + "./node_modules/@types" + ], + "types": ["jest"] + }, + "include": [ + "src", + "../.redwood/types/includes/all-*", + "../.redwood/types/includes/api-*", + "../types" + ] +} diff --git a/colorbar.png b/colorbar.png new file mode 100644 index 0000000000000000000000000000000000000000..7dbd05f4f1b4543b6420f26adf3853f1cbc027f6 GIT binary patch literal 965 zcmV;$13LVPP)|-`EJG(q25GksIa8*FXQH@f&kkq_q z8~pgrnfT9t|M%z-54gu2ZgIoJ4P{K?J4|63Qyy@Me=y?t6%|x@sBnP+E=^()-!phd zpZRapFl&S}s#m|!)oS`47m76j~B0tMUTimAY<4})O&9E0oB5!1) zDN3n(7;YlH&vhA9alT{!G`xuPF4tsdQH)aWQy9dq$VpjiYShzQc4Umj;J`^Q^rI{G zhqlJyMWlAl$V$_~Mq1A$xxA{jlImHL%XLdDX*CyQW!=n1YUWE@+8$$HbjMDHp&tYB zt;az}HcD!9sh%@-HQhx|{5{}PN3NC3E~a<;oR7n4+*`L}A*Xb`H;hXShvss|k)|(k z?X^8WY;*AtjvKt3OYL)2!~7PPkKu3`D@R(sDCaX*KKS(&e*Dv#GmBje zH|?2r{WG3#`^AHaEj)JYD!FT+{Lx!q&Hs+SYA)0r&!Bma18-cbp5t^DKk1qBUSZ9Nl71PlQN6;MStB*fe?uRGXZRkAE=?NX0uL2bQNi;ojPMUGO?fbl zDSU@XlreFG8{Fa!_jte~C51wvP { + result.write('./web/public/img/colorbar.png') +}) diff --git a/graphql.config.js b/graphql.config.js new file mode 100644 index 0000000..2da7862 --- /dev/null +++ b/graphql.config.js @@ -0,0 +1,5 @@ +const { getPaths } = require('@redwoodjs/internal') + +module.exports = { + schema: getPaths().generated.schema, +} diff --git a/jest.config.js b/jest.config.js new file mode 100644 index 0000000..c6b395c --- /dev/null +++ b/jest.config.js @@ -0,0 +1,8 @@ +// This the Redwood root jest config +// Each side, e.g. ./web/ and ./api/ has specific config that references this root +// More info at https://redwoodjs.com/docs/project-configuration-dev-test-build + +module.exports = { + rootDir: '.', + projects: ['/{*,!(node_modules)/**/}/jest.config.js'], +} diff --git a/package.json b/package.json new file mode 100644 index 0000000..b40911c --- /dev/null +++ b/package.json @@ -0,0 +1,35 @@ +{ + "private": true, + "workspaces": { + "packages": [ + "api", + "web" + ] + }, + "devDependencies": { + "@redwoodjs/core": "6.0.6" + }, + "eslintConfig": { + "extends": "@redwoodjs/eslint-config", + "root": true + }, + "engines": { + "node": "=18.x", + "yarn": ">=1.15" + }, + "prisma": { + "seed": "yarn rw exec seed" + }, + "packageManager": "yarn@3.6.1", + "dependencies": { + "browserify-zlib": "^0.2.0", + "https-browserify": "^1.0.0", + "jimp": "^0.16.2", + "js-file-download": "^0.4.12", + "node-polyfill-webpack-plugin": "^2.0.1", + "path-browserify": "^1.0.1", + "process": "^0.11.10", + "stream-browserify": "^3.0.0", + "stream-http": "^3.2.0" + } +} diff --git a/prettier.config.js b/prettier.config.js new file mode 100644 index 0000000..45058f7 --- /dev/null +++ b/prettier.config.js @@ -0,0 +1,18 @@ +// https://prettier.io/docs/en/options.html +/** @type {import('prettier').RequiredOptions} */ +module.exports = { + trailingComma: 'es5', + bracketSpacing: true, + tabWidth: 2, + semi: false, + singleQuote: true, + arrowParens: 'always', + overrides: [ + { + files: 'Routes.*', + options: { + printWidth: 999, + }, + }, + ], +} diff --git a/production.sh b/production.sh new file mode 100755 index 0000000..ec46143 --- /dev/null +++ b/production.sh @@ -0,0 +1,7 @@ +#!/bin/bash +source ~/.nvm/nvm.sh +nvm use 18 +if [ "$1" == "build" ]; then + yarn rw build +fi +yarn rw serve -p 8969 diff --git a/redwood.toml b/redwood.toml new file mode 100644 index 0000000..46bc0eb --- /dev/null +++ b/redwood.toml @@ -0,0 +1,22 @@ +# This file contains the configuration settings for your Redwood app. +# This file is also what makes your Redwood app a Redwood app. +# If you remove it and try to run `yarn rw dev`, you'll get an error. +# +# For the full list of options, see the "App Configuration: redwood.toml" doc: +# https://redwoodjs.com/docs/app-configuration-redwood-toml + +[web] + bundler = "webpack" + title = "Redwood App" + port = 8910 + apiUrl = "/.redwood/functions" # You can customize graphql and dbauth urls individually too: see https://redwoodjs.com/docs/app-configuration-redwood-toml#api-paths + includeEnvironmentVariables = [ + # Add any ENV vars that should be available to the web side to this array + # See https://redwoodjs.com/docs/environment-variables#web + ] +[api] + port = 8911 +[browser] + open = true +[notifications] + versionUpdates = ["latest"] diff --git a/scripts/.keep b/scripts/.keep new file mode 100644 index 0000000..e69de29 diff --git a/scripts/seed.ts b/scripts/seed.ts new file mode 100644 index 0000000..5797fd9 --- /dev/null +++ b/scripts/seed.ts @@ -0,0 +1,63 @@ +import type { Prisma } from '@prisma/client' +import { db } from 'api/src/lib/db' + +export default async () => { + try { + // + // Manually seed via `yarn rw prisma db seed` + // Seeds automatically with `yarn rw prisma migrate dev` and `yarn rw prisma migrate reset` + // + // Update "const data = []" to match your data model and seeding needs + // + const data: Prisma.UserExampleCreateArgs['data'][] = [ + // To try this example data with the UserExample model in schema.prisma, + // uncomment the lines below and run 'yarn rw prisma migrate dev' + // + // { name: 'alice', email: 'alice@example.com' }, + // { name: 'mark', email: 'mark@example.com' }, + // { name: 'jackie', email: 'jackie@example.com' }, + // { name: 'bob', email: 'bob@example.com' }, + ] + console.log( + "\nUsing the default './scripts/seed.{js,ts}' template\nEdit the file to add seed data\n" + ) + + // Note: if using PostgreSQL, using `createMany` to insert multiple records is much faster + // @see: https://www.prisma.io/docs/reference/api-reference/prisma-client-reference#createmany + Promise.all( + // + // Change to match your data model and seeding needs + // + data.map(async (data: Prisma.UserExampleCreateArgs['data']) => { + const record = await db.userExample.create({ data }) + console.log(record) + }) + ) + + // If using dbAuth and seeding users, you'll need to add a `hashedPassword` + // and associated `salt` to their record. Here's how to create them using + // the same algorithm that dbAuth uses internally: + // + // import { hashPassword } from '@redwoodjs/auth-dbauth-api' + // + // const users = [ + // { name: 'john', email: 'john@example.com', password: 'secret1' }, + // { name: 'jane', email: 'jane@example.com', password: 'secret2' } + // ] + // + // for (const user of users) { + // const [hashedPassword, salt] = hashPassword(user.password) + // await db.user.create({ + // data: { + // name: user.name, + // email: user.email, + // hashedPassword, + // salt + // } + // }) + // } + } catch (error) { + console.warn('Please define your seed data.') + console.error(error) + } +} diff --git a/scripts/tsconfig.json b/scripts/tsconfig.json new file mode 100644 index 0000000..babc7c4 --- /dev/null +++ b/scripts/tsconfig.json @@ -0,0 +1,42 @@ +{ + "compilerOptions": { + "noEmit": true, + "allowJs": true, + "esModuleInterop": true, + "target": "esnext", + "module": "esnext", + "moduleResolution": "node", + "baseUrl": "./", + "paths": { + "$api/*": [ + "../api/*" + ], + "api/*": [ + "../api/*" + ], + "$web/*": [ + "../web/*" + ], + "web/*": [ + "../web/*" + ], + "$web/src/*": [ + "../web/src/*", + "../.redwood/types/mirror/web/src/*" + ], + "web/src/*": [ + "../web/src/*", + "../.redwood/types/mirror/web/src/*" + ], + "types/*": ["../types/*", "../web/types/*", "../api/types/*"] + }, + "typeRoots": ["../node_modules/@types"], + "jsx": "preserve" + }, + "include": [ + ".", + "../.redwood/types/includes/all-*", + "../.redwood/types/includes/web-*", + "../types" + ] +} diff --git a/web/config/chakra.config.js b/web/config/chakra.config.js new file mode 100644 index 0000000..7477c5b --- /dev/null +++ b/web/config/chakra.config.js @@ -0,0 +1,4 @@ +// This object will be used to override Chakra-UI theme defaults. +// See https://chakra-ui.com/docs/styled-system/theming/theme for theming options +const theme = {} +export default theme diff --git a/web/config/storybook.preview.js b/web/config/storybook.preview.js new file mode 100644 index 0000000..0ce50bf --- /dev/null +++ b/web/config/storybook.preview.js @@ -0,0 +1,23 @@ +import * as React from 'react' +import { ChakraProvider, extendTheme } from '@chakra-ui/react' +import * as theme from 'config/chakra.config' +/** @type { import("@storybook/csf").GlobalTypes } */ +export const globalTypes = {} +/** + * An example, no-op storybook decorator. Use a function like this to create decorators. + * @param { import("@storybook/addons").StoryFn} StoryFn + * @param { import("@storybook/addons").StoryContext} context + * @returns StoryFn, unmodified. + */ +const _exampleDecorator = (StoryFn, _context) => { + return +} +const extendedTheme = extendTheme(theme) +const withChakra = (StoryFn) => { + return ( + + + + ) +} +export const decorators = [withChakra] diff --git a/web/config/webpack.config.js b/web/config/webpack.config.js new file mode 100644 index 0000000..bceefd2 --- /dev/null +++ b/web/config/webpack.config.js @@ -0,0 +1,40 @@ +/** @returns {import('webpack').Configuration} Webpack Configuration */ +const NodePolyfillPlugin = require('node-polyfill-webpack-plugin') +var webpack = require('webpack') + +module.exports = (config, { mode }) => { + if (mode === 'development') { + // Add dev plugin + } + + config.module.rules.push({ + resolve: { + fallback: { + fs: false, + path: require.resolve('path-browserify'), + http: require.resolve('stream-http'), + https: require.resolve('https-browserify'), + zlib: require.resolve('browserify-zlib'), + stream: require.resolve('stream-browserify'), + querystring: require.resolve('querystring-es3') + }, + }, + }) + + config.plugins.push( + // fix "process is not defined" error: + // (do "npm install process" before running the build) + new webpack.ProvidePlugin({ + process: 'process/browser.js', + Buffer: ['buffer', 'Buffer'], + }) + ) + + // Add custom rules for your project + // config.module.rules.push(YOUR_RULE) + + // Add custom plugins for your project + // config.plugins.push(YOUR_PLUGIN) + + return config +} diff --git a/web/jest.config.js b/web/jest.config.js new file mode 100644 index 0000000..0e54869 --- /dev/null +++ b/web/jest.config.js @@ -0,0 +1,8 @@ +// More info at https://redwoodjs.com/docs/project-configuration-dev-test-build + +const config = { + rootDir: '../', + preset: '@redwoodjs/testing/config/jest/web', +} + +module.exports = config diff --git a/web/package.json b/web/package.json new file mode 100644 index 0000000..3c26b8e --- /dev/null +++ b/web/package.json @@ -0,0 +1,28 @@ +{ + "name": "web", + "version": "0.0.0", + "private": true, + "browserslist": { + "development": [ + "last 1 version" + ], + "production": [ + "defaults" + ] + }, + "dependencies": { + "@chakra-ui/react": "^2", + "@emotion/react": "^11", + "@emotion/styled": "^11", + "@redwoodjs/forms": "6.0.6", + "@redwoodjs/router": "6.0.6", + "@redwoodjs/web": "6.0.6", + "framer-motion": "^9", + "prop-types": "15.8.1", + "react": "18.2.0", + "react-dom": "18.2.0" + }, + "devDependencies": { + "@redwoodjs/vite": "6.0.6" + } +} diff --git a/web/public/README.md b/web/public/README.md new file mode 100644 index 0000000..618395f --- /dev/null +++ b/web/public/README.md @@ -0,0 +1,35 @@ +# Static Assets +Use this folder to add static files directly to your app. All included files and folders will be copied directly into the `/dist` folder (created when Vite builds for production). They will also be available during development when you run `yarn rw dev`. +>Note: files will *not* hot reload while the development server is running. You'll need to manually stop/start to access file changes. + +### Example Use +A file like `favicon.png` will be copied to `/dist/favicon.png`. A folder containing a file such as `static-files/my-logo.jpg` will be copied to `/dist/static-files/my-logo.jpg`. These can be referenced in your code directly without any special handling, e.g. +``` + +``` +and +``` + alt="Logo" /> +``` + + +## Best Practices +Because assets in this folder are bypassing the javascript module system, **this folder should be used sparingly** for assets such as favicons, robots.txt, manifests, libraries incompatible with Vite, etc. + +In general, it's best to import files directly into a template, page, or component. This allows Vite to include that file in the bundle when small enough, or to copy it over to the `dist` folder with a hash. + +### Example Asset Import with Vite +Instead of handling our logo image as a static file per the example above, we can do the following: +``` +import React from "react" +import logo from "./my-logo.jpg" + + +function Header() { + return Logo +} + +export default Header +``` + +See Vite's docs for [static asset handling](https://vitejs.dev/guide/assets.html) diff --git a/web/public/favicon.png b/web/public/favicon.png new file mode 100644 index 0000000000000000000000000000000000000000..0f37d0f80143b1b6fee356d6f23e5ec00d6d1345 GIT binary patch literal 1681 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 gXgv-}aR+RE3u`4+9W*y85}Sb4q9e01{ktHUIzs literal 0 HcmV?d00001 diff --git a/web/public/img/bottom.png b/web/public/img/bottom.png new file mode 100644 index 0000000000000000000000000000000000000000..d648062aec44f8bf0e2573a691b2f3a718cdf563 GIT binary patch literal 5650 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 M6>a6WidLcj0YwA*PXGV_ literal 0 HcmV?d00001 diff --git a/web/public/img/colorbar.png b/web/public/img/colorbar.png new file mode 100644 index 0000000000000000000000000000000000000000..4defd579cdecc426a58cb3bd2552da64d6d61081 GIT binary patch literal 329 zcmeAS@N?(olHy`uVBq!ia0y~yVAKJ!IXKvWq_g>mxzU1*v)ej4Mq z8YqiLw<5%5yXSL0cbvC;Zu6-4oMr!yImPFc(S-O-3+AnfQGMwL3?c?kS3j3^P6&|tvz~REU(`LAK{ZA-@1$4~myN6h!lLhup&`9sm&b zcILEm*{}*a>Y}tW61ko+0Wec%Gur3smO>+ZZ{@aTTc=W+nwlV-DZ!MozR0(_Jfd2V z=1}lUzQ@_=>6{Na6F;W<6A!q&`dnZ9wIbTn+Sh1vn(EYdbf@VXfd3pwT^B~-pJ@q7foO4C*GbF4KQSWSI)b0(c@WdvvzrS`SO{trh7-asl|LNeqC=| z+%2v2vt_CJ!)IE(I?vo?&pNKfC;GRqel@76+E(yOfw`{BtB~Td^yDKW(|m0kI`ETDU_EHOQJ$^vX`!ZU`OBUM3oGa$Ao)YA zLlb`1unWWKj80NbX1Yvd;g9m{ar=_fcIv(P_L~Vh^Rid|Io#=mtv!ze6Q3X10LdNd z4(L<3o3|g`w5_y_2jjeeypujT0kSwc(cG2Grk-*Y+RN(^y%1mY%%p87;iM#q)O(57 zL=|!9K>pD3Daq7099#)gAL6I!io9yJXZ; zy2tiVK9%*}DSmdITJ1Aj7Ch)P7<$=w2EUm9b!%Q1%jwBY{q<>#ZbjNqaMiRhLCgbJ zpo#D|0C^ehb!rF68X`|vj!S`V=5Q7L^(p$!zG)paurH4;GRPsYkw$@L&^B)%@Ze z-TDldF-A>iQI^%`?l{+wi41++kQlSp2!>+}hTOc{=GcnFf85HIJ}TdF`n)UKK}d~} zD;NESu_b?3F6iyI3}o*uV_o74=@7K{t=1W?i`$RypyoUAgv zGpF}8_IbvYh^gzy$}~M;UY{S(y(PS?%XPmWxuc_^w*`|#Zg}acOq~Wq3G$l9YCFH? zL@Z${`ADVFKC-Z#WjNOQVw_e~f4+7{e!Uu-_A&+a>_CFN)Y~D$$d6+-+G?&Emp{7JrRB55QIx@Bg>0C!yBC3cpjY|)5`Xp1-Drzuo27?!{+I;r- z7AtmijjWe`Mr}bLLm~mw;3|7Wui+FqA|dU#JFAqz=+72b4 zlcbNo;`;GKQLSt47A;y4X2r=ag+{{b(afpCC%J@m%#W4|uNqR*P5?l#DaOV&X2!<< zJ&WNJ_-+*U2&n93gR zEX^ziGj~SmT-oC>75E?}hQCsE<;Gm7R{DhVGJo~`Lw!ftO)PDA%ZVYN$DM|wAW_N-hyNz5y%}yb8(cU!}O|F;(w2rGXxEB!8 zc{k;ZRutA|3sdjk^-HYx&-nJd(&!j{M58lrU}?G3`=Aw_r|p;X>)LI7&2MJmg*hfS z=)>M4V^uk=5xfa8&G~GL*l`lVX(LiT%qYvZ4Y~dp*Pk}{_~T(K1-j0nDosdE4w)s~ zc5XQ=G_S_@_dcB;%R&}10m(_jdhq24Y$98j5`lX_8PLKiIiv(vMEyvPfdHTZuWBeG z!g7p4;G#&7nWc#cAH;~L01fh2_QECEAQOimV_$D?cb_1@IKcgUkh>ccN(u6WOwBB9 zPBX-0VA(Y@BHj+ZWCu=Qcn}&_cYy1e5-XarNyqip{t=d2t3x249?i433iJMb{++Pnf>(dvfXF}d1kyvaHcnk+1$HcfGh{KI5iXjjn$FNZm0uZHF_MuP^ zcmx|u2T-LV53pD|w1UGj0aT+rhk&KiM}+VoOo)U)Jb+}1qZC0LHYyh&V`0_NKO4J5 zICxm~9ngXsi3-rBQS~bl^k!iI0RlL3<&F~GY0c^8Sa}r0!C^1MHWJ{NI4nTo*BAhb z2yu*H8!T#61h!#78*or1Y@-5f!^j23#04sxpiu%8G(t$k7r0TN!U0TyO3(;xz-9s! z1mHpDywG1(@b&~&Cw8@2no&#OaSqU903W$ zW252FieSAU^k1YH2>dfNaRu?eBv5JXZ;}>g5+*>KB%)wK3?v}Xw-XlxBGvI=Px0L* ONNQ$cMJyt?GXDV5=0^Ge literal 0 HcmV?d00001 diff --git a/web/public/img/snowglobs.png b/web/public/img/snowglobs.png new file mode 100644 index 0000000000000000000000000000000000000000..f342dd6a770e437dc1a4adade97cddbfb4179ad1 GIT binary patch literal 8825 zcmV- zaB^>EX>4U6ba`-PAZ2)IW&i+q+NGLlk|Q~eM*s5^a|D75Kpuyy+1y}`KM#~Lt1_pm zyV@qhBBbz;-01+mjmmKT=f7|GFa8xv8cMaMT{ZHr^wQJdNsFI<Jm$}@O~1zv#}ltl-{b4^D1+nmPbcc<`-l1KAIA4YseLc-J^tfDSw0tt z`+K7CJyA~IE0@pj^Lu6O&Hj7~-}m|L%lGkp2metr)v791_*qgk)6VmjVlnEreW7=y z?(eJfbDrOS4?kX-SW`M#a(qAAXO-E%a~?c4eoAKH{7j$c?|6Fkje(Z21XkX^wf)|z zSM!(J{CNDEf8n3p)^{9!Y4^`(-!TY3{*cRGkAa@=EAnsM)1Qy;c=-JfQvbT$ucv=n zk(KYW^f~_XM0fpGg!aexy}R9eK6f{Ya?AR?sUNo};hp5}=|k||rhFOSmFM=pI$sUA zEM{!S6(?Uyx+GeBrO1{cH^_NUmMqp-(n-c+N-O1etkH{}R`#YB|B{<5_q-X0loz9t zIhyVb-?hlQZhzNX(zx;#UYdlH`P}^5ukcS^^sm0|bu5WIh~!TPSXX*;DqhCyKY1A) zk^3!beFgsU_3J=>8=GliePwRk;E3~cidp54*~+icnb%bwKXWIxKkxSw6|P-bjAU96 zRdN-TXepTpg={nkI%_GHH0glUnl_6}C1<5KfPhN}-edA!n`-(Thb8Gm&6r_zqyc!$+-v>p}&XWg{txwYJ(EN7M3!m_!ovQyTRG{dHqK6i8Q5^!sn*x5LA7k|Dzm^gr|Qn!Vd08VsY;hNy;t?H*bC)d8@K9C z=bpT4E{l2|E*rx{Ds>NPQm)BuvKnWab@Sbsp|-eywW6CB9k9F^DPI@<4#=z||Q|pC^N@W_ed)Jk-eY$sg_d!*IO{Tn_(lxV=!D#R6GCpnR1^Zc? zY1Oy;O(|3E0j2GhXk_M9N&MxS`s$6dP`Wrl%LFK>?&77; zK?zK;rZgEXz%e`b&i7)dEE%*#Mb;eW^lK9c^AZ@nY0|^Hs<#ObCEa2`Sh;X!)}oIK zJ>#-*gFsH6jXD}F-7PQM^c-j`9q53<@Y|HRL^rtCut#e6$jO`aoO+KeV7eQhS=w|^1pmUnh zdY;_9arZHw+^HYU3w2T(rg%<>Lu+kufuy+B>XXKoxPV<}eCk-|GRZ@13b@cRhtJY5 zI7erI&ZsF<2r%=Bnx!F(8$Q9H!jtJjWGhpF<6{o?9YV1KeeXQHWpnj&@!3dhlG{^i zaAmDUPNSXl_9&IBV9@BOQDRaNaW+^deZ?nrBWC7f*^;4`*ScIl_ zbyDxL;9>q*s&IJ=pe*mTX+vEUheA*m=D&(5#Q{;5{Dk7X>u^g=03l=Sml|HF_%0H# zg;`OQ?H8TZb)mOYrl22*LgDl#YRSO%e0bUMDX^;%>T+Id$dWnMpe1QXiaJ&YXTQXx z4kek5?TMm{#mx2r%C#3!LDH$Y8YW@6js1pfxd=48%m0 ziUnr6?!{cMc9r=~9K8-UL6*3tqYxr&Fz3MlJF;HfTpsimXsbu^l@#W|C^&^v(E(es zuFZ1{oR`xYvN9P?ITL=#6;4AaD^+)8gZ%svhkOD9wqZ*m5<+{8QtibAZq~D4+uF~j z68AXEXHm33=uN>?8>0pqjh@4rf!>0dUE!g6;rL-5kQZdk>XWn{(V3wIp8^?Mfow&fTrPv$cn3FfFg$b3jezI{=}<$>cI&D-C{Mnp zaBt_1h5C^1b<)s|$kKhlrGllZoWXq^FKeJ5)(_B5PZYY~Y2goM)y>TDO0ssSR4Rpj zR@N~Z0W_FZnv0$@jbH_Syr3V={2&MJ#>&E9lcoCaHWt?SIjV;&l*|hUq29~Y=F0DV zNWGsZV^8Zmsol{#hPC%~8DLjb!};M&K{Yht+^{c%-l+9SW}GnB9$^irgM>{DqcHS^_Fd0vG-iV555`v`djKN$aX!^3 z#PUn+8+A11K?Ca5F>&4b37|#EL&bw9V?{xAYl~wTHlKSW!96g;sA#R%|0ujYh-3i| zRJhd5E}L%oOKSO_>e_aUXtBMmx2h$%nz8q>PGpYusp0*)Pf?9fAmA%W(V@T~Cu zjL899XP*6;EGjo12GzDt_~>mLgNouINeJZdK>M)OI%sDor@r5bm=ghbZ~o~afR1l& zmEu|HL3#Q<7L!eJfmISfgmuZ)>@x)KGVo3%tSs^v?<3DST0LX-Om^W{K!E}C5ELqy zc3xTCr&oMF?}tqmASWsk9gzTLshEo_Nf}~Bsk+I;b|93rChy>y|HbJKls=65tvO2T zbXZs}?HvQBl*~~IiIlhc4@|==pCi?A8ecu26;cuRg4s(H{WcuZcE9MJ9utc20dr=< z{xB8cS65MoxJ&tv2qNfAv`KOY9}7994_DI4{$7K@f*Oc|EeFoXh;f67m0bb(Zzn`C zR}y`683S(Ye0rDKnZsGlSqX}zy@#zN}S%W7Fjm}2INI@m&M8PGHeHyG838lxkenw7R z9rl<`j(Vs<5G#fa8;Onu@oXD0*<22-@jymK!Y-nTj09#4As2X)V=@R80uK{cKqC$r zuJ6Q{8*yx?D-6T$9bZAgoT?PNqD=8Z4rsqE5?n-8)9jE)=AP6?OuanF@QEh{*B?8+ ziM^fOs4`4{7Cor3SDp~@W1EZfeE=*lbzIR*&gi5|er-8eaXj8e&<6p7Mc zk2-eXS2Od6mkEDiW`=HON=egA+lAuri{h;N;OxEk9FJz#>S+)i+dcU1%tWLpv+uo7 zu7sh9d#|^O?AtEnN_2uiw8)@}TlIwV)?H^3gfErYb9`^%U|m_NEokHd0IVwT@(%~LS!r3v^6rt)xXDPDiq9!MBv7Ax zb|HXZJv?CdRuDm~AAXR9_!fx~P-(V$Xf!j)ZacGg%&EEg7N{GYF>s^+CIe%_0E(g` z?obTh2{gD6VFRv6*SGXoHTOnuH|V7AagZdA_!wp1?>Oy%e-~cY&3MhN9~lAHdHW&I zrUe+g61nf&oS(jvv#{?VelxthirioqvxiI}i5S+Cles;PjtTA!0mZnHjS_8+uCpY@ z<2+Ivq|vcu#MV=Y77xkCKMpFMSSk^saW#6Vq?cANom0c%D2ckROcrJ!eZ#5v=VKYr zP*1dM-h=^6q*s)M*r|bZy(?B6dT69=U3+I|VVxT=aTlJ_Y+`YIKk0qO#|vf>p9|w0XJ*VPJ4;5QPo#cu^#(PgN_UA1qcazRq%tx!h zazdd>p(p~Rh4#bcgS>H^EF6d^7H&czqsLpc^E}thjqOF;H*O@J&AYS@sj z0FAs*i-e$HHdvz&8r@ITOk~FJVA-;NaC#W&++yqgIR`OYD1KJSabs;$e~y>=_PI-Z5J8goPGcPF4i_xmSQG+d&6NYayF8?I#aaYHC%x48W7+@|xzEyPG4^$X|^>&^f zm~xlX1ku$%YF_8L417^t8eQgn&kow7_jmaLTr4@ZUSB;*j!VTiI!VGZhTDQ&g80S= zLRKNN?$TekZ?Deh%<$%jC+Dm3~Rgibj}`!dPP zE4J<1dPRNF7O{>nI`-lgJr-i~jb#7O!#5Hko+(sN3WL4;!_@bb%>3^j!*2)V*gkN4 zglfHe-TcEctneQA${E_-LukRES{@%wnyf0x^fz7Ux2=P$L zXS*MF4*Bg#hr~ZwgP@g}XTnxMG&@K*8v(O(JLGzR0EQa*C!-Ks>iDeJ|L4Kq-eLId z&HxL>@_zxEbK$9Ns|CLR00D(*LqkwWLqi~Na&Km7Y-Iodc$|HaJxIeq7>3`rN{d!R z>>%P0q&is;6>-!m6rn<>60W->T{x)gy;CW zhmY@fF`ngJ_vh$QawY?O0`V-<4U2e#czV;)IqwrkSW!}l&xywjx*+i**JYRAI2RoD z^UR2mPR$cXh=oEM%WcexhDtm|992|}^8Fc?70z3n)pC_J@5x^n&S@*lT&FdJ1QxLb z5dvgXQ9>COVzg_dm`Kri+`~WU_(gKb(i|qi@Or{kK5Zn%lSLIZhvdH1#TZ0~{OzqXo)d_jq@xy>I{4H2e1hOBZsNl-@QY z00006VoOIv0G$A`0LX)|I3fT5010qNS#tmYE+YT{E+YYWr9XB6000McNliru=Lii6 z4FXaAHfR6<02y>eSad^gZEa<4bO1wgWnpw>WFU8GbZ8()Nlj2!fese{01UNBL_t(| z+U;FEYg|Vcp1oQv7#j@2F4PzdF5)V-4=gy6a(C{TGv_%UGiMeTfBqvvaddokEh6ySufHq= z0000mJQltr6iYlHB>(^bAee+qI35j>T6 zY|oHM>FD_E`uU4P@%rTfCJF!mU{n%P3CKh~`%j()ec9!w3nMAbWc0^}b8+;_~swtPY{xd>?g$+IK`q)zJYy?t%i`}W=YYfKXW z06^y?WFnwkc=Y)Go$uROJrqLxs1X4G004brsYXF5!t#)3?cUoL-@jdooz>oLeFy*m z02m+=qVE21m1 zp6U7R*IyRB)N_2_Bj>rM7}?_dRaH8(7DAjtV#Dh8$j?GoSvfGlzLNAXIhVXO6m3p_QfRgoz+9Jv)aol z_i9OK|H-r9mKR(N215;r?(eu-52fg4t{Z2}f(;!{h*MG1vt7@3^<50n8Fd~~T%FRQ zAO{pvL!kzU+P_b|QiCIv7$cJ$27>F{ zTI#-wJrU3dx*cj`a@ZHsdP=ENHn|szgf?G4l0P*dG%}-3Fx9AW^zT!nLT(vR+BTK+ zQge)Tr2A%C*6q~xgTubKoHuG(Hzr%5qiimoy}zSMoW??k#D-l`$s?td3r7m77H zvEx^F^o-oeJZA)ws#`*ls}v@pJJmHTfq;DG{2{q7$+^u(03bas1ZSj zc<6R+BJF96EE^MaXKi|jib|r!Lx~(lJY_TyQV%FOXSwGMx?M)Mfd~?!Oup@a5~8BK zp>qo!tR-5h;t*jE^9J2+J~|tZgkmRKJuQoHJ1 z_W3eIMJEtHyj2(r0B9dEZP+H8F^MH2M^v;%RCI@w;lTsYZD7nuNHr#$C$&*4@(1$M z^?7#ybQ_L@*+z4;VViusS`H#Yt%y$CKv|C*n+KrVa2(QM=~{Rk?P_ENRPUSU?J_l> zKJwwXq15K20YGn#g02&eyaqVe!sJ?L+)WgHq9UjPklKrbM-GgLP#d13d@`9lK3k25 zi;PS!3nCgz(dYT*n0Js10Bj@Xw&B#}lgT&zpuPkL8R?=T5Yae_Zi-e=#0CH!aQ42` zMMWHTdJ1_7-R=hNs0c*Vq@u4=L@S7D0HAE9!c{)uu+v_><8*Z*?i3w}sA-%+1OU)Z zrXuzxS5u0PVz0d$nCe=1b4NuWBA`WY5dbRBRIr=@Rih%+PE+mmM0p`O{la|>ap}9@ z0l?;H0YGm=6iY^PR7BmpK)uT0)U}nHv-cbT)U=2JpeG`t1`j$rP1(FK#l95xiEii%=3FHAXJECxu)wF&B3Ax@~e16=|DdqV;$8R;S;-$n#eju+z!^?>>j zdsfIr6h1BDUHNM0762mJ&SG(pk!$mUj)P5c0otHw06R!K!<>q$t7j3?NjBbLeHaZ3RfUZ%Z4*|)#VAHMnzyJUJN-TGG zAaDS%*ln2{bPcR2PMo~PWjii67M{2W$i}c#0)f{$1`(+r8&qBL(Jc@V0QlML6!jw4 zv71Sk5()kIhI-V{4gh>~iwn?E0A$45uFc?Fp4=g!)Id57XbsR=0FK_>^}t!@pR;gF zPsSp9jBS4z@F z9TgbC;XFD%yI#r*8L5GKj4nK7H`(;I=|%T0y{$>;c3JqyK1U!TZ%&v{46WD50{J75 z*agU07wW0s9dsKo=>uv)of}mkkI{^+3)yn=EfWxQ8*u3dNT@S#rl&Z<^1uK3w`&I> zwkt?2=@IBQPzoTSPLQ&8K=JagHlKwMLVW*rIb`${N<=O)8rf*prrRx}Mm=IBA)Tj! zT|Q}%Ggaf0T6zBB@P>rG43$zaEqLEZP&E#7ixlqbP@vm%-R{Lf3dP%Z?~~f_iGwXo zcSXqmrg$?Wdw-?ho%;Q+Umh$2d0sJXd>s2!Wa^UrqsRBf z9Mw5eco0&j#5yzZ8fC|cjQzY)kEVxZ-SyN-RS=G^PyIrx?0aDj@F2}?Yhv7 z8U;~wC*+-7#|Joqi`$!3=O5?_%3NDbVM;V}&e*XI-fi2S*DntiM<33~ZMtL028K`wA;jsNehF=( zWNMR6U8J1k-)gHCPMyP{%|3OTi!egh$DFCs)g4_9R)}rqShp_OPo{+;zg(`=m~N-g zXfSy#zMhNeLzR1s@r<2694P#8{V{aAKHUx>AX{m}=N)Y8V$NLY(!VCztK0rlt}`Fm zZ(VBl)6a+Ef4`sK7FA`|Hgua`Mk)bO>?#+bmX^P6tV`QsPcc~2 zKy?!}avoicCUkSaWi(O~PzG78OGgbg04<;e5!NLdv zy*UFjX>im?W1=gS)Ueionh`}p006{ALoOl$0Rg>Z2@wDQ4@z_zDSZ|O01ycQKt<7z z8WC|2jc@b>0s=rn001H~5>YH6l{#n)03aa%Xpo3}2q^}fXY2p~Bm@8*6A>37jd~9e v01^TKfSHgu0P+DqLI40{B_t3J&=dXxRNf=`Zr=>B00000NkvXXu0mjfxp~xx literal 0 HcmV?d00001 diff --git a/web/public/img/snowglobs.xcf b/web/public/img/snowglobs.xcf new file mode 100644 index 0000000000000000000000000000000000000000..80931140f0442849587d365cc6602767ac818057 GIT binary patch literal 40011 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 a*6{bg@w-nC{(g+#k@;%1--O$y|NjHO`*l75 literal 0 HcmV?d00001 diff --git a/web/public/img/top.png b/web/public/img/top.png new file mode 100644 index 0000000000000000000000000000000000000000..23b9cdbc9732e4b14939fe3a1248a1bcd38fd6e1 GIT binary patch literal 5696 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 JFIBq~@js{4$HV{t literal 0 HcmV?d00001 diff --git a/web/public/robots.txt b/web/public/robots.txt new file mode 100644 index 0000000..eb05362 --- /dev/null +++ b/web/public/robots.txt @@ -0,0 +1,2 @@ +User-agent: * +Disallow: diff --git a/web/src/App.tsx b/web/src/App.tsx new file mode 100644 index 0000000..bb3d805 --- /dev/null +++ b/web/src/App.tsx @@ -0,0 +1,27 @@ +import { FatalErrorBoundary, RedwoodProvider } from '@redwoodjs/web' +import { RedwoodApolloProvider } from '@redwoodjs/web/apollo' + +import { ChakraProvider, ColorModeScript, extendTheme } from '@chakra-ui/react' +import * as theme from 'config/chakra.config' + +import FatalErrorPage from 'src/pages/FatalErrorPage' +import Routes from 'src/Routes' + +import './index.css' + +const extendedTheme = extendTheme(theme) + +const App = () => ( + + + + + + + + + + +) + +export default App diff --git a/web/src/Routes.tsx b/web/src/Routes.tsx new file mode 100644 index 0000000..0291fd9 --- /dev/null +++ b/web/src/Routes.tsx @@ -0,0 +1,21 @@ +// In this file, all Page components from 'src/pages` are auto-imported. Nested +// directories are supported, and should be uppercase. Each subdirectory will be +// prepended onto the component name. +// +// Examples: +// +// 'src/pages/HomePage/HomePage.js' -> HomePage +// 'src/pages/Admin/BooksPage/BooksPage.js' -> AdminBooksPage + +import { Router, Route } from '@redwoodjs/router' + +const Routes = () => { + return ( + + + + + ) +} + +export default Routes diff --git a/web/src/components/.keep b/web/src/components/.keep new file mode 100644 index 0000000..e69de29 diff --git a/web/src/entry.client.tsx b/web/src/entry.client.tsx new file mode 100644 index 0000000..ffee44f --- /dev/null +++ b/web/src/entry.client.tsx @@ -0,0 +1,17 @@ +import { hydrateRoot, createRoot } from 'react-dom/client' + +import App from './App' +/** + * When `#redwood-app` isn't empty then it's very likely that you're using + * prerendering. So React attaches event listeners to the existing markup + * rather than replacing it. + * https://reactjs.org/docs/react-dom-client.html#hydrateroot + */ +const redwoodAppElement = document.getElementById('redwood-app') + +if (redwoodAppElement.children?.length > 0) { + hydrateRoot(redwoodAppElement, ) +} else { + const root = createRoot(redwoodAppElement) + root.render() +} diff --git a/web/src/index.css b/web/src/index.css new file mode 100644 index 0000000..e69de29 diff --git a/web/src/index.html b/web/src/index.html new file mode 100644 index 0000000..e240b8e --- /dev/null +++ b/web/src/index.html @@ -0,0 +1,15 @@ + + + + + + + + + + + +
+ + + diff --git a/web/src/layouts/.keep b/web/src/layouts/.keep new file mode 100644 index 0000000..e69de29 diff --git a/web/src/pages/FatalErrorPage/FatalErrorPage.tsx b/web/src/pages/FatalErrorPage/FatalErrorPage.tsx new file mode 100644 index 0000000..b2bb436 --- /dev/null +++ b/web/src/pages/FatalErrorPage/FatalErrorPage.tsx @@ -0,0 +1,57 @@ +// This page will be rendered when an error makes it all the way to the top of the +// application without being handled by a Javascript catch statement or React error +// boundary. +// +// You can modify this page as you wish, but it is important to keep things simple to +// avoid the possibility that it will cause its own error. If it does, Redwood will +// still render a generic error page, but your users will prefer something a bit more +// thoughtful :) + +// This import will be automatically removed when building for production +import { DevFatalErrorPage } from '@redwoodjs/web/dist/components/DevFatalErrorPage' + +export default DevFatalErrorPage || + (() => ( +
+