diff --git a/assets/error_codes.txt b/assets/error_codes.txt new file mode 100644 index 0000000..41d51f3 --- /dev/null +++ b/assets/error_codes.txt @@ -0,0 +1,95 @@ +const cooldownConflictError = 4000 +const waypointNoAccessError = 4001 +const tokenEmptyError = 4100 +const tokenMissingSubjectError = 4101 +const tokenInvalidSubjectError = 4102 +const missingTokenRequestError = 4103 +const invalidTokenRequestError = 4104 +const invalidTokenSubjectError = 4105 +const accountNotExistsError = 4106 +const agentNotExistsError = 4107 +const accountHasNoAgentError = 4108 +const registerAgentExistsError = 4109 +const registerAgentSymbolReservedError = 4110 +const registerAgentConflictSymbolError = 4111 +const navigateInTransitError = 4200 +const navigateInvalidDestinationError = 4201 +const navigateOutsideSystemError = 4202 +const navigateInsufficientFuelError = 4203 +const navigateSameDestinationError = 4204 +const shipExtractInvalidWaypointError = 4205 +const shipExtractPermissionError = 4206 +const shipJumpNoSystemError = 4207 +const shipJumpSameSystemError = 4208 +const shipJumpMissingModuleError = 4210 +const shipJumpNoValidWaypointError = 4211 +const shipJumpMissingAntimatterError = 4212 +const shipInTransitError = 4214 +const shipMissingSensorArraysError = 4215 +const purchaseShipCreditsError = 4216 +const shipCargoExceedsLimitError = 4217 +const shipCargoMissingError = 4218 +const shipCargoUnitCountError = 4219 +const shipSurveyVerificationError = 4220 +const shipSurveyExpirationError = 4221 +const shipSurveyWaypointTypeError = 4222 +const shipSurveyOrbitError = 4223 +const shipSurveyExhaustedError = 4224 +const shipRefuelDockedError = 4225 +const shipRefuelInvalidWaypointError = 4226 +const shipMissingMountsError = 4227 +const shipCargoFullError = 4228 +const shipJumpFromGateToGateError = 4229 +const waypointChartedError = 4230 +const shipTransferShipNotFound = 4231 +const shipTransferAgentConflict = 4232 +const shipTransferSameShipConflict = 4233 +const shipTransferLocationConflict = 4234 +const warpInsideSystemError = 4235 +const shipNotInOrbitError = 4236 +const shipInvalidRefineryGoodError = 4237 +const shipInvalidRefineryTypeError = 4238 +const shipMissingRefineryError = 4239 +const shipMissingSurveyorError = 4240 +const shipMissingWarpDriveError = 4241 +const shipMissingMineralProcessorError = 4242 +const shipMissingMiningLasersError = 4243 +const shipNotDockedError = 4244 +const purchaseShipNotPresentError = 4245 +const shipMountNoShipyardError = 4246 +const shipMissingMountError = 4247 +const shipMountInsufficientCreditsError = 4248 +const shipMissingPowerError = 4249 +const shipMissingSlotsError = 4250 +const shipMissingMountsError = 4251 +const shipMissingCrewError = 4252 +const shipExtractDestabilizedError = 4253 +const shipJumpInvalidOriginError = 4254 +const shipJumpInvalidWaypointError = 4255 +const shipJumpOriginUnderConstructionError = 4256 +const shipMissingGasProcessorError = 4257 +const shipMissingGasSiphonsError = 4258 +const shipSiphonInvalidWaypointError = 4259 +const shipSiphonPermissionError = 4260 +const waypointNoYieldError = 4261 +const shipJumpDestinationUnderConstructionError = 4262 +const acceptContractNotAuthorizedError = 4500 +const acceptContractConflictError = 4501 +const fulfillContractDeliveryError = 4502 +const contractDeadlineError = 4503 +const contractFulfilledError = 4504 +const contractNotAcceptedError = 4505 +const contractNotAuthorizedError = 4506 +const shipDeliverTermsError = 4508 +const shipDeliverFulfilledError = 4509 +const shipDeliverInvalidLocationError = 4510 +const existingContractError = 4511 +const marketTradeInsufficientCreditsError = 4600 +const marketTradeNoPurchaseError = 4601 +const marketTradeNotSoldError = 4602 +const marketNotFoundError = 4603 +const marketTradeUnitLimitError = 4604 +const waypointNoFactionError = 4700 +const constructionMaterialNotRequired = 4800 +const constructionMaterialFulfilled = 4801 +const shipConstructionInvalidLocationError = 4802 \ No newline at end of file diff --git a/assets/space-traders-sdk-authentication.png b/assets/space-traders-sdk-authentication.png new file mode 100644 index 0000000..9e2c7da Binary files /dev/null and b/assets/space-traders-sdk-authentication.png differ diff --git a/assets/space-traders-sdk-error-handling.png b/assets/space-traders-sdk-error-handling.png new file mode 100644 index 0000000..2410a92 Binary files /dev/null and b/assets/space-traders-sdk-error-handling.png differ diff --git a/assets/space-traders-sdk-retries.png b/assets/space-traders-sdk-retries.png new file mode 100644 index 0000000..828e774 Binary files /dev/null and b/assets/space-traders-sdk-retries.png differ diff --git a/sdk/README.md b/sdk/README.md index 5b24152..ac4bc77 100644 --- a/sdk/README.md +++ b/sdk/README.md @@ -59,6 +59,33 @@ By default, the connector read the token from the process variable with the name If you use a different variable for the token, or store the token as a connector secret, you need to adjust the authentication property of the connector. +![SDK property authentication](../assets/space-traders-sdk-authentication.png) + +### API Retries + +API calls can fail due to various reasons. One reason is that the SpaceTraders API is rate limited. It fails a call if +you send too many requests per second. + +The SDK allows to configure the number of retries and the delay between retries. + +By default, it retries the call 10 times with a delay of 1 second between retries. + +![SDK property retries](../assets/space-traders-sdk-retries.png) + +### API Error handling + +API calls can fail due to various reasons, for example, network issues or invalid parameters. + +The SDK allows to configure the error handling. A common use case is to throw a BPMN error if the call fails with a +specific error code (e.g. `4236`). The BPMN error can be caught by an error boundary event with the given +error code and handled in the process. + +By default, the SDK throws a BPMN error if the call returns a `400` status code. The BPMN error code is the name to the +SpaceTrader's error code (e.g. `shipNotInOrbitError` for the code `4236`). All error codes are listed +[here](https://docs.spacetraders.io/api-guide/response-errors/). + +![SDK property error handling](../assets/space-traders-sdk-error-handling.png) + ## Useful resources and tools Resources: diff --git a/sdk/spacetraders.json b/sdk/spacetraders.json index 6948a98..8b1e6a3 100644 --- a/sdk/spacetraders.json +++ b/sdk/spacetraders.json @@ -6,7 +6,7 @@ "icon": { "contents": "" }, - "version": 2, + "version": 3, "appliesTo": [ "bpmn:Task" ], @@ -30,6 +30,16 @@ "id": "responseMapping", "label": "Response Mapping" }, + { + "id": "error", + "label": "API Error handling", + "tooltip": "Handle error response codes from SpaceTrader's API in the process as BPMN error events." + }, + { + "id": "retries", + "label": "API Retries", + "tooltip": "Define retries if SpaceTrader's API returns an error response. Useful to handle API rate limiting." + }, { "id": "modificationWarning", "label": "Generated (DO NOT MODIFY)" @@ -6955,6 +6965,86 @@ "equals": "register" }, "optional": true + }, + { + "id": "customErrorHandling", + "label": "How to handle errors?", + "description": "By default, it maps the an error to a BPMN error event. The error code is the name of the SpaceTrader's error.", + "group": "error", + "binding": { + "type": "zeebe:property", + "name": "custom_error_handling" + }, + "type": "Dropdown", + "choices": [ + { + "name": "BPMN error by name (default)", + "value": "by_error_name" + }, + { + "name": "Custom", + "value": "custom" + } + ], + "value": "by_error_name" + }, + { + "id": "errorExpression", + "label": "Error expression", + "value": "={\n status: error.code,\n error_code: extract(error.message, \"\\d{4}\")[1],\n error_codes: {\n \"4000\":\"cooldownConflictError\",\n \"4001\":\"waypointNoAccessError\",\n \"4100\":\"tokenEmptyError\",\n \"4101\":\"tokenMissingSubjectError\",\n \"4102\":\"tokenInvalidSubjectError\",\n \"4103\":\"missingTokenRequestError\",\n \"4104\":\"invalidTokenRequestError\",\n \"4105\":\"invalidTokenSubjectError\",\n \"4106\":\"accountNotExistsError\",\n \"4107\":\"agentNotExistsError\",\n \"4108\":\"accountHasNoAgentError\",\n \"4109\":\"registerAgentExistsError\",\n \"4110\":\"registerAgentSymbolReservedError\",\n \"4111\":\"registerAgentConflictSymbolError\",\n \"4200\":\"navigateInTransitError\",\n \"4201\":\"navigateInvalidDestinationError\",\n \"4202\":\"navigateOutsideSystemError\",\n \"4203\":\"navigateInsufficientFuelError\",\n \"4204\":\"navigateSameDestinationError\",\n \"4205\":\"shipExtractInvalidWaypointError\",\n \"4206\":\"shipExtractPermissionError\",\n \"4207\":\"shipJumpNoSystemError\",\n \"4208\":\"shipJumpSameSystemError\",\n \"4210\":\"shipJumpMissingModuleError\",\n \"4211\":\"shipJumpNoValidWaypointError\",\n \"4212\":\"shipJumpMissingAntimatterError\",\n \"4214\":\"shipInTransitError\",\n \"4215\":\"shipMissingSensorArraysError\",\n \"4216\":\"purchaseShipCreditsError\",\n \"4217\":\"shipCargoExceedsLimitError\",\n \"4218\":\"shipCargoMissingError\",\n \"4219\":\"shipCargoUnitCountError\",\n \"4220\":\"shipSurveyVerificationError\",\n \"4221\":\"shipSurveyExpirationError\",\n \"4222\":\"shipSurveyWaypointTypeError\",\n \"4223\":\"shipSurveyOrbitError\",\n \"4224\":\"shipSurveyExhaustedError\",\n \"4225\":\"shipRefuelDockedError\",\n \"4226\":\"shipRefuelInvalidWaypointError\",\n \"4227\":\"shipMissingMountsError\",\n \"4228\":\"shipCargoFullError\",\n \"4229\":\"shipJumpFromGateToGateError\",\n \"4230\":\"waypointChartedError\",\n \"4231\":\"shipTransferShipNotFound\",\n \"4232\":\"shipTransferAgentConflict\",\n \"4233\":\"shipTransferSameShipConflict\",\n \"4234\":\"shipTransferLocationConflict\",\n \"4235\":\"warpInsideSystemError\",\n \"4236\":\"shipNotInOrbitError\",\n \"4237\":\"shipInvalidRefineryGoodError\",\n \"4238\":\"shipInvalidRefineryTypeError\",\n \"4239\":\"shipMissingRefineryError\",\n \"4240\":\"shipMissingSurveyorError\",\n \"4241\":\"shipMissingWarpDriveError\",\n \"4242\":\"shipMissingMineralProcessorError\",\n \"4243\":\"shipMissingMiningLasersError\",\n \"4244\":\"shipNotDockedError\",\n \"4245\":\"purchaseShipNotPresentError\",\n \"4246\":\"shipMountNoShipyardError\",\n \"4247\":\"shipMissingMountError\",\n \"4248\":\"shipMountInsufficientCreditsError\",\n \"4249\":\"shipMissingPowerError\",\n \"4250\":\"shipMissingSlotsError\",\n \"4251\":\"shipMissingMountsError\",\n \"4252\":\"shipMissingCrewError\",\n \"4253\":\"shipExtractDestabilizedError\",\n \"4254\":\"shipJumpInvalidOriginError\",\n \"4255\":\"shipJumpInvalidWaypointError\",\n \"4256\":\"shipJumpOriginUnderConstructionError\",\n \"4257\":\"shipMissingGasProcessorError\",\n \"4258\":\"shipMissingGasSiphonsError\",\n \"4259\":\"shipSiphonInvalidWaypointError\",\n \"4260\":\"shipSiphonPermissionError\",\n \"4261\":\"waypointNoYieldError\",\n \"4262\":\"shipJumpDestinationUnderConstructionError\",\n \"4500\":\"acceptContractNotAuthorizedError\",\n \"4501\":\"acceptContractConflictError\",\n \"4502\":\"fulfillContractDeliveryError\",\n \"4503\":\"contractDeadlineError\",\n \"4504\":\"contractFulfilledError\",\n \"4505\":\"contractNotAcceptedError\",\n \"4506\":\"contractNotAuthorizedError\",\n \"4508\":\"shipDeliverTermsError\",\n \"4509\":\"shipDeliverFulfilledError\",\n \"4510\":\"shipDeliverInvalidLocationError\",\n \"4511\":\"existingContractError\",\n \"4600\":\"marketTradeInsufficientCreditsError\",\n \"4601\":\"marketTradeNoPurchaseError\",\n \"4602\":\"marketTradeNotSoldError\",\n \"4603\":\"marketNotFoundError\",\n \"4604\":\"marketTradeUnitLimitError\",\n \"4700\":\"waypointNoFactionError\",\n \"4800\":\"constructionMaterialNotRequired\",\n \"4801\":\"constructionMaterialFulfilled\",\n \"4802\":\"shipConstructionInvalidLocationError\"\n },\n bpmn_error_code: get or else(get value(error_codes, error_code), error_code),\n bpmn_error_message: substring(extract(error.message, \"\\\"message\\\":\\\"(\\w|\\d|\\s|\\.|\\-)+\")[1], 12),\n result: \n if status = \"400\" \n then bpmnError(bpmn_error_code, bpmn_error_message)\n else null\n}.result", + "group": "error", + "binding": { + "key": "errorExpression", + "type": "zeebe:taskHeader" + }, + "type": "Hidden", + "condition": { + "property": "customErrorHandling", + "equals": "by_error_name" + } + }, + { + "id": "customErrorExpression", + "label": "Error expression", + "description": "Expression to handle errors. Details in the documentation.", + "value": "=if error.code = \"400\"\nthen bpmnError(\"error-1\", \"Something wrong.\")\nelse null", + "feel": "required", + "group": "error", + "binding": { + "key": "errorExpression", + "type": "zeebe:taskHeader" + }, + "type": "Text", + "condition": { + "property": "customErrorHandling", + "equals": "custom" + } + }, + { + "id": "retryCount", + "label": "Retries", + "description": "Number of retries", + "value": "10", + "feel": "optional", + "group": "retries", + "binding": { + "property": "retries", + "type": "zeebe:taskDefinition" + }, + "type": "String" + }, + { + "id": "retryBackoff", + "label": "Retry backoff", + "description": "ISO-8601 duration to wait between retries", + "value": "PT1S", + "feel": "optional", + "group": "retries", + "binding": { + "key": "retryBackoff", + "type": "zeebe:taskHeader" + }, + "type": "String" } ] } \ No newline at end of file