App creators can build apps in AppSheet using the data exposed by APIs managed by Apigee with the Apigee data source type.
Note: Both Apigee X and Apigee Edge are supported.
At this time, there are primarily two options to connect AppSheet apps to Apigee APIs:
- Manual
- OpenAPI Spec
This article covers the detailed steps and the prerequisites for the API necessary to set up the connection from the AppSheet apps to Apigee managed APIs.
- Quick start video
- Set up the Apigee data source for AppSheet
- Configure an OpenAPI Spec Apigee connection
- Configure a Manual Apigee connection
- Prerequisites for your API proxy
- Enable pagination (OpenAPI Spec Apigee connection only)
- Specify filter parameters (OpenAPI Spec Apigee connection only)
- Support multiple parameters in an API path (OpenAPI Spec Apigee connection only)
- Limitations and known issues
- Troubleshoot
Quick start video
Watch a video to learn how to use an Apigee data source in AppSheet.
API-powered app development with Apigee and AppSheet
Set up the Apigee data source for AppSheet
To set up the Apigee data source for AppSheet, perform the following steps:
Set up authentication in Apigee
To authenticate with an Apigee data source, you can choose one of the options described in the following table
Authentication |
Description |
OAuth |
Set up OAuth validation in Apigee to provide an authorized way to access your API. For more information, see:
Note: Currently the Apigee data source in AppSheet only supports the client credentials grant type. For more information, see:
|
API key |
Set up API key validation in Apigee to provide an authorized way to access your API. For more information, see:
Note: The API key is passed back in the |
No authentication |
Select API key and put any non-empty value (such as, |
Create the Apigee data source in AppSheet
To create the Apigee data source in AppSheet to create a connection to Apigee:
- Sign in to appsheet.com.
- Click My account in the top navigation bar.
- Select the Sources tab.
- Click + New Data Source.
- Name your new Data Source and select Apigee.
Add Apigee API connection information dialog
Configure an OpenAPI Spec or Manual Apigee connection, as described in the following sections.
Configure an OpenAPI Spec Apigee connection
If you have an OpenAPI Specification for your API proxy, you can configure an OpenAPI Spec Apigee connection.
To configure an OpenAPI Spec Apigee connection:
- In the Add Apigee API connection information dialog, click OpenAPI Spec.
- To use OAuth authentication, select OAuth as the Authentication Type and configure the following settings.
Setting
Description
Apigee Client Id
Client ID generated by Apigee.
Apigee Client Secret
Client secret generated by Apigee.
OpenAPI Spec URL
URL pointing to the OpenAPI 3.0 Spec. AppSheet will accept either JSON or YAML format. The URL must be publicly accessible as a REST endpoint.
Appsheet will send a
GET
call to the URL to get the OpenAPI Spec. - To use API key authentication, select ApiKey as the Authentication Type and configure the following settings.
Setting
Description
Apigee API Key
API key generated by Apigee.
OpenAPI Spec URL
URL to the OpenAPI Spec.
Custom Headers Custom headers to send with all requests to Apigee. Click Add and specify the header name and value. Repeat to add additional custom headers.
Note: You can add up to 20 custom headers. Custom header names and values cannot exceed 1000 characters each.
- To validate the connection, click Validate.
AppSheet makes aGET
call to the OpenAPI Spec URL and parses the OpenAPI Spec. Then, AppSheet makesGET
calls to each table resource after they are inferred from the OpenAPI Spec. - Click Authorize Access to add the data source.
Configure a Manual Apigee connection
If you don't have an OpenAPI Specification available for your API, configure a Manual Apigee connection.
To configure a Manual Apigee connection:
- In the Add Apigee API connection information dialog, click Manual.
- To use OAuth authentication, select OAuth as the Authentication Type and configure the following settings.
Setting
Description
Apigee Authorization Token URL
URL of your API proxy that handles the Access token generation.
Apigee Client Id
Client ID generated by Apigee.
Apigee Client Secret
Client secret generated by Apigee.
Authorization Scopes (Optional)
Scopes of your authorization if defined in your policy.
Apigee API Base Path
Base path of your Apigee proxy.
API Resource Paths
Comma-separated list of API proxy endpoints. The paths correspond to table names in Appsheet.
- To use API key authentication, select ApiKey as the Authentication Type and configure the following settings.
Setting
Description
Apigee API Key
API key generated by Apigee.
Apigee API Base Path
Base path of your Apigee proxy.
API Resource Paths
Comma-separated list of API proxy endpoints. The paths correspond to table names in AppSheet.
Custom Headers Custom headers to send with all requests to Apigee. Click Add and specify the header name and value. Repeat to add additional custom headers.
Note: You can add up to 20 custom headers. Custom header names and values cannot exceed 1000 characters each.
- To test the connection, click Test Connection.
- For OAuth authentication: AppSheet makes a
POST
call with Apigee Client Id and Apigee Client Secret to the Apigee Authorization Token URL and gets the access token. Then, using the access token AppSheet makes aGET
call to all of the{Apigee API Base Path}/{API Resource Path}
paths to verify access. - For API key authentication: AppSheet makes a
GET
call to all of the{Apigee API Base Path}/{API Resource Path}
paths to verify access.
- For OAuth authentication: AppSheet makes a
- Click Authorize Access to add the data source.
Prerequisites for your API proxy
Depending on the type of Apigee data source setup (Manual or OpenAPI Spec), there are some additional requirements for the underlying Apigee API proxy APIs to ensure compatibility with AppSheet.
OpenAPI Spec Apigee connection
When you configure an OpenAPI Spec Apigee connection, ensure that the Apigee API proxy supports the required API operations described in Required operations for the API proxy.
In addition, the OpenAPI Spec must conform to the following guidelines:
- Servers must have exactly one entry as a base path
- For the OAuth authentication type:
- Must include
components:securitySchemes
section - When the
components:securitySchemes
type isoauth2
, flows must be specified asclientCredentials
tokenUrl
must be providedscopes
is optional
Examplecomponents
snippet in OpenAPI Spec:components:
securitySchemes:
oAuth2ClientCredentials:
type: oauth2
Description: Experimental Apigee APIs
Flows:
clientCredentials:
tokenUrl: https://appsheet-test.apigee.net/oauth2/token/
scopes: #Can be empty {}
read: Grant read-only access - Must include
- If you provide the schema of the response body for
GET /tableName/{id}
, AppSheet will infer the column type from it. Otherwise, AppSheet will directly try to infer the column type from the actual response.
The OpenAPI Spec endpoint can be hosted within or outside Apigee. For example (SwaggerHub): Sample OpenAPI Specification
Replace the app.swaggerhub.com
with api.swaggerhub.com
to get the correct API-based URL.
x-apikey
when making a request to fetch the OpenAPI Spec.Manual Apigee connection
When you configure a Manual Apigee connection, ensure that the Apigee API proxy supports the required API operations described in Required operations for the API proxy.
Required operations for the API proxy
For each table, the following operations are required:
- Get all entries in a table
- Get an entry in a table (required for write operations)
The following API operations are optional:
Get all entries in a table (Required)
To get all entries in a table, the Apigee API proxy must support the following API operation:
GET /{tableName}
Where:
{tableName}
- Name of the table.
The response body consists of a JSON payload with the table name, for example "orders"
, as the key that includes the details for all rows in the table. For example:
{
"orders": [
{
"ID": 1,
"TextValue": "Order1",
"DecimalValue": 1.10,
"DateTime": "1/1/2016 1:01:01 AM",
"PhoneValue": "111 111 1111",
"EmailValue": "email1@appsheet.com"
},
{
"ID": 2,
"TextValue": "Order2",
"DecimalValue": 1.11,
"DateTime": "2/1/2016 1:01:01 AM",
"PhoneValue": "111 112 1111",
"EmailValue": "email2@appsheet.com"
}
]
}
Get an entry in a table (Required for write operations)
To get an entry in a table, the Apigee API proxy must support the following API operation:
GET /{tableName}/{id}
Where:
{tableName}
- Name of the table.{id}
- Unique ID for the table entry. This ID is mapped to the key column by the app creator in the editor, as described in Set the key and editable columns in AppSheet.
The response body consists of a JSON payload that includes the details for the selected row.
Create an entry in a table (Optional)
To support creating an entry in a table, the Apigee API proxy must support the following API operation:
POST /{tableName}
Where:
{tableName}
- Name of the table.{id}
- Unique ID for the table entry. This ID is mapped to the key column by the app creator in the editor, as described in Set the key and editable columns in AppSheet.
The request body contains the details for the row to be created. The columns specified in the payload must be marked as Editable by the app creator in the editor, as described in Set the key and editable columns in AppSheet.
Update an entry in a table (Optional)
To update an entry in a table, the Apigee API proxy must support the following API operation:
PUT /{tableName}/{id}
Where:
{tableName}
- Name of the table.{id}
- Unique ID for the table entry. This ID is mapped to the key column by the app creator in the editor, as described in Set the key and editable columns in AppSheet.
The request body contains the details for the row to be updated. The updated columns must be marked as Editable by the app creator in the editor, as described in Set the key and editable columns in AppSheet.
Delete an entry in a table (Optional)
To delete an entry in a table, the Apigee API proxy must support the following API operation:
DELETE /{tableName}/{id}
Where:
{tableName}
- Name of the table.{id}
- Unique ID for the table entry. This ID is mapped to the key column by the app creator in the editor, as described in Set the key and editable columns in AppSheet.
Set the key and editable columns in AppSheet
To support write operations on table entries, you must enable the following settings in the table columns:
- Key? - Uniquely identifies a table entry. See What is a key?
- Editable? Enables you to perform write operations on the column.
/{tableName}/{id}
with the request body. In the above example, id
is the key column. See Best practices for the id field. The fields in the request body will include the columns marked as Editable, such as id
, age
, and name
in the above example.
Best practices for the ID field
The preferred solution is to have one ID field generated by AppSheet using expressions like UNIQUEID()
. If the generation of the id field occurs in the backend:
- We recommend using two fields in your API payload that can act as the primary key for that entry. One generated and sent by AppSheet in the
POST
request body and the other one generated by the backend. - For the AppSheet-generated ID, you can use the expression
UNIQUEID()
as the initial value. - The backend-generated ID field will be left empty in the
POST
request body and will be populated on the next Sync after the creation of a new row.
The API should operate with the {id}
parameter input with either the AppSheet ID or the backend-generated ID. In other words, the ID matching needs to work with either of the column values. AppSheet will always use the AppSheet-generated ID for its API calls.
If your API can only support one ID field that is generated by the backend, then:
- AppSheet will need to generate a temporary unique ID in the API’s
{id}
field (use expressionUNIQUEID()
in the Initial Value setting for the API ID column). - When the
POST
operation is called, the backend will generate its own key for the entry, which will overwrite the value that AppSheet generated and passed in thePOST
body. - On the next Sync after the create (
POST
) operation is finished, AppSheet will retrieve the ID generated by the backend, and the temporary ID generated by AppSheet will be overwritten. - This can cause issues in cases where a row is created and updated (or deleted) before a Sync operation occurs. Since the operations are queued, id used for the update operation will be the temporary id generated by AppSheet and this will not exist in the backend.
Enable pagination
To enable pagination for your API, the OpenAPI Spec must specify an x-pagination
extension for the GET ALL
operation as an OpenAPI extension. The Apigee data source supports offset-based and cursor-based pagination directives.
General parameters:
- Type (required):
offset
|cursor
- limitParam (required): The param key in parameters object. Example:
limit
- defaultLimit: Default value of limit; if not specified, it will be set as 50
Offset-based parameters:
- offsetParam (required): The offset param key in parameters object
- defaultOffset: Default value of offset, if not specified, it will be set as 0
Cursor-based parameters:
-
cursorParam (required): The cursor param key in parameters object
Example: Offset-based parameters in OpenAPI Spec YAML
/users:
get:
tags:
- person
parameters:
- name: limit
in: query
required: false
schema:
type: integer
format: int64
- name: offset
in: query
required: false
schema:
type: integer
format: int64
responses:
'200':
description: Array of all person in the system with filter and pagination
content:
application/json:
schema:
type: array
items:
$ref: '#/components/schemas/Person'
x-pagination:
type: offset
limitParam: limit
defaulLimit: 30
offsetParam: offset
defaultOffset: 0
URL:
api/users?limit=30&offset=0
Example: Cursor-based parameters in OpenAPI Spec YAML
/users:
get:
tags:
- person
parameters:
- name: page_size
in: query
required: false
schema:
type: integer
format: int64
- name: page_token
in: query
required: false
schema:
type: integer
format: int64
responses:
'200':
description: Array of all person in the system with filter and pagination
content:
application/json:
schema:
type: array
items:
$ref: '#/components/schemas/Person'
x-pagination:
type: cursor
limitParam: page_size
defaulLimit: 30
cursorParam: page_token
URL:
api/users?page_size=30&page_token=GIYDAOBNGEYS2MBWKQYDAO
Specify filter parameters (query parameters/security filters)
To specify a filter for a table, the OpenAPI Spec must specify an x-filter
extension for GET ALL
operation as an OpenAPI extension and must specify a type of filter. Apigee data source only supports RHS-Colon
and LHS-Brackets
.
Example: RHS-Colon
/users:
get:
tags:
- publisher
summary: Get all person
responses:
'200':
description: Array of all person in the system with filter and pagination
content:
application/json:
schema:
type: array
items:
$ref: '#/components/schemas/Person'
'500':
description: Internal server error.
x-filter:
type: RHS-Colon
Make sure that you put a space in between the type: and the RHS-Colon
or LHS-Brackets
.
For example, the following syntax won’t work: type:RHS-Colon
Correct syntax with a space: type: RHS-Colon
The x-filter
directive only supports the operators in the table below. The table also shows the mapping between security filter resources and URL request (using age in the users table as example).
AppSheet’s security filter for the Apigee data source will be passed down to the API via these query parameters using the scheme (RHS-Colon
or LHS-Brackets
) as specified in the OpenAPI Spec. The operators supported are shown in the following table.
AppSheet Security Filter Operator |
AppSheet Security Filter example |
Filter operator |
RHS-Colon example |
LHS-Brackets example |
= |
[age]=25 |
eq |
api/users?age=eq:25 |
api/users?age[eq]=25 |
<> |
[age]<>25 |
neq |
api/users?age=neq:25 |
api/users?age[neq]=25 |
> |
[age]>25 |
gt |
api/users?age=gt:25 |
api/users?age[gt]=25 |
>= |
[age]>=25 |
gte |
api/users?age=gte:25 |
api/users?age[gte]=25 |
< |
[age]<25 |
lt |
api/users?age=lt:25 |
api/users?age[lt]=25 |
<= |
[age]<=25 |
lte |
api/users?age=lte:25 |
api/users?age[lte]=25 |
Support multiple parameters in an API path
The OpenAPI Spec Apigee connection supports multiple parameters in API paths that are defined in an OpenAPI Spec. For example:
/parentName/{parent}/tableName/{id}:
get:
parameters:
- name: parent
in: path
required: true
schema:
type: integer
format: int64
- name: id
in: path
required: true
schema:
type: integer
format: int64
...
The path parameters {parent}
and {id}
should map to valid columns in the GET /myTable
response:
- The primary key for
tableName
is inferred as{id}
. - The parent table key for
tableName
is inferred as the{parent}
. - The parent table must be defined in the same OpenAPI Spec. For example, both
/publishers/{id}
and/publishers/{publisher}/books/{id}
must be defined.
If you use a child table to create an app using an OpenAPI Spec with two path parameters, the Apigee data source will generate two tables (one for the parent and one for the child table) and automatically set up a reference column between the tables. In this case, the parent table must be defined in the same OpenAPI Spec. For example, both /publishers/{id}
and /publishers/{publisher}/books/{id}
must be defined in the spec.
-
Currently, the Apigee data source only supports two path parameters in an OpenAPI spec.
-
Each OpenAPI Spec path must adhere to the OpenAPI Spec 3.0 requirements for the Apigee data source in AppSheet.
Limitations and known issues
The following lists the limitations and known issues:
- Row limits are not currently supported. Use filter params to limit the data returned.
- Response types of all APIs are expected to be
application/json
. - During create an entry in a table operation, ID values should be generated inside AppSheet as a best practice. See Best practices for the ID field.
- The path parameters should always be at the end of the path string.
- Only the REST operations defined in Required operations for the API proxy are supported.
- The fields in the responses should not be nested, the fields should be of simple types like int, string, or list that can be converted into AppSheet data column types.
The OpenAPI Spec-based Apigee connection doesn’t strictly adhere to OpenAPI:
- Doesn’t use resource definition in OpenAPI Spec, uses superset of responses from
GET /resources
to build columns. - AppSheet assigns emails and phone number data types as required during initial data source creation. The App creator can adjust data after the table is created in the app.
- Doesn’t recognize the security schemes for API Key. AppSheet sends the API Key as
x-apikey
header only.
Troubleshoot
You may encounter the following exceptions when creating and testing your app. Examples and possible solutions are provided.
Your client credentials are invalid. Please ensure your Apigee client credentials and token url are correct.
Ensure your credentials are correct. For the OAuth type, you can verify your credentials by calling the following curl command. (curl is available on Linux-based systems. For MS DOS, you may need to install a third-party curl program). If you need assistance, contact an Apigee administrator, developer, or IT staff.
curl -X POST -H "Content-Type: application/x-www-form-urlencoded" \
“tokenUrl” \
-d "client_id={key}&client_secret={secret}"
You should get a 200
response with access_token
in the response body. For example:
{
...
"token_type" : "BearerToken",
"client_id" : "xNnREu1DNGfiwzQZ5HUN8IAUwZSW1GZW",
"access_token" : "GTPY9VUHCqKVMRB0cHxnmAp0RXc0",
"refresh_token_expires_in" : "30000", //--in seconds
...
}
Then, call the API with the access token:
curl https://{org-name}-test.apigee.net/tablename \
-H "Authorization: Bearer {access-token}"
Parsing of OpenAPI Spec failed. Error: The Apigee data source requires both the `GET /tableName` and 'GET /tableName/id'
Make sure every table in the spec includes both GET tableName
and GET /tableName/id
operations. For more details, see Prerequisites for your API proxy.
Error: Table 'table' is no longer connected to its schema/column structure.
If the Apigee data source successfully passes the test step and is authorized, but shows this error while creating the app, check the response format of the GET tableName
. This API is expected to return a list of rows with the key for this list as the table name/proxy endpoint. For more details, see Prerequisites for your API proxy.