ashpack-soil
v0.0.280
Published
SOIL is a strongly-typed opinionated wrapper around Firebase's Real-Time Database meant to supercharge Firebase and allow for better relational management and out-of-the-box security rules. It comes with a lot of other features as well, such as type-safet
Downloads
115
Readme
SOIL Documentation
SOIL is a strongly-typed opinionated wrapper around Firebase's Real-Time Database meant to supercharge Firebase and allow for better relational management and out-of-the-box security rules. It comes with a lot of other features as well, such as type-safety, easy authentication, and solutions for pagination, infinite scroll, and more as built-in results to using database keys effectively.
Lastly, the package includes a nearly exhaustive list of helper functions, hooks, context generators, and components that make working with soil and building a serverless architecture nearly effortless.
This project was the brainchild of @dmurawsky and was given lots of love by him and a co-developer to flesh out the ecosystem that makes working with soil effective and enjoyable.
Initial Setup
Pull in the SoilContextProviderComponent
and use it like so:
<SoilContextProviderComponent firebaseOptions={FIREBASE_OPTIONS}>
{yourAppHere}
</SoilContextProviderComponent>
This will initialize the database connection.
Authentication
We provide authentication helper functions which will work once nested in the SoilContextProviderComponent
. A good place to start is to simply use the signUp
and signIn
functions, but check here for more:
ashpack-soil/services/auth.ts
An even faster approach would be to use the given components:
ashpack-soil/components/SignUp/index.tsx
ashpack-soil/components/SignUp/index.tsx
Data Types
The first step to building a project with soil is to think about your data types and begin building them out. To do this, navigate from the root to @types/ashpack-soil/index.d.ts
and paste the following:
import * as soilTypes from "ashpack-soil";
declare module "ashpack-soil" {
export interface SoilDatabase extends soilTypes.SoilDatabase {
/** Example JS Doc explaining the `manager` dataType */
manager: { firstName: string; lastName: string };
}
}
This will set soil up to receive the proper types. Now begin building out your types, likely importing them from another file and assigning them to the relevant keys here. For example, you might have a data type for each of the following: manager
, user
, profile
, project
, and message
.
The structure that will be saved is as follows: data/user/{uid}/userData
where userData
will include the information provided in your type plus a few soil-things (such as permissions and timestamps).
Data Keys
A data key is simply the key for accessing a particular instance of a data type. For example, the uid of a user is the data key for the user data type. You can use Firebase Push Keys for your data keys, but when possible, Soil really shines when you get creative with your keys. For example, the data key to a message could be {userId}__{projectId}__{pushKey}
. In this way, without even fetching the message data, with the key alone you would be able to fetch the user who created the message and the project that the message is under.
To generate keys like this, use the generateDbKey
function, and the parse the same, use parseDbKey
:
const key = generateDbKey("{userId}", "{projectId}", "{pushKey}");
const [userId, projectId, pushKey] = parseDbKey();
This is part of why the JS Docs are so important. Make sure that each data type has a doc explaining how it is keyed and any other useful information, such as what it will be connected to and its role in the database architecture.
Connection Data Lists (CDL)
Connecting data together is how we make Soil relational. Any time data is created or updated (using createData
or updateData
) you can pass in a list of connections. You can also directly call createConnection
. This creates a two-way connection between two data types, for example, between a user and all of the projects that user is managing.
As such, you can use soil functions to then get all of the connected data of a particular thing. For example, when you render a user's project list, you simply fetch all the connected projects' data for that user. Inversely, for a project, you could fetch all connected users' data.
If you want to create a unique list of existing data of any kind, you can do so by adding that list as an empty object in the types file like so:
import * as soilTypes from "ashpack-soil";
declare module "ashpack-soil" {
export interface SoilDatabase extends soilTypes.SoilDatabase {
manager: { firstName: string; lastName: string };
user: { email: string; };
favorite: {};
}
}
For example, if you have all of your users under the user
data type but want to save a specific sublist of users, you would make a favorite
data type that does not directly have any data because it is simply a list referencing the user data type. You would create a connection between a manager
and a favorite
using the {uid}
of the favorited user
. You would then be able to save, fetch, and remove favorited users of the manager this way, simply by creating and removing connections between manager
and favorite
via user
{uid}
.
User Data Lists (UDL)
User data lists are similar to connection data lists, but they are less flexible and specifically connect a user to a piece of data (rather than a piece of data to another piece of data). This exists to tie into the security system. Just like you can set connections
, you can also set owners
, and the owner of a piece of data has priviledges to modify that data.
NOTE: User data lists may end up being deprecated. If so, you would rely on connections
.
Security Access
In order to read data:
- The the user must either own the data or be connected to the data or...
- The data must be set as
publicAccess: true
or have the appropriateconnectionAccess
.
We should explain the connectionAccess
, so an example is provided below:
await createData({
data,
dataKey,
dataType: "project",
owners: [managerKey],
connectionAccess: {
connectionKey: managerKey,
connectionType: "manager",
uidDataType: "user",
read: true,
write: true,
},
});
Here, we are saying that any user
that is connected to the manager
that owns this project
also has access to read and write this project
.
Once/Get/On/Use
Firebase has a terminology which we have tried to extend: onValue
and onceValue
. The on
represents an open data connection using Firebase's built-in websockets. Use this when you want a live subscription to the data. This is often likely the default. Sometimes, though, that is unnecessary. If you only need to fetch the data as in a normal CRUD API, you use once
.
We have mostly extended this terminlogy, but we sometimes use (1) get
instead of once
and (2) use
instead of on
for reasons explained in the following section.
CRUD Operations
There are too many helper functions to mention here. It suffices to say that if there is a createData
and an updateData
, there is also a removeData
. If there is a createConnection
, there is a removeConnection
. IDE auto-complete and common sense are your allies in this regard.
When you want to fetch a piece of data, you can call getDataKeyValue
. If you want to subscribe to a piece of data, you can call useDataKeyValue
. The reason we use the terminology use
here instead of once
is because this is a custom hook. In order for the subscription to work, it needs a useEffect
and a useState
.
Nonetheless, all of this is obsfucated away from the developer who can simply reach for the data as they need it. Just be sure to check what is happening under-the-hood to make sure you aren't doing something otherwise silly, like calling a custom hook within a useEffect
.
Helper Functions & Hooks
Take a moment to skim these two files in order to understand the underlying functions for working with a Soil Database:
ashpack-soil/services/client-data.ts
ashpack-soil/services/server-data.ts
Feel free to look at ashpack-soil/hooks
, but you will probably mainly use the following to start:
useDataKeyValue
useGetDataKeyValue
useConnectionsTypeData
Context Generators
One of the neatest features that Soil provides is a way to quickly build out your global state using context generaters. See:
ashpack-soil/context
There are two likely to be used the most:
createConnectionsTypeDataContext
createConnectionsTypeConnectionsContext
Below is the example usage of createConnectionsTypeDataContext
:
import { createConnectionsTypeDataContext } from "ashpack-soil/context/createConnectionsTypeDataContext";
const {
useConnectionsTypeDataContext: useManagerUsersContext,
ConnectionsTypeDataContextProviderComponent: ManagerUsersContextProviderComponent,
} = createConnectionsTypeDataContext("manager", "user");
export { useManagerUsersContext, ManagerUsersContextProviderComponent };
<ManagerUsersContextProviderComponent>
{globalContextWrappedAppHere}
</ManagerUsersContextProviderComponent>
const { dataArray: usersConnectedToTheManager } = useManagerUsersContext("{managerUid}")
The optional argument initialChildEqualTo
(unused in the example) in the useManagerUsersContext
call is meant to speed up the initial fetch with a Firebase Query. See the relevant createConnectionsTypeDataContext
code and Firebase query documentation for more details.
Components
There are a few components that we have provided, they can be found in:
ashpack-soil/components
The one which helps with infinite scroll is the DataInViewItem
. There is another component, pending entry into Soil, which makes this truly powerful and easy to use. It is called the ConnectionsObserverHOC
and is used like so:
<ConnectionsObserverHOC
listItemMinHeight="80px"
listItemMinWidth="80px"
className={styles.productList}
version="connectionDataList"
parentDataType="manager"
parentDataKey={managerKey}
dataType="user"
sort="created newest"
ItemComponent={UserInView}
EmptyComponent={NoUsers}
/>
It has not yet been brought into the package, but look for it soon!
Impactful Conventions
There are a few conventions we employ that have real consequences in using Soil.
Initialization and Falsey Keys in Helpers
When using a helper function like useDataKeyValue
, if the dataKey
is falsey (an empty string), the fetch will not be made at all. This is very useful. For example, if you are hydrating data on load, it will not bother to fetch the manager's users until the manager has been logged in and we have retrieved the manager's data key. Furthermore, you can control this directly with a third boolean argument: initialized
.
Null vs Undefined
Note, passing undefined
to Firebase at any time will break. Firebase only allows null
, which will delete the data at that location. Therefore, your types won't perfectly reflect the reality that: Firebase may return undefined
to you when nothing is at a target database location but will break if you pass undefined
to a database location. Similarly, you may pass null
to a database location to delete that data, but if you fetch that location, you will get undefined
rather than null
. This is a Firebase quirk to be aware of. Also note, when deleting data key values (an instance of a data type, ie. a user), do not use null
, use the given Soil function: removeData
.
However, even though Firebase returns undefined
in such cases, most Soil helper functions override this with null
. We consider undefined
values to mean that the fetching has not yet been done and null
values to mean that the fetching was done but nothing was returned. In this way, not only do we harmonize Firebase's undefined
/null
descrepancy, but also we enable the developer to know if it is loading or loaded regardless of whether or not there was a valid resource at the target location just by checking the value of undefined
vs null
.