mirror of
https://github.com/Unleash/unleash.git
synced 2025-01-25 00:07:47 +01:00
docs: add ADRs
This commit is contained in:
parent
e30b830305
commit
41c9e1aa0d
19
website/docs/contributing/backend/ADR/naming.md
Normal file
19
website/docs/contributing/backend/ADR/naming.md
Normal file
@ -0,0 +1,19 @@
|
||||
# ADR: Naming
|
||||
|
||||
## Background
|
||||
|
||||
In the codebase, we have found a need to have a common way of naming things in order to ensure consistency. It's important that files are named after the contents of the file to ensure that it's easy to search for files. You should be able to find the file you need in the command line without the help of advanced IDEs. This can easily be solved by proper naming. It's also crucial that the naming is consistent across the project, if we are using different naming conventions in different places, it will be hard to navigate the codebase.
|
||||
|
||||
## Decision
|
||||
|
||||
We have decided to use a naming convention where the files are named after the main class that it contains. Example:
|
||||
|
||||
```js
|
||||
feature-toggle-service.ts
|
||||
|
||||
class FeatureToggleService {
|
||||
...
|
||||
}
|
||||
```
|
||||
|
||||
The reason for this decision is to remove mental clutter and free up capacity to easily navigate the codebase. Knowing that a file is named after the class that it contains allows you to quickly scan a file without watching for a context where the class is used in order to understand what it is.
|
@ -0,0 +1,9 @@
|
||||
# ADR: Preferred export
|
||||
|
||||
## Background
|
||||
|
||||
In the codebase, we have discovered that default exports create multiple problems. One is that you can rename the component when importing it, which can cause confusion. Another is that it is harder to find the component when you are looking for it, as you have to look for the file name instead of the component name (solved by ADR for naming, but still relevant).
|
||||
|
||||
## Decision
|
||||
|
||||
We have decided to use named exports. This will allow us to eliminate the possiblity of exporting a component and renaming it in another file. It also allows us easy access to advanced refactors across the project, because renaming in one place will propagate to all the other places where that import is referenced. This resolves the issues described in the background without any significant downsides.
|
13
website/docs/contributing/backend/overview.md
Normal file
13
website/docs/contributing/backend/overview.md
Normal file
@ -0,0 +1,13 @@
|
||||
# Backend
|
||||
|
||||
The frontend is written in nodejs/typescript. It's written as a REST API following a CSR (controller, service, repository/store) pattern. The following ADRs are defined for the backend:
|
||||
|
||||
## ADRs
|
||||
|
||||
We have created a set of ADRs to help guide the development of the backend. These are located in the [ADR folder](./ADR).
|
||||
|
||||
* [Naming](./ADR/naming.md)
|
||||
* [Preferred export](./ADR/preferred-export.md)
|
||||
|
||||
## Internal feature flagging
|
||||
|
@ -1,5 +1,40 @@
|
||||
## Introduction {#introduction}
|
||||
|
||||
This repository contains two main parts. The backend and the frontend of unleash. The backend is a Node.js application that is built using TypeScript. The frontend is a React application that is built using TypeScript. The backend specific code can be found in the `src` lib folder. The frontend specific code can be found in the `frontend` folder.
|
||||
|
||||
## Development philosophy
|
||||
|
||||
The development philosophy at unleash is centered at delivering high quality software. We do this by following a set of principles that we believe will help us achieve this goal. We believe that these principles will also help us deliver software that is easy to maintain and extend, serving as our north star.
|
||||
|
||||
We believe that the following principles will help us achieve our goal of delivering high quality software:
|
||||
|
||||
* We test our code always
|
||||
|
||||
Software is difficult. Being a software engineer is about acknowledging our limits, and taking every precaution necessary to avoid introducing bugs. We believe that testing is the best way to achieve this. We test our code always, and prefer automation over manual testing.
|
||||
|
||||
* We strive to write code that is easy to understand and maintain
|
||||
|
||||
We believe code is a language. Written code is a way to communicate intent. It's about explaining to the reader what this code does, in the shortest amount of time possible. As such, writing clean code is supremely important to us. We believe that this contributes to keeping our codebase maintainable, and helps us maintain speed in the long run.
|
||||
|
||||
* We think about solutions before comitting
|
||||
|
||||
We don't jump to implementation immediately. We think about the problem at hand, and try to examine the impact that this solution may have in a multitude of scenarios. As our product core is open source, we need to balance the solutions and avoid implementations that may be cumbersome for our community. The need to improve our paid offering must never come at the cost of our open source offering.
|
||||
|
||||
### Required reading
|
||||
|
||||
The following resources should be read before contributing to the project:
|
||||
|
||||
* [Clean code javascript](https://github.com/ryanmcdermott/clean-code-javascript)
|
||||
* [frontend overview](./frontend/overview.md)
|
||||
* [backend overview](./backend/overview.md)
|
||||
|
||||
### Recommended reading
|
||||
|
||||
*
|
||||
|
||||
|
||||
## Requirements
|
||||
|
||||
Before developing on this project you will need two things:
|
||||
|
||||
- PostgreSQL 10.x or newer
|
||||
|
27
website/docs/contributing/frontend/ADR/component-naming.md
Normal file
27
website/docs/contributing/frontend/ADR/component-naming.md
Normal file
@ -0,0 +1,27 @@
|
||||
# ADR: Component naming and file naming
|
||||
|
||||
## Background
|
||||
|
||||
In the codebase, we have found a need to have a common way of naming components so that components can be (a) easily searched for, (b) easily identified as react components and (c) be descriptive in what they do in the codebase.
|
||||
|
||||
## Decision
|
||||
|
||||
We have decided to use a naming convention for components that uppercases the first letter of the component. This also extends to the filename of the component. The two should always be the same:
|
||||
|
||||
```jsx
|
||||
// Do:
|
||||
// MyComponent.ts
|
||||
|
||||
const MyComponent = () => {};
|
||||
|
||||
// Don't:
|
||||
// someRandomName.ts
|
||||
|
||||
const MyComponent = () => {};
|
||||
```
|
||||
|
||||
The reason for this decision is to remove mental clutter and free up capacity to easily navigate the codebase. Knowing that a component name has the same name as the filename will remove any doubts about the file contents quickly and in the same way follow the React standard of uppercase component names.
|
||||
|
||||
### Deviations
|
||||
|
||||
In some instances, for simplicity we might want to create internal components or child components for a larger component. If these child components are small enough in size and it makes sense to keep them in the same file as the parent (AND they are used in no other external components) it's fine to keep in the same file as the parent component.
|
21
website/docs/contributing/frontend/ADR/interface-naming.md
Normal file
21
website/docs/contributing/frontend/ADR/interface-naming.md
Normal file
@ -0,0 +1,21 @@
|
||||
# ADR: Interface naming
|
||||
|
||||
## Background
|
||||
|
||||
In the codebase, we have found a need to have a common way of naming interfaces in order to ensure consistency.
|
||||
|
||||
## Decision
|
||||
|
||||
We have decided to use a naming convention of appending the letter `I` in front of interfaces to signify that we are in fact using an interface. For props, we use `IComponentNameProps`.
|
||||
|
||||
```jsx
|
||||
// Do:
|
||||
interface IMyInterface {}
|
||||
interface IMyComponentNameProps {}
|
||||
|
||||
// Don't:
|
||||
interface MyInterface {}
|
||||
interface MyComponentName {}
|
||||
```
|
||||
|
||||
The reason for this decision is to remove mental clutter and free up capacity to easily navigate the codebase. Knowing that an interface is prefixed with `I` allows you to quickly scan a file without watching for a context where the interface is used in order to understand what it is.
|
@ -0,0 +1,35 @@
|
||||
# ADR: Preferred component props usage
|
||||
|
||||
## Background
|
||||
|
||||
In the codebase, we have found a need to standardise how to use props, in order to easily be able to figure out what a component is doing and what properties it is given without having to look up the interface.
|
||||
|
||||
## Decision
|
||||
|
||||
We have decided to use props destructuring inline in components in order to quickly display what properties a component is using.
|
||||
|
||||
```tsx
|
||||
// Do:
|
||||
const MyComponent = ({ name, age, occupation }: IComponentProps) => {
|
||||
return (
|
||||
<div>
|
||||
<p>{age}</p>
|
||||
<p>{name}</p>
|
||||
<p>{occupation}</p>
|
||||
</>
|
||||
)
|
||||
};
|
||||
|
||||
// Don't:
|
||||
function MyComponent(props) {
|
||||
return (
|
||||
<div>
|
||||
<p>{props.age}</p>
|
||||
<p>{props.name}</p>
|
||||
<p>{props.occupation}</p>
|
||||
</>
|
||||
)
|
||||
}
|
||||
```
|
||||
|
||||
The reason for this decision is to remove mental clutter and free up capacity to easily navigate the codebase. In addition, when components grow, the ability to look at the signature and instantly know what dependencies this component uses gives you an advantage when scanning the codebase.
|
@ -0,0 +1,82 @@
|
||||
# ADR: Preferred data fetching method
|
||||
|
||||
## Background
|
||||
|
||||
We have found a need to standardise how we fetch data from APIs, in order to reduce complexity and simplify the data fetching process.
|
||||
|
||||
## Decision
|
||||
|
||||
We have decided to remove redux from our application and fetch all of our data via a third party library called `useSWR` (SWR stands for stale-while-revalidate and is a common cache strategy).
|
||||
|
||||
```tsx
|
||||
// Do:
|
||||
// useSegments.ts
|
||||
|
||||
import useSWR from 'swr';
|
||||
import { useCallback } from 'react';
|
||||
import { formatApiPath } from 'utils/formatPath';
|
||||
import handleErrorResponses from '../httpErrorResponseHandler';
|
||||
import { ISegment } from 'interfaces/segment';
|
||||
import useUiConfig from 'hooks/api/getters/useUiConfig/useUiConfig';
|
||||
import { IFlags } from 'interfaces/uiConfig';
|
||||
|
||||
export interface UseSegmentsOutput {
|
||||
segments?: ISegment[];
|
||||
refetchSegments: () => void;
|
||||
loading: boolean;
|
||||
error?: Error;
|
||||
}
|
||||
|
||||
export const useSegments = (strategyId?: string): UseSegmentsOutput => {
|
||||
const { uiConfig } = useUiConfig();
|
||||
|
||||
const { data, error, mutate } = useSWR(
|
||||
[strategyId, uiConfig.flags],
|
||||
fetchSegments
|
||||
);
|
||||
|
||||
const refetchSegments = useCallback(() => {
|
||||
mutate().catch(console.warn);
|
||||
}, [mutate]);
|
||||
|
||||
return {
|
||||
segments: data,
|
||||
refetchSegments,
|
||||
loading: !error && !data,
|
||||
error,
|
||||
};
|
||||
};
|
||||
|
||||
export const fetchSegments = async (
|
||||
strategyId?: string,
|
||||
flags?: IFlags
|
||||
): Promise<ISegment[]> => {
|
||||
if (!flags?.SE) {
|
||||
return [];
|
||||
}
|
||||
|
||||
return fetch(formatSegmentsPath(strategyId))
|
||||
.then(handleErrorResponses('Segments'))
|
||||
.then(res => res.json())
|
||||
.then(res => res.segments);
|
||||
};
|
||||
|
||||
const formatSegmentsPath = (strategyId?: string): string => {
|
||||
return strategyId
|
||||
? formatApiPath(`api/admin/segments/strategies/${strategyId}`)
|
||||
: formatApiPath('api/admin/segments');
|
||||
};
|
||||
|
||||
// Don't:
|
||||
const MyComponent = () => {
|
||||
useEffect(() => {
|
||||
const getData = () => {
|
||||
fetch(API_URL)
|
||||
.then(res => res.json())
|
||||
.then(setData);
|
||||
};
|
||||
getData();
|
||||
}, []);
|
||||
};
|
||||
```
|
||||
|
@ -0,0 +1,90 @@
|
||||
# ADR: Preferred data mutation method
|
||||
|
||||
## Background
|
||||
|
||||
Because our product is open-core, we have complexities and needs for our SaaS platform that are not compatible with the needs of our open-source product. We have found a need to standardise how we fetch data from APIs, in order to reduce complexity and simplify the data fetching process.
|
||||
|
||||
## Decision
|
||||
|
||||
We have decided to standardise data-fetching and error handling by implementing a top level `useAPI` hook that will take care of formatting the
|
||||
request in the correct way adding the basePath if unleash is hosted on a subpath, wrap with error handlers and return the data in a consistent way.
|
||||
|
||||
Example:
|
||||
|
||||
```tsx
|
||||
import { ITagPayload } from 'interfaces/tags';
|
||||
import useAPI from '../useApi/useApi';
|
||||
|
||||
const useTagTypesApi = () => {
|
||||
const { makeRequest, createRequest, errors, loading } = useAPI({
|
||||
propagateErrors: true,
|
||||
});
|
||||
|
||||
const createTag = async (payload: ITagPayload) => {
|
||||
const path = `api/admin/tag-types`;
|
||||
const req = createRequest(path, {
|
||||
method: 'POST',
|
||||
body: JSON.stringify(payload),
|
||||
});
|
||||
|
||||
try {
|
||||
const res = await makeRequest(req.caller, req.id);
|
||||
|
||||
return res;
|
||||
} catch (e) {
|
||||
throw e;
|
||||
}
|
||||
};
|
||||
|
||||
const validateTagName = async (name: string) => {
|
||||
const path = `api/admin/tag-types/validate`;
|
||||
const req = createRequest(path, {
|
||||
method: 'POST',
|
||||
body: JSON.stringify({ name }),
|
||||
});
|
||||
try {
|
||||
const res = await makeRequest(req.caller, req.id);
|
||||
return res;
|
||||
} catch (e) {
|
||||
throw e;
|
||||
}
|
||||
};
|
||||
const updateTagType = async (tagName: string, payload: ITagPayload) => {
|
||||
const path = `api/admin/tag-types/${tagName}`;
|
||||
const req = createRequest(path, {
|
||||
method: 'PUT',
|
||||
body: JSON.stringify(payload),
|
||||
});
|
||||
|
||||
try {
|
||||
const res = await makeRequest(req.caller, req.id);
|
||||
return res;
|
||||
} catch (e) {
|
||||
throw e;
|
||||
}
|
||||
};
|
||||
|
||||
const deleteTagType = async (tagName: string) => {
|
||||
const path = `api/admin/tag-types/${tagName}`;
|
||||
const req = createRequest(path, { method: 'DELETE' });
|
||||
|
||||
try {
|
||||
const res = await makeRequest(req.caller, req.id);
|
||||
return res;
|
||||
} catch (e) {
|
||||
throw e;
|
||||
}
|
||||
};
|
||||
|
||||
return {
|
||||
createTag,
|
||||
validateTagName,
|
||||
updateTagType,
|
||||
deleteTagType,
|
||||
errors,
|
||||
loading,
|
||||
};
|
||||
};
|
||||
|
||||
export default useTagTypesApi;
|
||||
```
|
37
website/docs/contributing/frontend/ADR/preferred-export.md
Normal file
37
website/docs/contributing/frontend/ADR/preferred-export.md
Normal file
@ -0,0 +1,37 @@
|
||||
# ADR: Preferred export
|
||||
|
||||
## Background
|
||||
|
||||
We have seen a need to standardize how to export from files in the project, in order to achieve consistency and avoid situations where we can have a component default exported as one name and renamed as something else in a different file. For example:
|
||||
|
||||
```
|
||||
// Problem example
|
||||
// File A
|
||||
|
||||
const MyComponent = () => {
|
||||
|
||||
}
|
||||
|
||||
export default MyComponent;
|
||||
|
||||
// File B
|
||||
import NewName from '../components/MyComponent/MyComponent.tsx';
|
||||
```
|
||||
|
||||
The above can cause massive confusion and make it hard to navigate the codebase.
|
||||
|
||||
## Decision
|
||||
|
||||
We have decided to standardise exports on named exports. This will allow us to eliminate the possiblity of exporting a component and renaming it in another file.
|
||||
|
||||
```jsx
|
||||
// Do:
|
||||
export const MyComponent = () => {};
|
||||
|
||||
// Don't:
|
||||
const MyComponent = () => {};
|
||||
|
||||
export default MyComponent;
|
||||
```
|
||||
|
||||
The reason for this decision is to remove mental clutter and free up capacity to easily navigate the codebase. If you can always deduce that the component is named as it is defined, then finding that component becomes a lot easier. This will ensure that we remove unnecessary hurdles to understand and work within the codebase.
|
@ -0,0 +1,30 @@
|
||||
# ADR: Preferred folder structure
|
||||
|
||||
## Background
|
||||
|
||||
Folder structure is important in how easy it is to navigate and reason about the codebase. It's important to have a clear structure that is easy to understand and follow, while grouping related files together in such a way that is easy to find and remove.
|
||||
|
||||
## Decision
|
||||
|
||||
We have decided to create tree-like folder structure that mimics as closely as possible the relationship of the React components in the project. This has a number of benefits:
|
||||
|
||||
* If you are looking for a component, you can easily find it by looking at the folder structure.
|
||||
* If you need to delete a component, you can be sure that all of the files connected to that component will be deleted if you delete the folder. This is supremely important, because it allows us to get rid of dead code easily and without having to worry about the consequences of deleting a file and worrying about whether it's used somewhere else.
|
||||
|
||||
## Folder structure example:
|
||||
|
||||
```
|
||||
ProfilePage
|
||||
ProfilePage.tsx
|
||||
ProfilePage.styles.ts
|
||||
ProfileSettings
|
||||
ProfileSettings.tsx
|
||||
ProfileSettings.styles.ts
|
||||
ProfilePicture
|
||||
ProfilePicture.tsx
|
||||
ProfilePicture.styles.ts
|
||||
```
|
||||
|
||||
Now you can clearly see that if you need to delete the `ProfilePage` component, you can simply delete the `ProfilePage` folder and all of the files connected to that component will be deleted.
|
||||
|
||||
If you experience that you need to create a component that is used in multiple places, the component should be moved to the closest possible ancestor. If this is not possible, the component should be moved to the `common` folder.
|
@ -0,0 +1,21 @@
|
||||
# ADR: Preferred function type
|
||||
|
||||
## Background
|
||||
|
||||
In the codebase, we have found a need to standardise function types in order to keep the codebase recognizible across different sections, and to encourage / discourage certain patterns.
|
||||
|
||||
## Decision
|
||||
|
||||
We have decided to use arrow functions across the board in the project. Both for helper functions and for react components.
|
||||
|
||||
```jsx
|
||||
// Do:
|
||||
const myFunction = () => {};
|
||||
const MyComponent = () => {};
|
||||
|
||||
// Don't:
|
||||
function myFunction() {}
|
||||
function MyComponent() {}
|
||||
```
|
||||
|
||||
The reason for this decision is to remove mental clutter and free up capacity to easily navigate the codebase. In addition, using arrow functions allows you to avoid the complexity of losing the scope of `this` for nested functions, and keeps `this` stable without any huge drawbacks. Losing hoisting is an acceptable compromise.
|
@ -0,0 +1,50 @@
|
||||
# ADR: Preferred styles import placement
|
||||
|
||||
## Background
|
||||
|
||||
SUPERSEDED BY [ADR: Preferred styling method](./preferred-styling-method.md)
|
||||
|
||||
In the codebase, we have found a need to standardise where to locate the styles import. When using CSS modules, the styles import placement matters for the priority of the styles if you are passing through styles to other components. IE:
|
||||
|
||||
```
|
||||
// import order matters, because the useStyles in MyComponent now
|
||||
// is after the useStyles import it will not take precedence if it has
|
||||
// a styling conflict.
|
||||
import useStyles from './SecondComponent.styles.ts';
|
||||
import MyComponent from '../MyComponent/MyComponent.tsx';
|
||||
|
||||
const SecondComponent = () => {
|
||||
const styles = useStyles();
|
||||
|
||||
return <MyComponent className={styles.overrideStyles} />
|
||||
}
|
||||
```
|
||||
|
||||
## Decision
|
||||
|
||||
We have decided to always place style imports as the last import in the file, so that any components that the file may use can safely be overriden with styles from the parent component.
|
||||
|
||||
```tsx
|
||||
// Do:
|
||||
import MyComponent from '../MyComponent/MyComponent.tsx';
|
||||
|
||||
import useStyles from './SecondComponent.styles.ts';
|
||||
|
||||
const SecondComponent = () => {
|
||||
const styles = useStyles();
|
||||
|
||||
return <MyComponent className={styles.overrideStyles} />;
|
||||
};
|
||||
|
||||
// Don't:
|
||||
import useStyles from './SecondComponent.styles.ts';
|
||||
import MyComponent from '../MyComponent/MyComponent.tsx';
|
||||
|
||||
const SecondComponent = () => {
|
||||
const styles = useStyles();
|
||||
|
||||
return <MyComponent className={styles.overrideStyles} />;
|
||||
};
|
||||
```
|
||||
|
||||
The reason for this decision is to remove the posibillity for hard to find bugs, that are not obvious to detect and that might be time consuming to find a solution to.
|
20
website/docs/contributing/frontend/overview.md
Normal file
20
website/docs/contributing/frontend/overview.md
Normal file
@ -0,0 +1,20 @@
|
||||
## Frontend overview
|
||||
|
||||
The frontend is written in react/typescript. It's is a single page application that communicates with the backend via a REST API. The frontend is built using vite and served by the backend.
|
||||
|
||||
## ADRs
|
||||
|
||||
We have created a set of ADRs to help guide the development of the frontend. These are located in the [ADR folder](./ADR).
|
||||
|
||||
* [Component naming](./ADR/component-naming.md)
|
||||
* [Interface naming](./ADR/interface-naming.md)
|
||||
* [Preferred component props usage](./ADR/preferred-component-props-usage.md)
|
||||
* [Preferred export](./ADR//preferred-export.md)
|
||||
* [Preferred function type](./ADR/preferred-function-type.md)
|
||||
* [Preferred style import placement](./ADR/preferred-styles-import-placement.md)
|
||||
* [Preferred styling method](./ADR/preferred-styling-method.md)
|
||||
* [Preferred data mutation method](./ADR/preferred-data-mutation-method.md)
|
||||
* [Preferred data fetching method](./ADR/preferred-data-fetching-method.md)
|
||||
* [Preferred folder structure](./ADR/preferred-folder-structure.md)
|
||||
|
||||
##
|
Loading…
Reference in New Issue
Block a user