Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add getServerSideProps docs to data-fetching markdown #73

Merged
merged 4 commits into from
Jul 12, 2024
Merged
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
155 changes: 147 additions & 8 deletions docs/reference/data-fetching.md
Original file line number Diff line number Diff line change
Expand Up @@ -2,15 +2,28 @@

Getting content and data into your React modules partials can take many forms as the sources are varied and nuanced.

## HubSpot Content - Server Side
## Server Side

In an ideal world the HubSpot GraphQL integration would be the go to for getting all of your HubSpot content into React components. Currently however, GraphQL only supports querying HubDB and Custom Objects - refer to the [GraphQL](#graphql) documentation above. There are some key advantages to using the GraphQL integration with React modules.
### GraphQL

GraphQL (pro/enterprise) currently exposes the following internal HubSpot data:

* HubDB
* CRM Objects
* Blog
* *Coming Soon* - Knowlege Base

Visit https://app.hubspot.com/graphiql/[portal-id] to explore your schema.

In an ideal world the HubSpot GraphQL integration would be the go to for getting all of your HubSpot content into React components. Currently however, GraphQL only supports what is listed above - refer to the [GraphQL](./js-modules#graphql) documentation. There are some key advantages to using the GraphQL integration with React modules.
bmatto marked this conversation as resolved.
Show resolved Hide resolved

- Co-located Query and Component
- One single Query for needed associations e.g. contact->company
- Tight coupling with prerendering: updates to the query or relevant data will update any pages containing the query

That said there are other kinds of HubSpot data you might want access to within your components. The best way to accomplish this today is by passing that information from a HubL template to a React component via the `js_partial` or `module` HubL tags. For example:
### hublParameters

At the template level you can pass information accessed in the HubL context to your React Module. Within your react component you can access this information via `props.hublParameters`.

```handlebars
{% module "contact_profile"
Expand All @@ -37,14 +50,140 @@ export const Component = (props) => {

Whether you are passing data via the HubL tags or querying via GraphQL these solutions account only for reading data, not for creating or updating data in your HubSpot portal. React modules and partials today don't offer any new avenues for manipulating your HubSpot Data.

## HubSpot Content - Client Side
### hublDataTemplate

`hublParameters` does not work for cases where modules are added to the page by a marketer via DnD. For this use case there `hublDataTemplate` can be leveraged. See [hublDataTemplate](./js-modules#hublDataTemplate) for more information

### Server Data Fetching with getServerSideProps

"Server Data Fetching" (pro/enterprise) allows developers to export a function `getServerSideProps` from their CMS React Module definition. `getServerSideProps` must return an object with a `serverSideProps` property and a `cacheConfig` property which configures caching of the module. In the React component the information returned in `serverSideProps` can be accessed via `props.serverSideProps`.

#### Dependency Helpers

In Data Fetching scenarios, you often need to fetch specific data based on various dependencies such as URLs, query parameters, or the HubSpot Contact object. The utility functions `withModuleProps`, `withUrlPath`, `withUrlAndQuery`, and `withContact` help wrap your data-fetching functions and automatically inject relevant dependencies (with TypeScript types). These helpers ensure that your module has the necessary context to fetch and process data effectively.

##### `withModuleProps`

**Purpose:**
Wraps a function to provide module properties without any additional dependencies. Access to `fieldValues`, `hublData`, `dataQueryResult` etc is available.

**Usage:**

```typescript
import { withModuleProps } from 'path/to/helpers';

const fetchData = (props: ModulePropsWithoutSSP) => {
// Your data fetching logic
};

export const getServerSideProps = withModuleProps(fetchData);
```

##### `withUrlPath`

**Purpose:**
Wraps a function to provide the module properties along with a URL without query parameters. Everything that was present in `withModuleProps`, plus the page URL without the query parameters. Caching at the module level is partly based on the props and dependencies used in data fetching. Specifying you only need the URL and not the Query can optimize that caching as it will not create new cache records for every query param variation.

**Usage:**

```typescript
import { withUrlPath } from 'path/to/helpers';

const fetchData = (props: ModulePropsWithoutSSP, { url }: { url: URLWithoutQuery }) => {
// Your data fetching logic
};

export const getServerSideProps = withUrlPath(fetchData);
```

##### `withUrlAndQuery`

**Purpose:**
Wraps a function to provide the module properties along with a URL including query parameters. Building on `withUrlPath`, the passed extra dependency will have the query parameters as well. As stated above a new cache record will be created for each permutation the url with query for this module.

**Usage:**

```typescript
import { withUrlAndQuery } from 'path/to/helpers';

const fetchData = (props: ModulePropsWithoutSSP, { url }: { url: URL }) => {
// Your data fetching logic
};

export const getServerSideProps = withUrlAndQuery(fetchData);
```

##### `withContact`

**Purpose:**
Wraps a function to provide the module properties along with a URL (with query parameters) and a contact. This will pass the url, with the query parameters, and the HubSpot Contact object.

**Usage:**

```typescript
import { withContact } from 'path/to/helpers';

const fetchData = (props: ModulePropsWithoutSSP, { url, contact }: { url: URL; contact: Contact }) => {
// Your data fetching logic
};

export const getServerSideProps = withContact(fetchData);
```

#### How It Works

Each of these helper functions wraps your data-fetching function and injects the relevant dependencies based on the type of data you need to fetch. This process ensures that your function has the necessary context and dependencies to operate correctly, streamlining your data-fetching logic and maintaining consistency across your modules.

For example, when you use `withUrlPath`, the wrapped function will receive a URL without query parameters, making it easy to fetch data based on the path alone. Similarly, `withContact` ensures that your function has access to both the URL, query, and contact information, allowing for more complex data-fetching scenarios.

#### Caching

Under the hood, the use of `getServerSideProps` introduces a new architecture which creates a cache between our edge CDN and data center where we render the React Modules. This caching strategy is outside our current [prerendering strategy](https://developers.hubspot.com/docs/cms/developer-reference/cdn/prerendering). This means that other parts of the page beside the module can be statically prerendered and the module itself can always be dynamic or cached by caching rules the developer defines. We have set a default 10 second cache (`Cache-control: max-age=10`) for these Data Fetching modules.

Cache "keys" are based on the following:

* Project Build Number
* Module Props
* This includes fieldValues, hublData, dataQueryResult etc..
* Injected dependency values
* For example if `withUrlPath` is used a unique cache key will be created each time the module is rendered within a page of a new URL path.

Knowing this, if you hadn't made changes to any data flowing into the module, but still wanted to bust the cache, a new project build would suffice.

The `cacheConfig` property that is returned from `getServerSideProps` currently has one property `cacheControl`. This represents the `Cache-Control` header, and the properties can be any of the standard directives.

```typescript
import { withModuleProps } from 'path/to/helpers';

const fetchData = async (props: ModulePropsWithoutSSP) => {
// Your data fetching logic
const results = await fetch(...).then(response => response.json())

return {
serverSideProps {
results,
cacheConfig: {
cacheControl: {
maxAge 60
}
}
}
}
};

export const getServerSideProps = withModuleProps(fetchData);
```

In the above example the module will be cached for 60 seconds, and any request after that will trigger a re-cache.

In local development the module is rendered on the developers machine, and so no caching is at play.

As was the case without CMS React components, you can make use of public APIs to fetch your HubSpot data from the browser. While these new components don't offer any HubSpot specific tools for data fetching on the client, we think the introduction of [Islands](#islands) will allow for more optimized and ergonomic client side data fetching. Relative to updating your HubSpot data - the recommended path would still be to implement a [Serverless Function](https://developers.hubspot.com/docs/cms/data/serverless-functions) that is responsible for securely making calls to HubSpot APIs. The serverless function would then expose an endpoint to respond to requests to from the client.
## Client Side

## External Content - Server Side
### HubSpot Data

While there is no pathway for this currently, our goal is to open up a pathway for developers to make asynchronous requests dynamically at render time, including server-side API requests to third-party services. In addition to potential performance benefits, with server-side data fetching a developer can make requests that require secrets or authentication safely. We do not currently have a timeline for when this will be available.
As was the case without CMS React components, you can make use of public APIs to fetch your HubSpot data from the browser. While these new components don't offer any HubSpot specific tools for data fetching on the client, we think the introduction of [Islands](./islands) will allow for more optimized and ergonomic client side data fetching. Relative to updating your HubSpot data - the recommended path would still be to implement a [Serverless Function](https://developers.hubspot.com/docs/cms/data/serverless-functions) that is responsible for securely making calls to HubSpot APIs. The serverless function would then expose an endpoint to respond to requests to from the client.

## External Content - Client Side
### External Content

Similar to the client side HubSpot Content scenario there is no real "change" in in terms of what is possible for fetching data on the client.
Loading