Develop
Root Item
The first thing we'll want to do is to expose the root of our graph. Every connector has a root
item,
which defines the entry point of its graph. This root
item, having not been described by a schema, usually
only defines relations. In our case, our root
item will only link to one relation — our pokémons!
Open up the root
handler (src/handlers/root.ts), and modify it as follow;
import { GetItemHandler, Api } from '@unito/integration-sdk';
export const getItem: GetItemHandler = async () => {
return {
fields: {},
relations: [
{
name: 'pokemons',
label: 'Pokémons',
path: '/pokemons',
schema: {
label: 'Pokémon',
fields: [
{
name: 'id',
label: 'Unique ID',
type: Api.FieldValueType.INTEGER,
},
{
name: 'name',
label: 'Name',
type: Api.FieldValueType.STRING,
},
{
name: 'base_experience',
label: 'Base experience',
type: Api.FieldValueType.NUMBER,
},
],
}
}
],
};
};
With that done, start the CLI (integration-cli dev
), navigate to the debugger
tab, and step into your first item!
Pokémons List
Alright, our root
item specifies that we can find Pokémons at the /pokemons
path, but we haven't coded that yet!
First, create a new pokemons
handler (src/handlers/pokemons.ts), and modify it as follow;
import { GetCollectionHandler } from '@unito/integration-sdk';
export const getCollection: GetCollectionHandler = async () => {
return {
info: {},
data: [],
};
};
And then, open up the index
file (src/index.ts) and modify it as follow;
import { Integration } from '@unito/integration-sdk';
import * as meHandler from './handlers/me.js';
import * as rootHandler from './handlers/root.js';
import * as pokemonsHandler from './handlers/pokemons.js';
const integration = new Integration();
integration.addHandler('/', rootHandler);
integration.addHandler('/me', meHandler);
integration.addHandler('/pokemons', pokemonsHandler);
integration.start();
Alright, now we can step into our root
, and step again into our collection! It works, but it sadly sits empty for now.
Let's add some Pokémons!
Provider
Create a new provider.ts
(src/provider.ts) file and modify it as follow;
import { Provider } from '@unito/integration-sdk';
const provider = new Provider({
prepareRequest: _context => {
return {
url: 'https://pokeapi.co/api/v2',
headers: {}, // Here we would normaly provide the user's credentials in the provider's defined header of choice. Since the pokeapi is unauthenticated, we will leave empty
};
},
});
export default provider;
Now let's go back to our pokemons
handler, and modify it as follow;
import { GetCollectionContext, GetCollectionHandler, HttpErrors } from '@unito/integration-sdk';
import provider from '../provider.js';
export const getCollection: GetCollectionHandler = async (context: GetCollectionContext<{}, { offset?: string }>) => {
const offset = Number(context.query.offset) || 0;
const limit = 100;
const response = await provider.get<{ next: string; results: { name: string }[] }>('/pokemon', {
...context,
queryParams: { limit: limit.toString(), offset: offset.toString() },
});
return {
info: {
nextPage: response.body.next ? `/pokemons?offset=${offset + limit}` : undefined,
},
data: response.data?.results.map(pokemon => ({
path: `/pokemons/${pokemon.name}`,
})),
};
};
Try again the CLI, and you should see a list of Pokémons!
Single Pokémon
The only thing missing now is a way to retrieve a single Pokémon. For this, we'll need two things.
First, open up index.ts
(src/index.ts), we need to add a way for our handler's path to distingish between a
collection and an individual item. This is done by adding a path identifier to our path.
import { Integration } from '@unito/integration-sdk';
import * as meHandler from './handlers/me.js';
import * as rootHandler from './handlers/root.js';
import * as pokemonsHandler from './handlers/pokemons.js';
const integration = new Integration();
integration.addHandler('/', rootHandler);
integration.addHandler('/me', meHandler);
integration.addHandler('/pokemons/:name', pokemonsHandler);
integration.start();
And finally, from within the same pokemons
handler, add a new getItem
property as follow;
import { GetCollectionContext, GetCollectionHandler, GetItemHandler, GetItemContext, HttpErrors } from '@unito/integration-sdk';
import provider from '../provider.js';
export const getCollection: GetCollectionHandler = async (context: GetCollectionContext<{}, { offset?: string }>) => {
const offset = Number(context.query.offset) || 0;
const limit = 100;
const response = await provider.get<{ next: string; results: { name: string }[] }>('/pokemon', {
...context,
queryParams: { limit: limit.toString(), offset: offset.toString() },
});
return {
info: {
nextPage: response.data.next ? `/pokemons?offset=${offset + limit}` : undefined,
},
data: response.data?.results.map(pokemon => ({
path: `/pokemons/${pokemon.name}`,
})),
};
};
export const getItem: GetItemHandler = async (context: GetItemContext<{ name: string }>) => {
const response = await provider.get<{
id: number;
name: string;
base_experience: number;
}>(`pokemon/${context.params.name}`, context);
return {
fields: {
id: response.data.id,
name: response.data.name,
base_experience: response.data.base_experience,
},
relations: [],
};
};
And with that, magically, we should be able to crawl all of our Pokémons!
Canonical Path
Sometime, you might have access to a canonical path for your items. A canonical path is a path (or an id) that uniquely represent an item. It allows Unito to identify an item, even if it can be reached through multiple relative paths. For example, a user might be reachable through /container/users/1
, /project/users/1
, etc. In this case, if a canonical path exist in the provider it could be /users/1
. This route does not need to be exposed in the connector, but it must be provided in every route providing Item
or ItemSummary[]
in the response as it will be used by Unito to identify the item uniquely.
To provide a canonical path, add a canonicalPath
property to the getCollection
, getItem
, updateItem
and createItem
response.
return {
fields: {
id: response.data.id,
name: response.data.name,
base_experience: response.data.base_experience,
},
relations: [],
canonicalPath: '/user/1',
};