Datasources & Dataformat
APICHAP differs datasources and dataformat. The concept of a datasource is the type of system where to get data from. The dataformat on the other hand is the format that data is received from a datasource.
For example: If you are calling an API (A Stateless GET request) to recieve a JSON entity, the Datasource is API and the Dataformat is JSON.
In another case you might need to create an XML file on an FTP Server. Meaning your datasource is FTP-SERVER and the Dataformat is XML.
Most of our datasources types and dataformats can be mixed up to meet your needs.
Datasources
APICHAP is designed to work with multiple datasources. This means that you can get, write and manage data for multiple systems.
| Type | Information | 
|---|---|
| Database | Connect to a SQL database. | 
| NoSQL Database | Connect to a No-SQL database. | 
| API | Send and retrieve data via request to a 3rd party | 
| FTP-Server | Read, Write and Manage files on an ftp server. | 
Roadmap: We are constantly expanding the list of supported datasources and datatypes. Connect us if you are missing one: Conteact.
Whenever you want to read, write or manage data in a fulfillment you must specify a datasource name. That datasource must be 
specified in the datasources tag. Depending on the type of your datasource it has different configuration options.
Database
APICHAP currently supports the following databases:
- mariadb (You are able to connect with a mysql datbase by using a mariadb URL in your connector)
- postgres
- TSQL (Microsoft SQL, Azure SQL)
A database datasource has the following configuration options:
| Field | Description | |
|---|---|---|
| type | Specifies the type of datasource. Use the value dbfor database. | Mandatory | 
| url | The JDBC URL for the datasource. In this case, the URL is "jdbc:mariadb://host.docker.internal:3306/sample-products". | Mandatory for db | 
| user | The username for the datasource. | Optional, user can also be part of the url | 
| password | The password for the datasource. | Optional, password can also be part of the url | 
datasources:
  demodb:
    type: db
    url: jdbc:mariadb://host.docker.internal:3306/sample-products
    user: testuser
    password: samplepwd
NoSQL-Database
Coming Soon
API
Specifying an API as a datasource. Right now an API datasource does not have any additional configuration options.
| Field | Description | |
|---|---|---|
| type | Specifies the type of datasource. Use the value apifor APIs. | Mandatory | 
FTP-Server
Specify the connection to a FTP-Server to read, write and move file.
| Field | Description | |
|---|---|---|
| type | Specifies the type of datasource. Use the value ftpfor FTP-Server. | Mandatory | 
| host | The hostname of the FTP-Server. | Mandatory | 
| port | The port of your FTP-Server. | Optional | 
| user | The user. | Mandatory | 
| password | The password. | Mandatory | 
datasources:
  myftp:
    type: ftp
    host: The ftp hostname
    port: 21
    user: The ftp user
    password: ${ENV(FTP_PWD)}  # This uses a environment variable feature for sensitive data.
Since your API is shipped in a container we are only working with FTP passive-mode. If your FTP server blocks passive-mode the connection will not work.
Dataformat
The supported dataformats are:
| Type | Content-Type | 
|---|---|
| JSON | application/json | 
| XML | text/xmlorapplication/xml | 
| CSV | text/csv | 
| PLAIN-TEXT | text/plain | 
| SPREADSHEET | application/vnd.openxmlformats-officedocument.spreadsheetml.sheetorapplication/vnd.ms-excel | 
No matter which format, the configuration is always very similar.
Read: Whenever you receive data from one of the supported dataformats they are automatically transformed into a JSON format. Therefore you will always be able to receive data via the common VALUE() method using the JSONPath syntax.
Write:
No matter which dataformat you want to write data in. The syntax will always work very similar, by creating what we call a schema.
To learn how you can build a schema take a look at Schema Generation.
However, some of the dataformats have a few special rules on how to build a schema.
JSON
- application/json
This schema:
schema:
    person:
        item:
            firstname: TEXT(Maria)
            lastname: TEXT(Lee)
            countries_visited: 
                list: LIST_BY_VALUES(Austria,Japan,Kenya)
                item: 
                    name: VALUE(@, $)
{
  "person": {
    "firstname": "Maria",
    "lastname": "Lee",
    "countries_visited": [
      {
        "name": "Austria"
      },
      {
        "name": "Japan"
      },
      {
        "name": "Kenya"
      }
    ]
  }
}
XML
- text/xml
- application/xml
Building an XML schema is similar to building a JSON schema, but there is one important note:
- The rootElementis always the first key of the schema. This means that XML files must always start with a single root object.
Currently, XML attributes are not supported.
schema:
    person:
        item:
            firstname: TEXT(Maria)
            lastname: TEXT(Lee)
            countries_visited: 
                list: LIST_BY_VALUES(Austria,Japan,Kenya)
                item: 
                    name: VALUE(@, $)
<?xml version='1.0' encoding='UTF-8'?>
<person>
    <firstname>Maria</firstname>
    <lastname>Lee</lastname>
    <countries_visited>
        <name>Austria</name>
        <name>Japan</name>
        <name>Kenya</name>
    </countries_visited>
</person>
Internally XML content will be transformed into a JSON format and is then accessible using the VALUE(ffName, JSONPath to field) method.
```JSON Output [{ "firstname": "Maria", "lastname" : "Lee", ... }]
### CSV <a name="section-csv"></a>
- `text/csv`
By design CSV files can only have a single level of data, meaning it is not possible to create nested objects in them. The same applies for 
the CSV data you are building with APICHAP. It only allowes a single level of data variables.
> The column seperator can be defined in the instruction. See [details](../fulfillments/#section-file)
```YAML 
schema:
    firstname: TEXT(Maria)
    lastname: TEXT(Lee)
    countries_visited_1: TEXT(Austria)
    countries_visited_2: TEXT(Japan)
    countries_visited_3: TEXT(Kenya)
    countries_visited_4: NULL()
firstname;lastname;countries_visited_1;countries_visited_2;countries_visited_3;countries_visited_4;
Maria;Lee;Austria;Japan;Kenya;null;
Internally CSV content will be transformed into a JSON format and is then accessible using the VALUE(ffName, JSONPath to field) method.
```JSON Output [{ "firstname": "Maria", "lastname" : "Lee", ... }]
> Spaces within column names will be replaced with a _ character. Meaning a column named 'First Name' would become 'First_Name' and is then readable using 
> the VALUE(ff1, $[0].First_Name) method.
### Plain Text <a name="section-plain"></a>
- `text/plain`
If you are only working with unknown or text data use the `plain/text` dataformat. This can be used to for example read a txt file uploaded 
to your server.
For writing plain data your `schema` is only allowed to have a single field: `plain`.
```YAML 
schema:
    plain: TEXT(Hello World)
SPREADSHEET
- application/vnd.openxmlformats-officedocument.spreadsheetml.sheetfor .xlsx files
- application/vnd.ms-excelfor .xls files
APICHAP supports to read spreadsheets in .xls and .xlsx format in two different mode. - Parsed (default) - RawMode
When choosing normal you´re spreadsheet must have columnNames in the first rows and content in the rows afterwards. We don´t support empty lines or calculated values using excel formulas.
- request:
    url: https://myexcel.com?myfile.xlsx
    contentType: application/vnd.openxmlformats-officedocument.spreadsheetml.sheet
    rawMode: false
Using RawMode means that the excel is not parsed into a JSON like seen above, but you are able to get every cell on its own, 
by using the excel Cell identifiers (A1, C3, AB5, ...)
```YAML
SPREADSHEET(fulfillmentName, CellIdentifier)
SPREADSHEET(excel_download, A3)
Writing into Excel files is currently not supported.