# Base swaps contract v1 specification
This contract allows two parties to safely exchange their FA2 tokens (assets).
Several types of tokens can participate in exchange on each side.
Table of contents:
[Toc]
## State machine diagram
This contract may contain a multitude of swaps, each of them follows the life-cycle presented in the diagram:
```mermaid
stateDiagram-v2
[*] --> InProgress : start
InProgress --> Cancelled : cancel
InProgress --> Finished : accept
Cancelled --> [*]
Finished --> [*]
```
## Types
The types that are following will be given in cameligo.
### Basic types
A `swap_id` type represents identifier assigned to a single exchange event.
```ocaml
type swap_id = nat
```
A `fa2_tokens` type represents several FA2 tokens of specific type.
```ocaml
type fa2_tokens =
[@layout:comb]
{ token_id : nat
; amount : nat
}
```
A `fa2_assets` type represents a list of tokens within one FA2 contract.
```ocaml
type fa2_assets =
[@layout:comb]
{ fa2_address : address
; tokens : fa2_tokens list
}
```
Note that we expect that provided list of tokens will be grouped by FA2 contracts the tokens belong to. Despite the more complicated interface, this lets us require fewer `get_entrypoint_opt` calls and make use of FA2 batched transfers and thus be much more efficient.
A `swap_status` type represents the current state of the swap.
```ocaml
type swap_status =
[@layout:comb]
| Open
| Finished of address
| Cancelled
```
To keep more info about the deal on-chain, we make `Finished` state contain the address of the participant that accepted the swap.
==Thus cancelled/accepted swaps are still present in the big_map, with the respective status.==
A `swap_offer` type contains all the data provided at swap creation.
```ocaml
type swap_offer =
[@layout:comb]
{ assets_offered : fa2_assets list
; assets_requested : fa2_assets list
}
```
A `swap_info` type represents all the full information stored about a single swap.
```ocaml
type swap_info =
[@layout:comb]
{ swap_offer : swap_offer
; seller : address
; status : swap_status
}
```
### Storage
Must contain `%swaps (big_map nat swap_info)`.
## Entrypoints
Notes:
* By *transfer* (in the context of FA2 tokens) we mean "performing a zero-xtz transaction that invokes FA2 `transfer` entrypoint".
* Entrypoints that perform FA2 transfer may throw the respective errors enlisted in [FA2 interface](https://gitlab.com/tzip/tzip/-/blob/master/proposals/tzip-12/tzip-12.md#error-handling).
### Start swap
```ocaml
| Start of swap_offer
```
Proposes an exchange.
This transfers the `assets_offered` tokens from the `sender` to our contract (thus assuming that our contract has been made an operator of the sender's tokens).
The swap will have `Open` state, and its `seller` will be initialized to the current `sender`.
The contract automatically assigns a `swap_id` (basing on an auto-incrementing counter) to the swap. We assume that it is the caller's responsibility to extract and record the `swap_id`, it will be available in the applied operation.
In case if some of `assets_offered.fa2_address` refers to an invalid FA2 contract (not originated or without `transfer` entrypoint), `SWAP_OFFERED_FA2_INVALID` is raised.
In case if tokens transfer from `sender` to our contract fails (due to lack of tokens or other issues), the respective FA2 error is propagated.
`assets_requested` is not validated and must be checked by the client for a better UX.
We do not check whether any assets list is empty on purpose.
### Cancel swap
```ocaml
| Cancel of swap_id
```
Marks the swap as canceled and returns the tokens to the `seller`. Further operations with this swap won't be possible.
Can be called only by the `seller`, otherwise `NOT_SWAP_SELLER` error is raised.
This does not remove the swap info from the big_map, for the sake of more detailed errors. ==Is it good?== Any further operation with this swap will raise `SWAP_CANCELED` error. If the provided `swap_id` is not registered, `SWAP_NOT_EXIST` error is raised.
### Accept swap
```ocaml
| Accept of swap_id
```
This
1. Transfers the offered assets from our contract to the `sender`;
2. Transfers the requested assets from the `sender` to the `seller`.
Marks the swap as finished by the current `sender`. Further operations with this swap won't be possible.
This can be called by *any* address.
This does not remove the swap info from the big_map, for the sake of more detailed errors. ==Is it good?== Any further operation with this swap will raise `SWAP_FINISHED` error. If the provided `swap_id` is not registered, `SWAP_NOT_EXIST` error is raised.
In case if some of `assets_requested.fa2_address` refers to an invalid FA2 contract (not originated or without `transfer` entrypoint), `SWAP_REQUESTED_FA2_INVALID` is raised. This error normally should not happen, as the client is supposed to validate `assets_requested` on swap start.
In case if any tokens transfer mentioned above fails (due to lack of tokens or other issues), the respective FA2 error is propagated.
## Notes to the middleware implementors
Any client working with this contract has to care about supplying the proper input.
### Grouping by FA2 address
Assets accepted by `Start` entrypoint has to be grouped by FA2 token address.
In case this is not done, the performance (and fees) may be far from optimal.
### Validating `assets_requested`
Upon executing `Start` entrypoint, the client has to check `assets_requested`:
* That the referred FA2 token addresses exist in the chain;
* That contract with those addresses contain `%transfer` entrypoint;
* That referred `token_id`s are supported by the FA2 (can be checked in TZIP-16 metadata).
Checking `assets_offered` this way is not strictly necessary as the contract will raise an error in case offered assets are not correct.
### Errors handling
Be sure to handle not only the errors listed in the entrypoints description, but also errors inherent to FA2 interface.