Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/hviana/faster_react
Tiny Full-Stack React framework. Avoid Overengineering. Automatic routes, reload and component bundle. It uses its own RSC engine, combining SSR and CSR. 100% Deno, no Node dependencies. Fully compatible with Deno Deploy and Serverless Environments.
https://github.com/hviana/faster_react
deno full-stack-web-development react rsc
Last synced: 7 days ago
JSON representation
Tiny Full-Stack React framework. Avoid Overengineering. Automatic routes, reload and component bundle. It uses its own RSC engine, combining SSR and CSR. 100% Deno, no Node dependencies. Fully compatible with Deno Deploy and Serverless Environments.
- Host: GitHub
- URL: https://github.com/hviana/faster_react
- Owner: hviana
- License: mit
- Created: 2024-09-17T18:56:28.000Z (4 months ago)
- Default Branch: main
- Last Pushed: 2024-12-26T12:51:18.000Z (24 days ago)
- Last Synced: 2025-01-05T12:07:07.501Z (14 days ago)
- Topics: deno, full-stack-web-development, react, rsc
- Language: TypeScript
- Homepage:
- Size: 90.8 KB
- Stars: 97
- Watchers: 6
- Forks: 6
- Open Issues: 0
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
- awesome-deno - faster_react - Full Stack web framework with React + Faster. Fully compatible with Deno Deploy. (Modules / Web framework)
README
![](https://img.shields.io/badge/development_stage-stable-blue)
![](https://img.shields.io/badge/tests-pass-green)# π **faster_react**
> [!IMPORTANT]\
> **Please give a star!** β---
## π Introduction
`faster_react` is a tiny Full-Stack React framework. He avoids Overengineering.
This framework **uses its own RSC engine, combining SSR and CSR**, and
automatically generates routes for React components. To utilize this, you must
use the routes helper provided by the framework
([React Router](#-react-router)). The framework's configuration file is located
at `options.json`.### π― **What Does `faster_react` Do for You?**
Focus solely on development! This framework handles:
- π£οΈ **Automatic route generation** for React components.
- π **Automatic inclusion** of new React components when
`framework => "dev": true`.
- π¦ **Automatic frontend bundling** when `framework => "dev": true`.
- β»οΈ **Automatic browser reload** when `framework => "dev": true`.
- ποΈ **Automatic frontend minification** when `framework => "dev": false`.
- π **Automatic backend reload** when changes are detected and
`framework => "dev": true`.
- π **Automatic detection** of Deno Deploy environment. Test in other
serverless environments by setting `framework => "serverless": true`.> **Note:** The project includes a simple application example demonstrating each
> functionality. The example uses Tailwind CSS, but this is optional. You can
> use whatever CSS framework you want.---
### β‘ **About Faster**
This framework uses a middleware library called Faster. Faster is an optimized
middleware server with an incredibly small codebase (~300 lines), built on top
of native HTTP APIs with no dependencies. It includes a collection of useful
middlewares:- π **Log file**
- ποΈ **Serve static**
- π **CORS**
- π **Session**
- β±οΈ **Rate limit**
- π‘οΈ **Token**
- π₯ **Body parsers**
- π **Redirect**
- π **Proxy**
- π€ **Handle upload**Fully compatible with Deno Deploy and other enviroments. Examples of all
resources are available in the [README](https://github.com/hviana/faster).
Faster's ideology is simple: all you need is an optimized middleware manager;
all other functionality is middleware.---
## π **Contents**
- [β‘ Benchmarks](#-benchmarks)
- [ποΈ Architecture](#%EF%B8%8F-architecture)
- [π App Structure](#-app-structure)
- [π¦ Get Deno Kv and Deno Kv Fs](#-get-deno-kv-and-deno-kv-fs)
- [π Backend API](#-backend-api)
- [𧩠Backend Components](#-backend-components)
- [π Backend Files](#-backend-files)
- [π₯οΈ Frontend Components](#%EF%B8%8F-frontend-components)
- [π¨ Frontend CSS](#-frontend-css)
- [π Frontend Files](#-frontend-files)
- [π Frontend Translations](#-frontend-translations)
- [ποΈ Static](#%EF%B8%8F-static)
- [π§ React Router](#-react-router)
- [π¦ Packages Included](#-packages-included)
- [π οΈ Creating a Project](#%EF%B8%8F-creating-a-project)
- [π Running a Project](#-running-a-project)
- [π Deploy](#-deploy)
- [π References](#-references)
- [π¨βπ» About](#-about)---
## β‘ **Benchmarks**
`faster_react` has only **0.9%** of the code quantity of Deno Fresh.
**Benchmark Command:**
```bash
# Deno Fresh
git clone https://github.com/denoland/fresh.git
cd fresh
git ls-files | xargs wc -l
# Output: 104132 (version 1.7.3)# faster_react
git clone https://github.com/hviana/faster_react.git
cd faster_react
git ls-files | xargs wc -l
# Output: 1037 (version 20.1)
```---
## ποΈ **Architecture**
This framework utilizes **Headless Architecture** [[1]](#1) to build the
application, combined with the **Middleware Design Pattern** [[2]](#2) for
defining API routes in the backend.- **Headless Architecture** provides complete freedom to the developer, reducing
the learning curve. Despite this freedom, there is an **explicit separation
between backend and frontend**, which aids in development.
- The **Middleware Design Pattern** offers a practical and straightforward
method for defining API routes.![Architecture Diagram](https://raw.githubusercontent.com/hviana/faster_react_core/refs/heads/main/docs/graph.svg)
---
## π **App Structure**
All application folders are inside the `app` folder.
### π¦ **Get Deno Kv and Deno Kv Fs**
On the backend, if a **Deno KV** instance is available, access instances via
`Server.kv` and `Server.kvFs`:```typescript
import { Server } from "faster";
```See **Deno KV** settings in `options.json`.
- **Deno KV File System (`Server.kvFs`):** Compatible with Deno Deploy. Saves
files in 64KB chunks. Organize files into directories, control the KB/s rate
for saving and reading files, impose rate limits, set user space limits, and
limit concurrent operationsβuseful for controlling uploads/downloads. Utilizes
the Web Streams API.More details: [deno_kv_fs](https://github.com/hviana/deno_kv_fs)
---
### π **Backend API**
- **Imports:** Import your backend libraries here.
- **Organization:** Files can be organized into subdirectories.
- **File Extension:** Use `.ts` files.
- **Structure:** Flexible file and folder structure that doesn't influence
anything.
- **Routing:** Define routes using any pattern you prefer.
- **Exports:** Must have a `default export` with a function (can be
asynchronous).
- **Function Input:** Receives an instance of `Server` from `faster`.
- **Usage:** Perform backend manipulations here (e.g., fetching data from a
database), including asynchronous calls.
- **Routes:** Define your custom API routes. For help, see:
[faster](https://github.com/hviana/faster)---
### 𧩠**Backend Components**
- **Optionality:** A backend component is optional for a frontend component.
- **Imports:** Import your backend libraries here.
- **Organization:** Organize files into subdirectories.
- **File Extension:** Use `.ts` files.
- **Correspondence:** Each file should have the same folder structure and name
as the corresponding frontend component but with a `.ts` extension.- **Example:**
- Frontend: `frontend/components/checkout/cart.tsx`
- Backend: `backend/components/checkout/cart.ts`- **Exports:** Must have a `default export` with an object of type
`BackendComponent`:```typescript
import { type BackendComponent } from "@helpers/backend/types.ts";
```- **Usage:** Intercept a frontend component request:
- **Before Processing (`before?: RouteFn[]`):** List of middleware functions
(see: [faster](https://github.com/hviana/faster)). Use to check headers
(`ctx.req.headers`) or search params (`ctx.url.searchParams`), like tokens,
impose rate limits etc.> **Note:** To cancel page processing, do not call `await next()` at the end
> of a middleware function.> **Important:** If you want the page to be processed, **do not** consume
> the `body` of `ctx.req`, or it will cause an error in the framework.- **After Processing
(`after?: (props: JSONObject) => void | Promise`):** Function receives
the `props` that will be passed to the component. Add backend data to these
`props`, such as data from a database. Can be asynchronous.
> **Note:** Only use props data in JSON-like representation, or hydration
> will fail.---
### π **Backend Files**
- **Imports:** Import your backend libraries here.
- **Organization:** Organize files into subdirectories.
- **File Extension:** Use `.ts` files.
- **Usage:** Free to make exports or calls (including asynchronous).
- **Purpose:** Group common functions/objects for `backend/api`,
`backend/components`, and other `backend/files`, such as user validations.---
### π₯οΈ **Frontend Components**
- **Imports:** Use only frontend libraries.
- **Organization:** Organize files into subdirectories.
- **File Extension:** Use `.tsx` files.
- **Rendering:** Rendered on the server and hydrated on the client.
- **Routes Generated:** Two routes per file (e.g.,
`frontend/components/checkout/cart.tsx`):
- **Page Route:** For rendering as a page, e.g., `/pages/checkout/cart`.
- **Component Route:** For rendering as a component, e.g.,
`/components/checkout/cart`.
- **Initial Route (`/`):** Points to `frontend/components/index.tsx`.
- **Exports:** Must have a `default export` with the React Function/Component.
- **Props Passed to Component:**
- Form-submitted data (or JSON POST).
- URL search parameters (e.g., `/pages/myPage?a=1&b=2` results in
`{a:1, b:2}`).
- Manipulations from `backend/components`.---
### π¨ **Frontend CSS**
Application CSS style files.
- **Multiple Files:** Automatically compiled.
- **Organization:** Organize files into subdirectories.---
### π **Frontend Files**
- **Imports:** Use only frontend libraries.
- **Organization:** Organize files into subdirectories.
- **File Extensions:** Use `.ts` and `.js` files.
- **Usage:** Free to make exports or calls (including asynchronous).
- **Difference from Components:** Scripts are not automatically delivered to the
client. They need to be imported by the `frontend/components`.
- **Purpose:** Group common functions/objects for React Functions/Components,
like form field validations. Can have `frontend/files` common to other
`frontend/files`.---
### π **Frontend Translations**
- **File Extensions:** Use `.json` files.
- **Correspondence:** Each file should have the same folder structure and name
as the corresponding frontend component but with a `.json` extension.- **Example:**
- Frontend: `frontend/components/checkout/cart.tsx`
- Backend: `frontend/translations/checkout/cart.json`
- **Usage:**In `frontend/components/index.tsx`:
```jsx
import {
detectedLang,
useTranslation,
} from "@helpers/frontend/translations.ts";
const Home = () => {
const t = useTranslation();
//Any .init parameter of i18next (minus ns) is valid in useTranslation.
//Ex: useTranslation({ lng: ["es"], fallbackLng: "en" }) etc.
//On the client side, the language is automatically detected (if you don't specify).
//On the server, the language is "en" (if you don't specify).
//The "en" is also the default fallbackLng.
return (
{t("index.appName", { endExample: "!" })}
);
};
export default Home;
```In `frontend/translations/en/index.json`:
```json
{
"appName": "My SaaS App {{endExample}}"
}
```The framework translation is just a wrapper over i18next. See the i18next
documentation if you have questions.---
### ποΈ **Static**
Files served statically. Routes are generated based on the folder and file
structure.- **Example:** `localhost:8080/static/favicon.ico` matches `static/favicon.ico`.
---
## π§ **React Router**
Since the framework has its own routing system, a third-party routing library is
unnecessary. Use the framework helper:> **Note:** Direct form submissions for page routes path also work.
```typescript
import { getJSON, route } from "@helpers/frontend/route.ts";
```### **Interface Parameters:**
```typescript
interface Route {
headers?: Record; // When routing to a page, headers are encoded in the URL. Intercept them in ctx.url.searchParams in a backend/components file.
content?:
| Record
| (() => Record | Promise>);
path: string;
startLoad?: () => void | Promise;
endLoad?: () => void | Promise;
onError?: (e: Error) => void | Promise;
disableSSR?: boolean; //For component routes. Disables SSR; defaults to false.
elSelector?: string; // Required for component routes.
method?: string; // Only for API routes. Optional; defaults to GET or POST.
}
```### **Examples**
**Navigating to a Page with Search Params:**
```jsx
// URL search params passed as properties to the page. Props receive `{a:1}`Go to Test Page
;
```**Passing Additional Parameters:**
```jsx
// Props receive `{a:1, example:"exampleStr"}`Go to Test Page with Extra Data
;
```**Using Asynchronous Content:**
```jsx
// Props receive `{a:1, ...JSONResponse}`
{
return await getJSON({
path: "/example/json",
content: {
test: "testData",
},
});
},
})}
>
Go to Test Page with Async Data
;
```**Programmatic Routing:**
```typescript
(async () => {
if (user.loggedIn) {
await route({
path: "/pages/dash",
content: { userId: user.id, token: token },
})();
} else {
await route({ path: "/pages/users/login" })();
}
})();
```**Loading a Component:**
```jsx
Load Counter Component
;
```**Making an API Call:**
```jsx
{
const res = await getJSON({
path: "/example/json",
content: {
test: "testData",
},
});
console.log(res);
alert(JSON.stringify(res));
}}
>
Fetch JSON Data
;
```In the case of page routes, you can use this example to pass the URL parameters
for the headers in the backend (if you really need it):```typescript
const signupBackendComponent: BackendComponent = {
before: [
async (ctx: Context, next: NextFunc) => {
ctx.req = new Request(ctx.req, {
headers: {
...Object.fromEntries(ctx.req.headers as any),
"Authorization": `Bearer token ${ctx.url.searchParams.get("token")}`,
},
});
await next();
},
],
};
export default signupBackendComponent;
```Forms submit for page routes work. For components, you can use the following:
```tsx
{
event.preventDefault();
const data: any = new FormData(event.target as any);
const formObject = Object.fromEntries(data.entries());
await route({
startLoad: () => setLoading(true), //useState
endLoad: () => setLoading(false),
path: "/components/register",
elSelector: "#dash-content",
content: formObject,
})();
}}
>
```---
## π¦ **Packages Included**
Several packages are included to assist in developing React applications. Here
are some examples of imports you can use without additional configuration:```typescript
import {/* your imports */} from "react";
import {/* your imports */} from "react/";
import {/* your imports */} from "i18next";
import {/* your imports */} from "react-dom";
import {/* your imports */} from "react-dom/server";
import {/* your imports */} from "react-dom/client";
import {/* your imports */} from "react/jsx-runtime";
s;
import {/* your imports */} from "@helpers/frontend/route.ts";
import {/* your imports */} from "@helpers/frontend/translations.ts";
import {/* your imports */} from "@helpers/backend/types.ts";
import {/* your imports */} from "faster";
import {/* your imports */} from "deno_kv_fs";
import {/* your imports */} from "jose"; //manage tokens
import { options, server } from "@core"; // Useful for accessing the server instance.
```---
## π οΈ **Creating a Project**
You can simply download this repository. Alternatively, use the command
(requires `git` installed and configured):```bash
deno run -A -r "https://deno.land/x/faster_react_core/new.ts" myProjectFolder
```Customize and configure the server in `options.json`.
---
## π **Running a Project**
Execute the command:
Development:
```bash
deno task serve
```Production:
```bash
deno serve main.ts #Add your permissions, port, certificate etc. see: https://docs.deno.com/runtime/reference/cli/serve
```---
## π **Deploy**
- **Install Deployctl:**
```bash
deno install -A --global jsr:@deno/deployctl
```- **Deploy Your Project:**
```bash
deployctl deploy
```> **Note:** For production, set `framework => "dev": false` in `options.json`.
---
## π **References**
[1] Dragana Markovic, Milic Scekic, Alessio Bucaioni, and Antonio
Cicchetti. 2022. _Could Jamstack Be the Future of Web Applications Architecture?
An Empirical Study._ In _Proceedings of the 37th ACM/SIGAPP Symposium on Applied
Computing_ (SAC '22). Association for Computing Machinery, New York, NY, USA,
1872β1881. DOI:
[10.1145/3477314.3506991](https://doi.org/10.1145/3477314.3506991)[2] Brown, Ethan. _Web Development with Node and Express:
Leveraging the JavaScript Stack_. O'Reilly Media, 2019. URL:
[http://www.oreilly.com/catalog/9781492053484](http://www.oreilly.com/catalog/9781492053484)---
## π¨βπ» **About**
**Author:** Henrique Emanoel Viana, a Brazilian computer scientist and web
technology enthusiast.- π **Phone:** +55 (41) 99999-4664
- π **Website:**
[https://sites.google.com/view/henriqueviana](https://sites.google.com/view/henriqueviana)> **Improvements and suggestions are welcome!**
---