@rabbitio/ui-kit 1.0.0-beta.6 → 1.0.0-beta.61
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.gitlab-ci.yml +29 -0
- package/.husky/commit-msg +14 -0
- package/.husky/pre-push +1 -0
- package/CHANGELOG.md +0 -0
- package/README.md +27 -18
- package/coverage/base.css +224 -0
- package/coverage/block-navigation.js +87 -0
- package/coverage/clover.xml +17210 -0
- package/coverage/coverage-final.json +160 -0
- package/coverage/favicon.png +0 -0
- package/coverage/index.html +1241 -0
- package/coverage/prettify.css +1 -0
- package/coverage/prettify.js +2 -0
- package/coverage/sort-arrow-sprite.png +0 -0
- package/coverage/sorter.js +196 -0
- package/coverage/ui-kit/index.html +116 -0
- package/coverage/ui-kit/index.js.html +88 -0
- package/coverage/ui-kit/src/assets/wrappedImages/arrowIcon.jsx.html +142 -0
- package/coverage/ui-kit/src/assets/wrappedImages/arrowTosca.jsx.html +145 -0
- package/coverage/ui-kit/src/assets/wrappedImages/arrowWhite.jsx.html +145 -0
- package/coverage/ui-kit/src/assets/wrappedImages/darkRectangle.jsx.html +139 -0
- package/coverage/ui-kit/src/assets/wrappedImages/determinedError.jsx.html +472 -0
- package/coverage/ui-kit/src/assets/wrappedImages/failedValidationIcon.jsx.html +244 -0
- package/coverage/ui-kit/src/assets/wrappedImages/index.html +251 -0
- package/coverage/ui-kit/src/assets/wrappedImages/infoIcon.jsx.html +151 -0
- package/coverage/ui-kit/src/assets/wrappedImages/noticeQuestionIcon.jsx.html +265 -0
- package/coverage/ui-kit/src/assets/wrappedImages/successfulValidationIcon.jsx.html +205 -0
- package/coverage/ui-kit/src/assets/wrappedImages/walletIcon.jsx.html +199 -0
- package/coverage/ui-kit/src/common/adapters/axiosAdapter.js.html +190 -0
- package/coverage/ui-kit/src/common/adapters/index.html +131 -0
- package/coverage/ui-kit/src/common/adapters/qrUtils.js.html +139 -0
- package/coverage/ui-kit/src/common/amountUtils.js.html +1393 -0
- package/coverage/ui-kit/src/common/errorUtils.js.html +211 -0
- package/coverage/ui-kit/src/common/external-apis/apiGroups.js.html +250 -0
- package/coverage/ui-kit/src/common/external-apis/index.html +131 -0
- package/coverage/ui-kit/src/common/external-apis/ipAddressProviders.js.html +499 -0
- package/coverage/ui-kit/src/common/fiatCurrenciesService.js.html +568 -0
- package/coverage/ui-kit/src/common/index.html +146 -0
- package/coverage/ui-kit/src/common/models/blockchain.js.html +115 -0
- package/coverage/ui-kit/src/common/models/coin.js.html +556 -0
- package/coverage/ui-kit/src/common/models/index.html +146 -0
- package/coverage/ui-kit/src/common/models/protocol.js.html +100 -0
- package/coverage/ui-kit/src/common/utils/cache.js.html +889 -0
- package/coverage/ui-kit/src/common/utils/emailAPI.js.html +139 -0
- package/coverage/ui-kit/src/common/utils/index.html +161 -0
- package/coverage/ui-kit/src/common/utils/logging/index.html +131 -0
- package/coverage/ui-kit/src/common/utils/logging/logger.js.html +229 -0
- package/coverage/ui-kit/src/common/utils/logging/logsStorage.js.html +268 -0
- package/coverage/ui-kit/src/common/utils/postponeExecution.js.html +118 -0
- package/coverage/ui-kit/src/common/utils/safeStringify.js.html +235 -0
- package/coverage/ui-kit/src/components/atoms/AssetIcon/AssetIcon.jsx.html +277 -0
- package/coverage/ui-kit/src/components/atoms/AssetIcon/index.html +116 -0
- package/coverage/ui-kit/src/components/atoms/AssetSelection/AssetSelection.jsx.html +310 -0
- package/coverage/ui-kit/src/components/atoms/AssetSelection/index.html +116 -0
- package/coverage/ui-kit/src/components/atoms/BackgroundTitle/BackgroundTitle.jsx.html +217 -0
- package/coverage/ui-kit/src/components/atoms/BackgroundTitle/index.html +116 -0
- package/coverage/ui-kit/src/components/atoms/InformationMessage/InformationMessage.jsx.html +265 -0
- package/coverage/ui-kit/src/components/atoms/InformationMessage/index.html +116 -0
- package/coverage/ui-kit/src/components/atoms/Input/Input.jsx.html +679 -0
- package/coverage/ui-kit/src/components/atoms/Input/index.html +116 -0
- package/coverage/ui-kit/src/components/atoms/LoadingDots/LoadingDots.jsx.html +256 -0
- package/coverage/ui-kit/src/components/atoms/LoadingDots/index.html +116 -0
- package/coverage/ui-kit/src/components/atoms/NoticeIcon/NoticeIcon.jsx.html +298 -0
- package/coverage/ui-kit/src/components/atoms/NoticeIcon/index.html +116 -0
- package/coverage/ui-kit/src/components/atoms/QrCode/QrCode.jsx.html +187 -0
- package/coverage/ui-kit/src/components/atoms/QrCode/index.html +116 -0
- package/coverage/ui-kit/src/components/atoms/RateSelector/RateSelector.jsx.html +196 -0
- package/coverage/ui-kit/src/components/atoms/RateSelector/index.html +116 -0
- package/coverage/ui-kit/src/components/atoms/SupportChat/SupportChat.jsx.html +229 -0
- package/coverage/ui-kit/src/components/atoms/SupportChat/index.html +116 -0
- package/coverage/ui-kit/src/components/atoms/Textarea/Textarea.jsx.html +559 -0
- package/coverage/ui-kit/src/components/atoms/Textarea/index.html +116 -0
- package/coverage/ui-kit/src/components/atoms/TitleBox/TitleBox.jsx.html +577 -0
- package/coverage/ui-kit/src/components/atoms/TitleBox/index.html +116 -0
- package/coverage/ui-kit/src/components/atoms/Tooltip/Tooltip.jsx.html +295 -0
- package/coverage/ui-kit/src/components/atoms/Tooltip/index.html +116 -0
- package/coverage/ui-kit/src/components/atoms/TwoLinesOfText/LinesOfText.jsx.html +325 -0
- package/coverage/ui-kit/src/components/atoms/TwoLinesOfText/index.html +116 -0
- package/coverage/ui-kit/src/components/atoms/Validation/Validation.jsx.html +202 -0
- package/coverage/ui-kit/src/components/atoms/Validation/index.html +116 -0
- package/coverage/ui-kit/src/components/atoms/buttons/Button/Button.jsx.html +802 -0
- package/coverage/ui-kit/src/components/atoms/buttons/Button/index.html +116 -0
- package/coverage/ui-kit/src/components/atoms/buttons/Close/Close.jsx.html +262 -0
- package/coverage/ui-kit/src/components/atoms/buttons/Close/index.html +116 -0
- package/coverage/ui-kit/src/components/atoms/buttons/LinkButton/LinkButton.jsx.html +430 -0
- package/coverage/ui-kit/src/components/atoms/buttons/LinkButton/index.html +116 -0
- package/coverage/ui-kit/src/components/atoms/buttons/RadioButtonWithText/RadioButtonWithText.jsx.html +469 -0
- package/coverage/ui-kit/src/components/atoms/buttons/RadioButtonWithText/index.html +116 -0
- package/coverage/ui-kit/src/components/hooks/index.html +131 -0
- package/coverage/ui-kit/src/components/hooks/useCallHandlingErrors.js.html +163 -0
- package/coverage/ui-kit/src/components/hooks/useReferredState.js.html +157 -0
- package/coverage/ui-kit/src/components/molecules/AmountInput/AmountInput.jsx.html +1348 -0
- package/coverage/ui-kit/src/components/molecules/AmountInput/index.html +116 -0
- package/coverage/ui-kit/src/components/molecules/CoinPicker/CoinPicker.jsx.html +2011 -0
- package/coverage/ui-kit/src/components/molecules/CoinPicker/index.html +116 -0
- package/coverage/ui-kit/src/components/molecules/LineWithIconLink/LineWithIconLink.jsx.html +205 -0
- package/coverage/ui-kit/src/components/molecules/LineWithIconLink/index.html +116 -0
- package/coverage/ui-kit/src/components/molecules/TitledLineWithIconLink/TitledLineWithIconLink.jsx.html +175 -0
- package/coverage/ui-kit/src/components/molecules/TitledLineWithIconLink/index.html +116 -0
- package/coverage/ui-kit/src/components/organisms/CoinPickerDialogStep/CoinPickerDialogStep.jsx.html +283 -0
- package/coverage/ui-kit/src/components/organisms/CoinPickerDialogStep/index.html +116 -0
- package/coverage/ui-kit/src/components/organisms/Dialog/Dialog.jsx.html +1630 -0
- package/coverage/ui-kit/src/components/organisms/Dialog/DialogButtons/DialogButtons.jsx.html +451 -0
- package/coverage/ui-kit/src/components/organisms/Dialog/DialogButtons/index.html +116 -0
- package/coverage/ui-kit/src/components/organisms/Dialog/DialogStep/DialogStep.jsx.html +2146 -0
- package/coverage/ui-kit/src/components/organisms/Dialog/DialogStep/index.html +116 -0
- package/coverage/ui-kit/src/components/organisms/Dialog/index.html +116 -0
- package/coverage/ui-kit/src/components/organisms/SwapForm/SwapForm.jsx.html +3883 -0
- package/coverage/ui-kit/src/components/organisms/SwapForm/index.html +116 -0
- package/coverage/ui-kit/src/components/templates/DeterminedErrorDialogStep/DeterminedErrorDialogStep.jsx.html +316 -0
- package/coverage/ui-kit/src/components/templates/DeterminedErrorDialogStep/index.html +116 -0
- package/coverage/ui-kit/src/components/utils/index.html +161 -0
- package/coverage/ui-kit/src/components/utils/inputValueProviders.js.html +259 -0
- package/coverage/ui-kit/src/components/utils/textUtils.js.html +139 -0
- package/coverage/ui-kit/src/components/utils/uiUtils.js.html +127 -0
- package/coverage/ui-kit/src/components/utils/urlQueryUtils.js.html +346 -0
- package/coverage/ui-kit/src/constants/atoms/Close/close.jsx.html +100 -0
- package/coverage/ui-kit/src/constants/atoms/Close/index.html +116 -0
- package/coverage/ui-kit/src/constants/atoms/LinkButton/index.html +116 -0
- package/coverage/ui-kit/src/constants/atoms/LinkButton/linkButton.jsx.html +103 -0
- package/coverage/ui-kit/src/constants/atoms/Tooltip/index.html +116 -0
- package/coverage/ui-kit/src/constants/atoms/Tooltip/tooltip.jsx.html +109 -0
- package/coverage/ui-kit/src/constants/globalConstants.jsx.html +97 -0
- package/coverage/ui-kit/src/constants/index.html +116 -0
- package/coverage/ui-kit/src/constants/organisms/dialog/DialogStep/dialogStep.js.html +88 -0
- package/coverage/ui-kit/src/constants/organisms/dialog/DialogStep/index.html +116 -0
- package/coverage/ui-kit/src/constants/organisms/dialog/dialog.js.html +172 -0
- package/coverage/ui-kit/src/constants/organisms/dialog/index.html +116 -0
- package/coverage/ui-kit/src/index.html +116 -0
- package/coverage/ui-kit/src/index.js.html +367 -0
- package/coverage/ui-kit/src/robustExteranlApiCallerService/cacheAndConcurrentRequestsResolver.js.html +1762 -0
- package/coverage/ui-kit/src/robustExteranlApiCallerService/cachedRobustExternalApiCallerService.js.html +649 -0
- package/coverage/ui-kit/src/robustExteranlApiCallerService/cancelProcessing.js.html +172 -0
- package/coverage/ui-kit/src/robustExteranlApiCallerService/concurrentCalculationsMetadataHolder.js.html +394 -0
- package/coverage/ui-kit/src/robustExteranlApiCallerService/externalApiProvider.js.html +553 -0
- package/coverage/ui-kit/src/robustExteranlApiCallerService/externalServicesStatsCollector.js.html +331 -0
- package/coverage/ui-kit/src/robustExteranlApiCallerService/index.html +206 -0
- package/coverage/ui-kit/src/robustExteranlApiCallerService/robustExternalAPICallerService.js.html +1249 -0
- package/coverage/ui-kit/src/swaps-lib/external-apis/index.html +131 -0
- package/coverage/ui-kit/src/swaps-lib/external-apis/swapProvider.js.html +727 -0
- package/coverage/ui-kit/src/swaps-lib/external-apis/swapspaceSwapProvider.js.html +2899 -0
- package/coverage/ui-kit/src/swaps-lib/models/baseSwapCreationInfo.js.html +214 -0
- package/coverage/ui-kit/src/swaps-lib/models/existingSwap.js.html +304 -0
- package/coverage/ui-kit/src/swaps-lib/models/existingSwapWithFiatData.js.html +487 -0
- package/coverage/ui-kit/src/swaps-lib/models/index.html +146 -0
- package/coverage/ui-kit/src/swaps-lib/services/index.html +116 -0
- package/coverage/ui-kit/src/swaps-lib/services/publicSwapService.js.html +2191 -0
- package/coverage/ui-kit/src/swaps-lib/utils/index.html +116 -0
- package/coverage/ui-kit/src/swaps-lib/utils/swapUtils.js.html +742 -0
- package/coverage/ui-kit/stories/atoms/BackgroundTitle.stories.jsx.html +202 -0
- package/coverage/ui-kit/stories/atoms/LinesOfText.stories.jsx.html +283 -0
- package/coverage/ui-kit/stories/atoms/LoadingDots.stories.jsx.html +226 -0
- package/coverage/ui-kit/stories/atoms/QrCode.stories.jsx.html +175 -0
- package/coverage/ui-kit/stories/atoms/RateSelector.stories.jsx.html +136 -0
- package/coverage/ui-kit/stories/atoms/Validation.stories.jsx.html +178 -0
- package/coverage/ui-kit/stories/atoms/buttons/Button.stories.jsx.html +946 -0
- package/coverage/ui-kit/stories/atoms/buttons/Close.stories.jsx.html +214 -0
- package/coverage/ui-kit/stories/atoms/buttons/LinkButton.stories.jsx.html +295 -0
- package/coverage/ui-kit/stories/atoms/buttons/index.html +146 -0
- package/coverage/ui-kit/stories/atoms/index.html +191 -0
- package/coverage/ui-kit/stories/molecules/LineWithIconLink.stories.jsx.html +154 -0
- package/coverage/ui-kit/stories/molecules/TitledLineWithIconLink.stories.jsx.html +160 -0
- package/coverage/ui-kit/stories/molecules/index.html +131 -0
- package/coverage/ui-kit/stories/organisms/Dialog/Dialog.stories.jsx.html +589 -0
- package/coverage/ui-kit/stories/organisms/Dialog/DialogButtons/DialogButtons.stories.jsx.html +328 -0
- package/coverage/ui-kit/stories/organisms/Dialog/DialogButtons/index.html +116 -0
- package/coverage/ui-kit/stories/organisms/Dialog/DialogStep/DialogStep.stories.jsx.html +337 -0
- package/coverage/ui-kit/stories/organisms/Dialog/DialogStep/index.html +116 -0
- package/coverage/ui-kit/stories/organisms/Dialog/index.html +116 -0
- package/coverage/ui-kit/stories/stubs/exampleContent.jsx.html +145 -0
- package/coverage/ui-kit/stories/stubs/index.html +116 -0
- package/coverage/ui-kit/stories/templates/DeterminedErrorDialogStep.stories.jsx.html +193 -0
- package/coverage/ui-kit/stories/templates/index.html +116 -0
- package/dist/index.cjs +9963 -9
- package/dist/index.cjs.map +1 -1
- package/dist/index.css +71409 -1641
- package/dist/index.css.map +1 -1
- package/dist/index.modern.js +7672 -11
- package/dist/index.modern.js.map +1 -1
- package/dist/index.module.js +9883 -11
- package/dist/index.module.js.map +1 -1
- package/dist/index.umd.js +9954 -12
- package/dist/index.umd.js.map +1 -1
- package/index.js +1 -1
- package/package.json +27 -5
- package/src/assets/image/icons/arrow-tosca.svg +3 -0
- package/src/assets/wrappedImages/arrowIcon.jsx +19 -0
- package/src/assets/wrappedImages/arrowTosca.jsx +20 -0
- package/src/assets/wrappedImages/arrowWhite.jsx +20 -0
- package/src/assets/wrappedImages/darkRectangle.jsx +18 -0
- package/src/assets/wrappedImages/determinedError.jsx +129 -0
- package/src/assets/wrappedImages/failedValidationIcon.jsx +53 -0
- package/src/assets/wrappedImages/infoIcon.jsx +22 -0
- package/src/assets/wrappedImages/noticeQuestionIcon.jsx +60 -0
- package/src/assets/wrappedImages/successfulValidationIcon.jsx +40 -0
- package/src/assets/wrappedImages/walletIcon.jsx +38 -0
- package/src/common/adapters/axiosAdapter.js +35 -0
- package/src/common/adapters/qrUtils.js +18 -0
- package/src/common/amountUtils.js +436 -0
- package/src/common/errorUtils.js +42 -0
- package/src/common/external-apis/apiGroups.js +55 -0
- package/src/common/external-apis/ipAddressProviders.js +138 -0
- package/src/common/fiatCurrenciesService.js +161 -0
- package/src/common/models/blockchain.js +10 -0
- package/src/common/models/coin.js +157 -0
- package/src/common/models/protocol.js +5 -0
- package/src/common/tests/integration/external-apis/ipAddressProviders/getClientIpAddress.test.js +14 -0
- package/src/common/utils/cache.js +268 -0
- package/src/common/utils/emailAPI.js +18 -0
- package/src/common/utils/logging/logger.js +48 -0
- package/src/common/utils/logging/logsStorage.js +61 -0
- package/src/common/utils/postponeExecution.js +11 -0
- package/src/common/utils/safeStringify.js +50 -0
- package/src/components/atoms/AssetIcon/AssetIcon.jsx +64 -0
- package/src/components/atoms/AssetIcon/asset-icon.module.scss +42 -0
- package/src/components/atoms/AssetSelection/AssetSelection.jsx +75 -0
- package/src/components/atoms/AssetSelection/asset-selection.module.scss +55 -0
- package/src/components/atoms/BackgroundTitle/BackgroundTitle.jsx +44 -0
- package/src/components/atoms/BackgroundTitle/background-title.module.scss +52 -0
- package/src/components/atoms/InformationMessage/InformationMessage.jsx +60 -0
- package/src/components/atoms/InformationMessage/information-message.module.scss +36 -0
- package/src/components/atoms/Input/Input.jsx +198 -0
- package/src/components/atoms/Input/input.module.scss +107 -0
- package/{stories → src/components}/atoms/LoadingDots/LoadingDots.module.scss +1 -1
- package/src/components/atoms/NoticeIcon/NoticeIcon.jsx +71 -0
- package/src/components/atoms/NoticeIcon/notice-icon.module.scss +14 -0
- package/src/components/atoms/QrCode/QrCode.jsx +34 -0
- package/src/components/atoms/QrCode/qr-code.module.scss +9 -0
- package/src/components/atoms/RateSelector/RateSelector.jsx +37 -0
- package/src/components/atoms/RateSelector/rate-selector.module.scss +47 -0
- package/src/components/atoms/SupportChat/SupportChat.jsx +48 -0
- package/src/components/atoms/Textarea/Textarea.jsx +158 -0
- package/src/components/atoms/Textarea/textarea.module.scss +71 -0
- package/src/components/atoms/TitleBox/TitleBox.jsx +164 -0
- package/src/components/atoms/TitleBox/title-box.module.scss +30 -0
- package/src/components/atoms/Tooltip/Tooltip.jsx +70 -0
- package/src/components/atoms/Tooltip/tooltip.module.scss +237 -0
- package/src/components/atoms/TwoLinesOfText/LinesOfText.jsx +80 -0
- package/src/components/atoms/TwoLinesOfText/lines-of-text.module.scss +65 -0
- package/src/components/atoms/Validation/Validation.jsx +39 -0
- package/src/components/atoms/Validation/validation.module.scss +15 -0
- package/{stories → src/components}/atoms/buttons/Button/Button.jsx +5 -1
- package/{stories → src/components}/atoms/buttons/Button/Button.module.scss +1 -1
- package/src/components/atoms/buttons/Close/Close.jsx +59 -0
- package/src/components/atoms/buttons/Close/close.module.scss +75 -0
- package/src/components/atoms/buttons/LinkButton/LinkButton.jsx +115 -0
- package/src/components/atoms/buttons/LinkButton/link-button.module.scss +49 -0
- package/src/components/atoms/buttons/RadioButtonWithText/RadioButtonWithText.jsx +128 -0
- package/src/components/atoms/buttons/RadioButtonWithText/radio-button-with-text.module.scss +86 -0
- package/src/components/hooks/useCallHandlingErrors.js +26 -0
- package/src/components/hooks/useReferredState.js +24 -0
- package/src/components/molecules/AmountInput/AmountInput.jsx +421 -0
- package/src/components/molecules/AmountInput/amount-input.module.scss +189 -0
- package/src/components/molecules/CoinPicker/CoinPicker.jsx +642 -0
- package/src/components/molecules/CoinPicker/coin-picker.module.scss +210 -0
- package/src/components/molecules/LineWithIconLink/LineWithIconLink.jsx +40 -0
- package/src/components/molecules/LineWithIconLink/line-with-icon-link.module.scss +25 -0
- package/src/components/molecules/TitledLineWithIconLink/TitledLineWithIconLink.jsx +30 -0
- package/src/components/organisms/CoinPickerDialogStep/CoinPickerDialogStep.jsx +66 -0
- package/src/components/organisms/Dialog/Dialog.jsx +515 -0
- package/src/components/organisms/Dialog/DialogButtons/DialogButtons.jsx +122 -0
- package/src/components/organisms/Dialog/DialogButtons/dialog-buttons.module.scss +25 -0
- package/src/components/organisms/Dialog/DialogStep/DialogStep.jsx +687 -0
- package/src/components/organisms/Dialog/DialogStep/dialog-step.module.scss +383 -0
- package/src/components/organisms/Dialog/dialog.module.scss +226 -0
- package/src/components/organisms/SwapForm/SwapForm.jsx +1266 -0
- package/src/components/organisms/SwapForm/swap-form.module.scss +134 -0
- package/src/components/templates/DeterminedErrorDialogStep/DeterminedErrorDialogStep.jsx +77 -0
- package/src/components/tests/utils/inputValueProviders/provideFormatOfFloatValueByInputString.test.js +139 -0
- package/src/components/tests/utils/urlQueryUtils/getQueryParameterValues.test.js +71 -0
- package/src/components/tests/utils/urlQueryUtils/saveQueryParameterAndValues.test.js +144 -0
- package/src/components/utils/inputValueProviders.js +58 -0
- package/src/components/utils/textUtils.js +18 -0
- package/src/components/utils/uiUtils.js +14 -0
- package/src/components/utils/urlQueryUtils.js +87 -0
- package/src/constants/atoms/Close/close.jsx +5 -0
- package/src/constants/atoms/LinkButton/linkButton.jsx +6 -0
- package/src/constants/atoms/Tooltip/tooltip.jsx +8 -0
- package/src/constants/globalConstants.jsx +4 -0
- package/src/constants/organisms/dialog/DialogStep/dialogStep.js +1 -0
- package/src/constants/organisms/dialog/dialog.js +29 -0
- package/src/index.js +94 -0
- package/src/robustExteranlApiCallerService/cacheAndConcurrentRequestsResolver.js +559 -0
- package/src/robustExteranlApiCallerService/cachedRobustExternalApiCallerService.js +188 -0
- package/src/robustExteranlApiCallerService/cancelProcessing.js +29 -0
- package/src/robustExteranlApiCallerService/concurrentCalculationsMetadataHolder.js +103 -0
- package/src/robustExteranlApiCallerService/externalApiProvider.js +156 -0
- package/src/robustExteranlApiCallerService/externalServicesStatsCollector.js +82 -0
- package/src/robustExteranlApiCallerService/robustExternalAPICallerService.js +388 -0
- package/src/robustExteranlApiCallerService/tests/robustExternalAPICallerService/robustExternalAPICallerService/callExternalAPI/_performCallAttempt.test.js +787 -0
- package/src/robustExteranlApiCallerService/tests/robustExternalAPICallerService/robustExternalAPICallerService/callExternalAPI/callExternalAPI.test.js +745 -0
- package/src/robustExteranlApiCallerService/tests/robustExternalAPICallerService/robustExternalAPICallerService/constructor.test.js +31 -0
- package/src/swaps-lib/external-apis/swapProvider.js +214 -0
- package/src/swaps-lib/external-apis/swapspaceSwapProvider.js +938 -0
- package/src/swaps-lib/models/baseSwapCreationInfo.js +43 -0
- package/src/swaps-lib/models/existingSwap.js +73 -0
- package/src/swaps-lib/models/existingSwapWithFiatData.js +134 -0
- package/src/swaps-lib/services/publicSwapService.js +702 -0
- package/src/swaps-lib/test/external-apis/swapspaceSwapProvider/_fetchSupportedCurrenciesIfNeeded.test.js +506 -0
- package/src/swaps-lib/test/external-apis/swapspaceSwapProvider/createSwap.test.js +1311 -0
- package/src/swaps-lib/test/external-apis/swapspaceSwapProvider/getAllSupportedCurrencies.test.js +76 -0
- package/src/swaps-lib/test/external-apis/swapspaceSwapProvider/getDepositCurrencies.test.js +82 -0
- package/src/swaps-lib/test/external-apis/swapspaceSwapProvider/getSwapInfo.test.js +1892 -0
- package/src/swaps-lib/test/external-apis/swapspaceSwapProvider/getWithdrawalCurrencies.test.js +111 -0
- package/src/swaps-lib/test/utils/swapUtils/safeHandleRequestsLimitExceeding.test.js +88 -0
- package/src/swaps-lib/utils/swapUtils.js +219 -0
- package/stories/stubs/exampleContent.jsx +20 -0
- package/styles/_placeholder.scss +1 -1
- package/styles/fonts/NunitoSans-Bold.ttf +0 -0
- package/styles/fonts/NunitoSans-ExtraBold.ttf +0 -0
- package/styles/fonts/NunitoSans-Light.ttf +0 -0
- package/styles/fonts/NunitoSans-Regular.ttf +0 -0
- package/styles/fonts/NunitoSans-SemiBold.ttf +0 -0
- package/styles/global-styles-index.scss +1 -1
- package/styles/index.scss +5 -3
- package/stories/index.js +0 -2
- /package/{stories → src/components}/atoms/LoadingDots/LoadingDots.jsx +0 -0
|
@@ -0,0 +1,436 @@
|
|
|
1
|
+
import { BigNumber } from "bignumber.js";
|
|
2
|
+
|
|
3
|
+
import { FiatCurrenciesService } from "./fiatCurrenciesService.js";
|
|
4
|
+
import { improveAndRethrow } from "./errorUtils.js";
|
|
5
|
+
|
|
6
|
+
// TODO: [dev] return addCommasToAmountString internal method to encapsulate commas adding
|
|
7
|
+
|
|
8
|
+
export class AmountUtils {
|
|
9
|
+
static significantDecimalCount = 8;
|
|
10
|
+
static collapsedDecimalCount = 2;
|
|
11
|
+
static maxTotalLength = 12;
|
|
12
|
+
static extraSmallMaxTotalLength = 9; // >=10 breaks transactions list (mobile) and it is hard to avoid this
|
|
13
|
+
static periods = "..";
|
|
14
|
+
|
|
15
|
+
static defaultFiatParams = {
|
|
16
|
+
ticker: true, // If true, currency code will be shown
|
|
17
|
+
enableCurrencySymbols: true, // Enables currency symbols where available. Requires "ticker: true"
|
|
18
|
+
collapsible: true, // Enables minimization of amounts over 1 million (example: 1.52M)
|
|
19
|
+
limitTotalLength: true, // Limits the total amount length to maxTotalLength
|
|
20
|
+
extraSmallLength: false, // Limits the total amount length to extraSmallMaxTotalLength
|
|
21
|
+
};
|
|
22
|
+
|
|
23
|
+
static fiatXs(amount, code) {
|
|
24
|
+
return this.fiat(amount, code, { extraSmallLength: true });
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
/**
|
|
28
|
+
* Universal method for rendering of fiat amounts, taking into account the rules of
|
|
29
|
+
* the passed fiat currency code.
|
|
30
|
+
*
|
|
31
|
+
* TODO: [feature, high] remove 'number' from accepted types list task_id=1e692bcfabbe487a9d1638fc8ff17448
|
|
32
|
+
* @param amount {BigNumber|number|string|null|undefined} The number value to be trimmed
|
|
33
|
+
* @param currencyCode {string|null} The currency code. Can be omitted if { ticker: false } in the config
|
|
34
|
+
* @param [passedParams={}] {object} Formatting parameters
|
|
35
|
+
* @return {string} Formatted fiat amount string
|
|
36
|
+
*/
|
|
37
|
+
static fiat(amount, currencyCode, passedParams = {}) {
|
|
38
|
+
try {
|
|
39
|
+
const params = { ...this.defaultFiatParams, ...passedParams };
|
|
40
|
+
|
|
41
|
+
if (
|
|
42
|
+
this._checkIfAmountInvalid(amount, true) ||
|
|
43
|
+
typeof currencyCode !== "string"
|
|
44
|
+
)
|
|
45
|
+
return "NULL";
|
|
46
|
+
|
|
47
|
+
const currencySymbol =
|
|
48
|
+
FiatCurrenciesService.getCurrencySymbolByCode(currencyCode);
|
|
49
|
+
const currencyDecimalCount =
|
|
50
|
+
FiatCurrenciesService.getCurrencyDecimalCountByCode(
|
|
51
|
+
currencyCode
|
|
52
|
+
);
|
|
53
|
+
|
|
54
|
+
const trimmedByMaxDigits = BigNumber(amount).toFixed(
|
|
55
|
+
currencyDecimalCount,
|
|
56
|
+
BigNumber.ROUND_FLOOR
|
|
57
|
+
);
|
|
58
|
+
|
|
59
|
+
let processedAmount = BigNumber(trimmedByMaxDigits);
|
|
60
|
+
if (
|
|
61
|
+
params.collapsible &&
|
|
62
|
+
processedAmount.gte(BigNumber("1000000"))
|
|
63
|
+
) {
|
|
64
|
+
processedAmount = this._collapseToMillionsAndFormat(
|
|
65
|
+
processedAmount,
|
|
66
|
+
this.collapsedDecimalCount,
|
|
67
|
+
params
|
|
68
|
+
);
|
|
69
|
+
} else {
|
|
70
|
+
const limitResult = this._limitTotalAmountLengthIfNeeded(
|
|
71
|
+
trimmedByMaxDigits,
|
|
72
|
+
params
|
|
73
|
+
);
|
|
74
|
+
processedAmount = BigNumber(
|
|
75
|
+
limitResult.processedAmount
|
|
76
|
+
).toFormat(); // Adds commas to integer part
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
// Add the currency code or currency symbol, if symbol is enabled and available
|
|
80
|
+
if (params.ticker) {
|
|
81
|
+
if (
|
|
82
|
+
typeof currencySymbol === "string" &&
|
|
83
|
+
params.enableCurrencySymbols
|
|
84
|
+
) {
|
|
85
|
+
processedAmount =
|
|
86
|
+
currencySymbol +
|
|
87
|
+
(currencySymbol.length > 1 ? " " : "") +
|
|
88
|
+
processedAmount;
|
|
89
|
+
} else {
|
|
90
|
+
processedAmount = processedAmount + " " + currencyCode;
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
return processedAmount;
|
|
95
|
+
} catch (e) {
|
|
96
|
+
improveAndRethrow(e, "fiat", `Passed: ${amount}`);
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
|
|
100
|
+
static defaultCryptoParams = {
|
|
101
|
+
ticker: true, // If true, asset ticker will be shown
|
|
102
|
+
collapsible: true, // Enables minimization of amounts over 1 million (example: 1.52M)
|
|
103
|
+
trim: true, // Cuts the right part of the amount if necessary, and adds ".." in the end
|
|
104
|
+
limitTotalLength: true, // Limits the total amount length to maxTotalLength
|
|
105
|
+
extraSmallLength: false, // Limits the total amount length to extraSmallMaxTotalLength
|
|
106
|
+
periods: true, // Whether we add periods ("..") as suffix for trimmed numbers
|
|
107
|
+
numberPartsSeparator: true, // Whether we add separators e.g. for 1000000 -> 1,000,000
|
|
108
|
+
};
|
|
109
|
+
|
|
110
|
+
static cryptoWoTicker(amount, digits) {
|
|
111
|
+
return this.crypto(amount, null, digits, { ticker: false });
|
|
112
|
+
}
|
|
113
|
+
|
|
114
|
+
static cryptoWoTickerXs(amount, digits) {
|
|
115
|
+
return this.crypto(amount, null, digits, {
|
|
116
|
+
ticker: false,
|
|
117
|
+
extraSmallLength: true,
|
|
118
|
+
});
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
static cryptoXs(amount, ticker, digits) {
|
|
122
|
+
return this.crypto(amount, ticker, digits, {
|
|
123
|
+
extraSmallLength: true,
|
|
124
|
+
periods: false,
|
|
125
|
+
});
|
|
126
|
+
}
|
|
127
|
+
|
|
128
|
+
static cryptoFull(amount, ticker, digits) {
|
|
129
|
+
return this.crypto(amount, ticker, digits, {
|
|
130
|
+
collapsible: false,
|
|
131
|
+
trim: false,
|
|
132
|
+
limitTotalLength: false,
|
|
133
|
+
});
|
|
134
|
+
}
|
|
135
|
+
|
|
136
|
+
static cryptoFullPureNumber(amount, digits) {
|
|
137
|
+
return this.crypto(amount, null, digits, {
|
|
138
|
+
ticker: false,
|
|
139
|
+
collapsible: false,
|
|
140
|
+
trim: false,
|
|
141
|
+
limitTotalLength: false,
|
|
142
|
+
numberPartsSeparator: false,
|
|
143
|
+
});
|
|
144
|
+
}
|
|
145
|
+
|
|
146
|
+
/**
|
|
147
|
+
* Universal method for rendering of crypto amounts, taking into account the rules of
|
|
148
|
+
* the passed ticker. Requires the number of digits after period to be less of equal to
|
|
149
|
+
* the number of digits, supported by the passed ticker.
|
|
150
|
+
*
|
|
151
|
+
* @param amount {BigNumber|string|null|undefined} The number value to be formatted
|
|
152
|
+
* @param ticker {string|null} Coin ticker
|
|
153
|
+
* @param [digits=8] {number} max digits after the dot
|
|
154
|
+
* @param passedParams {object} Formatting parameters
|
|
155
|
+
* @return {string} Formatted crypto amount string
|
|
156
|
+
*/
|
|
157
|
+
static crypto(
|
|
158
|
+
amount,
|
|
159
|
+
ticker,
|
|
160
|
+
digits = this.significantDecimalCount,
|
|
161
|
+
passedParams
|
|
162
|
+
) {
|
|
163
|
+
try {
|
|
164
|
+
const params = { ...this.defaultCryptoParams, ...passedParams };
|
|
165
|
+
|
|
166
|
+
if (
|
|
167
|
+
this._checkIfAmountInvalid(amount) ||
|
|
168
|
+
(typeof ticker !== "string" && params.ticker)
|
|
169
|
+
)
|
|
170
|
+
return "NULL";
|
|
171
|
+
|
|
172
|
+
let addPeriods = false;
|
|
173
|
+
|
|
174
|
+
const amountBigNumber = BigNumber(amount);
|
|
175
|
+
|
|
176
|
+
let processedAmount = amountBigNumber.toFixed(
|
|
177
|
+
digits,
|
|
178
|
+
BigNumber.ROUND_FLOOR
|
|
179
|
+
);
|
|
180
|
+
processedAmount =
|
|
181
|
+
this.removeRedundantRightZerosFromNumberString(processedAmount);
|
|
182
|
+
const originalAmountDecimalPlaces =
|
|
183
|
+
BigNumber(processedAmount).decimalPlaces();
|
|
184
|
+
// Check decimal count and throw an error, if the amount has more decimal digits than supported by the asset
|
|
185
|
+
if (originalAmountDecimalPlaces > digits) {
|
|
186
|
+
const errorMessage = `An attempt to render a crypto value with too many digits after period was made: ${amount}, allowed digits: ${digits}. This is a no-op, since the logical and visually rendered values would differ, which is not acceptable for crypto amounts. Please trim the amount before rendering, using the trimCryptoAmountByCoin(amount, coin) method.`;
|
|
187
|
+
// throw new Error(errorMessage);
|
|
188
|
+
// eslint-disable-next-line no-console
|
|
189
|
+
console.log(errorMessage, "crypto");
|
|
190
|
+
}
|
|
191
|
+
|
|
192
|
+
// Shortening the value to general significant number of digits after period
|
|
193
|
+
if (params.trim) {
|
|
194
|
+
processedAmount =
|
|
195
|
+
this.removeRedundantRightZerosFromNumberString(
|
|
196
|
+
amountBigNumber.toFixed(
|
|
197
|
+
this.significantDecimalCount,
|
|
198
|
+
BigNumber.ROUND_FLOOR
|
|
199
|
+
)
|
|
200
|
+
);
|
|
201
|
+
addPeriods =
|
|
202
|
+
originalAmountDecimalPlaces > this.significantDecimalCount;
|
|
203
|
+
}
|
|
204
|
+
|
|
205
|
+
const limitResult = this._limitTotalAmountLengthIfNeeded(
|
|
206
|
+
processedAmount,
|
|
207
|
+
params
|
|
208
|
+
);
|
|
209
|
+
processedAmount = limitResult.processedAmount;
|
|
210
|
+
addPeriods ||= limitResult.addPeriods;
|
|
211
|
+
|
|
212
|
+
let wereMillionsCollapsed = false;
|
|
213
|
+
if (params.collapsible && amountBigNumber.gte("1000000")) {
|
|
214
|
+
// Collapse the 1M+ amounts if applicable
|
|
215
|
+
processedAmount = this._collapseToMillionsAndFormat(
|
|
216
|
+
BigNumber(processedAmount),
|
|
217
|
+
this.collapsedDecimalCount,
|
|
218
|
+
params
|
|
219
|
+
);
|
|
220
|
+
wereMillionsCollapsed = true;
|
|
221
|
+
} else if (params.numberPartsSeparator) {
|
|
222
|
+
// Add separators to integer part of the amount
|
|
223
|
+
processedAmount = BigNumber(processedAmount).toFormat();
|
|
224
|
+
}
|
|
225
|
+
|
|
226
|
+
// Adding periods, if the amount was shortened
|
|
227
|
+
if (params.periods && addPeriods && !wereMillionsCollapsed) {
|
|
228
|
+
processedAmount = processedAmount + this.periods;
|
|
229
|
+
}
|
|
230
|
+
|
|
231
|
+
// Adding an adaptive (printable/full) ticker
|
|
232
|
+
if (params.ticker && ticker) {
|
|
233
|
+
processedAmount = processedAmount + " " + ticker;
|
|
234
|
+
}
|
|
235
|
+
|
|
236
|
+
return processedAmount;
|
|
237
|
+
} catch (e) {
|
|
238
|
+
improveAndRethrow(
|
|
239
|
+
e,
|
|
240
|
+
"crypto",
|
|
241
|
+
`Passed: ${amount}, ${ticker}, ${digits}`
|
|
242
|
+
);
|
|
243
|
+
}
|
|
244
|
+
}
|
|
245
|
+
|
|
246
|
+
static _checkIfAmountInvalid(amount, allowNumbers = false) {
|
|
247
|
+
return (
|
|
248
|
+
amount == null ||
|
|
249
|
+
amount === "" ||
|
|
250
|
+
(!BigNumber.isBigNumber(amount) &&
|
|
251
|
+
typeof amount !== "string" &&
|
|
252
|
+
(!allowNumbers || typeof amount !== "number"))
|
|
253
|
+
);
|
|
254
|
+
}
|
|
255
|
+
|
|
256
|
+
/**
|
|
257
|
+
* Trims all digits after period that exceed the number of digits provided.
|
|
258
|
+
* Use this everywhere when calculating some amount value to ensure the result is correct in terms
|
|
259
|
+
* of max digits allowed by specific currency.
|
|
260
|
+
*
|
|
261
|
+
* @param amount {BigNumber|number|string|null|undefined} The number value to be trimmed.
|
|
262
|
+
* HEX strings also allowed "0x..." and JS hex numbers
|
|
263
|
+
* @param digits {number} allowed digits
|
|
264
|
+
* @return {string|null} String with trimmed number or null for invalid amount
|
|
265
|
+
*/
|
|
266
|
+
static trim(amount, digits) {
|
|
267
|
+
try {
|
|
268
|
+
if (this._checkIfAmountInvalid(amount, true)) return null;
|
|
269
|
+
return BigNumber(amount).toFixed(digits, BigNumber.ROUND_FLOOR);
|
|
270
|
+
} catch (e) {
|
|
271
|
+
improveAndRethrow(e, "trim", `Passed: ${amount}, ${digits}`);
|
|
272
|
+
}
|
|
273
|
+
}
|
|
274
|
+
|
|
275
|
+
/**
|
|
276
|
+
* Returns integer part of number as a string.
|
|
277
|
+
*
|
|
278
|
+
* @param amount {BigNumber|number|string|null|undefined} The number value to be trimmed.
|
|
279
|
+
* HEX strings also allowed "0x..." and JS hex numbers
|
|
280
|
+
* @return {string|null}
|
|
281
|
+
*/
|
|
282
|
+
static toIntegerString(amount) {
|
|
283
|
+
return this.trim(amount, 0);
|
|
284
|
+
}
|
|
285
|
+
|
|
286
|
+
/**
|
|
287
|
+
* Shortens the line length by using a "1.52M" representation of big amounts.
|
|
288
|
+
*
|
|
289
|
+
* @param amountBigNumber {BigNumber} The number value to be trimmed
|
|
290
|
+
* @param decimalCount {number} The number of digits after period to keep in millions representation
|
|
291
|
+
* @param params {object} params object
|
|
292
|
+
* @return {string} A shortened string, converted into "millions" format, if the amount exceeds 1 million
|
|
293
|
+
*/
|
|
294
|
+
static _collapseToMillionsAndFormat(
|
|
295
|
+
amountBigNumber,
|
|
296
|
+
decimalCount,
|
|
297
|
+
params = {}
|
|
298
|
+
) {
|
|
299
|
+
try {
|
|
300
|
+
// TODO: [feature, moderate] use local format here - take from JS locales (comma/dot etc.)
|
|
301
|
+
const millionBigNumber = BigNumber("1000000");
|
|
302
|
+
const millions = amountBigNumber
|
|
303
|
+
.div(millionBigNumber)
|
|
304
|
+
.toFixed(decimalCount, BigNumber.ROUND_FLOOR);
|
|
305
|
+
const limitedResult = this._limitTotalAmountLengthIfNeeded(
|
|
306
|
+
millions,
|
|
307
|
+
params
|
|
308
|
+
);
|
|
309
|
+
const formatted = BigNumber(
|
|
310
|
+
limitedResult.processedAmount
|
|
311
|
+
).toFormat();
|
|
312
|
+
|
|
313
|
+
return formatted + "M";
|
|
314
|
+
} catch (e) {
|
|
315
|
+
improveAndRethrow(
|
|
316
|
+
e,
|
|
317
|
+
"_collapseAmountAndFormat",
|
|
318
|
+
`Passed: ${amountBigNumber.toFixed()}, ${decimalCount}`
|
|
319
|
+
);
|
|
320
|
+
}
|
|
321
|
+
}
|
|
322
|
+
|
|
323
|
+
/**
|
|
324
|
+
* @param amountString {string} The amount to be restricted by length
|
|
325
|
+
* @param params {object} Params object used for formatting
|
|
326
|
+
* @return {{processedAmount:string, addPeriods: boolean}} A shortened string
|
|
327
|
+
*/
|
|
328
|
+
static _limitTotalAmountLengthIfNeeded(amountString, params) {
|
|
329
|
+
try {
|
|
330
|
+
let addPeriods = false;
|
|
331
|
+
if (params.limitTotalLength || params.extraSmallLength) {
|
|
332
|
+
const maxLength = params.extraSmallLength
|
|
333
|
+
? this.extraSmallMaxTotalLength
|
|
334
|
+
: this.maxTotalLength;
|
|
335
|
+
if (amountString.length > maxLength) {
|
|
336
|
+
const delta = amountString.length - maxLength;
|
|
337
|
+
const currentDecimalsCount =
|
|
338
|
+
BigNumber(amountString).decimalPlaces();
|
|
339
|
+
const newDecimalCount = currentDecimalsCount - delta;
|
|
340
|
+
amountString = BigNumber(amountString).toFixed(
|
|
341
|
+
newDecimalCount > 2 ? newDecimalCount : 2,
|
|
342
|
+
BigNumber.ROUND_FLOOR
|
|
343
|
+
);
|
|
344
|
+
addPeriods = currentDecimalsCount > newDecimalCount;
|
|
345
|
+
}
|
|
346
|
+
}
|
|
347
|
+
|
|
348
|
+
return { addPeriods: addPeriods, processedAmount: amountString };
|
|
349
|
+
} catch (e) {
|
|
350
|
+
improveAndRethrow(
|
|
351
|
+
e,
|
|
352
|
+
"_limitTotalAmountLengthIfNeeded",
|
|
353
|
+
`Passed: ${amountString}, ${params}`
|
|
354
|
+
);
|
|
355
|
+
}
|
|
356
|
+
}
|
|
357
|
+
|
|
358
|
+
/**
|
|
359
|
+
* Safely composes rate string (handles small/big rates)
|
|
360
|
+
*
|
|
361
|
+
* @param leftTicker {string}
|
|
362
|
+
* @param rightTicker {string}
|
|
363
|
+
* @param rate {number|string|BigNumber}
|
|
364
|
+
* @param [rightCurrencyDigitsAfterDots=8] {number}
|
|
365
|
+
* @return {string}
|
|
366
|
+
*/
|
|
367
|
+
static composeRateText(
|
|
368
|
+
leftTicker,
|
|
369
|
+
rightTicker,
|
|
370
|
+
rate,
|
|
371
|
+
rightCurrencyDigitsAfterDots = this.significantDecimalCount
|
|
372
|
+
) {
|
|
373
|
+
try {
|
|
374
|
+
/* Here we try to calculate a clear rate for the user. The difficulty is that the rate value can be pretty
|
|
375
|
+
* small as some coins have significantly higher price than the other. For such cases we calculate
|
|
376
|
+
* not the "1 <coin_A> is X <coin B>" but "Y <coin_A> is X <coin B>" where Y is one of the powers of 100.
|
|
377
|
+
*/
|
|
378
|
+
let leftNumber = BigNumber("1");
|
|
379
|
+
const multiplier = BigNumber("100");
|
|
380
|
+
const maxAttemptsToGetRate = 10;
|
|
381
|
+
let right = null;
|
|
382
|
+
const rateBigNumber = BigNumber(rate);
|
|
383
|
+
for (let i = 0; i < maxAttemptsToGetRate; ++i) {
|
|
384
|
+
const rightNumberAttempt = rateBigNumber
|
|
385
|
+
.times(leftNumber)
|
|
386
|
+
.toFixed(
|
|
387
|
+
rightCurrencyDigitsAfterDots,
|
|
388
|
+
BigNumber.ROUND_FLOOR
|
|
389
|
+
);
|
|
390
|
+
if (!BigNumber(rightNumberAttempt).eq(BigNumber("0"))) {
|
|
391
|
+
right = BigNumber(rightNumberAttempt);
|
|
392
|
+
break;
|
|
393
|
+
} else {
|
|
394
|
+
leftNumber = leftNumber.times(multiplier);
|
|
395
|
+
}
|
|
396
|
+
}
|
|
397
|
+
const leftAmountString = AmountUtils.toIntegerString(leftNumber);
|
|
398
|
+
const rightAmountString =
|
|
399
|
+
right != null
|
|
400
|
+
? right.toFixed(
|
|
401
|
+
rightCurrencyDigitsAfterDots,
|
|
402
|
+
BigNumber.ROUND_FLOOR
|
|
403
|
+
)
|
|
404
|
+
: null;
|
|
405
|
+
return `${leftAmountString} ${leftTicker} ~ ${
|
|
406
|
+
rightAmountString ?? "?"
|
|
407
|
+
} ${rightTicker}`;
|
|
408
|
+
} catch (e) {
|
|
409
|
+
// eslint-disable-next-line no-console
|
|
410
|
+
console.log("composeRateText", e);
|
|
411
|
+
}
|
|
412
|
+
return "-";
|
|
413
|
+
}
|
|
414
|
+
|
|
415
|
+
/**
|
|
416
|
+
* @param numberAsAString {string}
|
|
417
|
+
* @return {string}
|
|
418
|
+
*/
|
|
419
|
+
static removeRedundantRightZerosFromNumberString(numberAsAString) {
|
|
420
|
+
try {
|
|
421
|
+
const parts = ("" + numberAsAString).split(".");
|
|
422
|
+
let right = parts[1];
|
|
423
|
+
while (right?.length && right[right.length - 1] === "0") {
|
|
424
|
+
right = right.slice(0, right.length - 1);
|
|
425
|
+
}
|
|
426
|
+
|
|
427
|
+
return `${parts[0]}${right?.length ? `.${right}` : ""}`;
|
|
428
|
+
} catch (e) {
|
|
429
|
+
improveAndRethrow(
|
|
430
|
+
e,
|
|
431
|
+
"removeRedundantRightZerosFromNumberString",
|
|
432
|
+
`Passed: ${numberAsAString}`
|
|
433
|
+
);
|
|
434
|
+
}
|
|
435
|
+
}
|
|
436
|
+
}
|
|
@@ -0,0 +1,42 @@
|
|
|
1
|
+
import { Logger } from "./utils/logging/logger.js";
|
|
2
|
+
import { safeStringify } from "./utils/safeStringify.js";
|
|
3
|
+
|
|
4
|
+
/**
|
|
5
|
+
* This function improves the passed error object (its message) by adding the passed function name
|
|
6
|
+
* and additional message to it.
|
|
7
|
+
* This is useful as Javascript doesn't guarantee the stack-traces, so we should manually add these details to errors
|
|
8
|
+
* to be able to troubleshoot.
|
|
9
|
+
*
|
|
10
|
+
* @param e {Error}
|
|
11
|
+
* @param settingFunction {string}
|
|
12
|
+
* @param [additionalMessage=""] {string|undefined}
|
|
13
|
+
* @throws {Error} always rethrows the original passed error but with an improved message
|
|
14
|
+
*/
|
|
15
|
+
export function improveAndRethrow(e, settingFunction, additionalMessage = "") {
|
|
16
|
+
const message = improvedErrorMessage(e, settingFunction, additionalMessage);
|
|
17
|
+
if (e) {
|
|
18
|
+
e.message = message;
|
|
19
|
+
throw e; // to preserve existing stacktrace if present
|
|
20
|
+
}
|
|
21
|
+
throw new Error(message);
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
function improvedErrorMessage(e, settingFunction, additionalMessage) {
|
|
25
|
+
let message = `\nFunction call ${settingFunction ?? ""} failed. `;
|
|
26
|
+
e && e.message && (message += `Error message: ${e.message}. `);
|
|
27
|
+
additionalMessage && (message += `${additionalMessage} `);
|
|
28
|
+
|
|
29
|
+
return message;
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
export function logErrorOrOutputToConsole(e) {
|
|
33
|
+
try {
|
|
34
|
+
// TODO: [dev] remove this after few weeks of testing output in real life
|
|
35
|
+
// eslint-disable-next-line no-console
|
|
36
|
+
console.log("BEFORE SAFE", e);
|
|
37
|
+
Logger.log("logErrorOrOutputToConsole", safeStringify(e));
|
|
38
|
+
} catch (e) {
|
|
39
|
+
// eslint-disable-next-line no-console
|
|
40
|
+
console.log("logErrorOrOutputToConsole", e);
|
|
41
|
+
}
|
|
42
|
+
}
|
|
@@ -0,0 +1,55 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Models a group of APIs provided by the same owner and used for different services in our app.
|
|
3
|
+
* It means we need to mention RPS several times for each usage and also have some holder of last call timestamp per
|
|
4
|
+
* api group. So this concept allows to use it for exact ExternalApiProvider and make sure that you use the same
|
|
5
|
+
* RPS value and make decisions on base of the same timestamp of last call to the API group owner.
|
|
6
|
+
*/
|
|
7
|
+
export class ApiGroup {
|
|
8
|
+
constructor(id, rps, backendProxyIdGenerator = null) {
|
|
9
|
+
this.id = id;
|
|
10
|
+
this.rps = rps;
|
|
11
|
+
this.lastCalledTimestamp = null;
|
|
12
|
+
this.backendProxyIdGenerator = backendProxyIdGenerator;
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
isRpsExceeded() {
|
|
16
|
+
return (this.lastCalledTimestamp ?? 0) + Math.floor(1000 / this.rps) > Date.now();
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
actualizeLastCalledTimestamp() {
|
|
20
|
+
this.lastCalledTimestamp = Date.now();
|
|
21
|
+
}
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
export const ApiGroups = {
|
|
25
|
+
/**
|
|
26
|
+
* Currently we use free version of etherscan provider with 0.2 RPS. But we have API key with 100k requests free
|
|
27
|
+
* per month. So we can add it if not enough current RPS.
|
|
28
|
+
*/
|
|
29
|
+
ETHERSCAN: new ApiGroup("etherscan", 0.17), // Actually 0.2 but fails sometime, so we use smaller
|
|
30
|
+
ALCHEMY: new ApiGroup("alchemy", 0.3, networkKey => `alchemy-${networkKey}`),
|
|
31
|
+
BLOCKSTREAM: new ApiGroup("blockstream", 0.2),
|
|
32
|
+
BLOCKCHAIN_INFO: new ApiGroup("blockchain.info", 1),
|
|
33
|
+
BLOCKNATIVE: new ApiGroup("blocknative", 0.5),
|
|
34
|
+
ETHGASSTATION: new ApiGroup("ethgasstation", 0.5),
|
|
35
|
+
TRONGRID: new ApiGroup("trongrid", 0.3, networkKey => `trongrid-${networkKey}`),
|
|
36
|
+
TRONSCAN: new ApiGroup("tronscan", 0.3),
|
|
37
|
+
GETBLOCK: new ApiGroup("getblock", 0.3),
|
|
38
|
+
COINCAP: new ApiGroup("coincap", 0.5), // 200 per minute without API key
|
|
39
|
+
COINGECKO: new ApiGroup("coingecko", 0.9), // actually 0.13-0.5 according to the docs but we use smaller due to expirienced frequent abuses
|
|
40
|
+
MESSARI: new ApiGroup("messari", 0.2),
|
|
41
|
+
BTCCOM: new ApiGroup("btccom", 0.2),
|
|
42
|
+
BITAPS: new ApiGroup("bitaps", 0.25), // Docs say that RPS is 3 but using it causes frequent 429 HTTP errors
|
|
43
|
+
CEX: new ApiGroup("cex", 0.5), // Just assumption for RPS
|
|
44
|
+
BIGDATACLOUD: new ApiGroup("bigdatacloud", 1), // Just assumption for RPS
|
|
45
|
+
TRACKIP: new ApiGroup("trackip", 1), // Just assumption for RPS
|
|
46
|
+
IPIFY: new ApiGroup("ipify", 1), // Just assumption for RPS
|
|
47
|
+
WHATISMYIPADDRESS: new ApiGroup("whatismyipaddress", 1), // Just assumption for RPS
|
|
48
|
+
EXCHANGERATE: new ApiGroup("exchangerate", 1), // Just assumption for RPS
|
|
49
|
+
FRANKFURTER: new ApiGroup("frankfurter", 1), // Just assumption for RPS
|
|
50
|
+
BITGO: new ApiGroup("bitgo", 1), // Just assumption for RPS
|
|
51
|
+
BITCOINER: new ApiGroup("bitcoiner", 1), // Just assumption for RPS
|
|
52
|
+
BITCORE: new ApiGroup("bitcore", 1), // Just assumption for RPS
|
|
53
|
+
// BLOCKCHAIR: new ApiGroup("blockchair", 0.04), // this provider require API key for commercial use (10usd 10000 reqs), we will add it later
|
|
54
|
+
MEMPOOL: new ApiGroup("mempool", 0.2), // Just assumption for RPS
|
|
55
|
+
};
|
|
@@ -0,0 +1,138 @@
|
|
|
1
|
+
import EventBusInstance from "eventbusjs";
|
|
2
|
+
import { ExternalApiProvider } from "../../robustExteranlApiCallerService/externalApiProvider.js";
|
|
3
|
+
import { ApiGroups } from "./apiGroups.js";
|
|
4
|
+
import { CachedRobustExternalApiCallerService } from "../../robustExteranlApiCallerService/cachedRobustExternalApiCallerService.js";
|
|
5
|
+
import { Cache } from "../utils/cache.js";
|
|
6
|
+
import { improveAndRethrow } from "../errorUtils.js";
|
|
7
|
+
|
|
8
|
+
class BigdatacloudIpAddressProvider extends ExternalApiProvider {
|
|
9
|
+
constructor() {
|
|
10
|
+
super(
|
|
11
|
+
"https://api.bigdatacloud.net/data/client-ip",
|
|
12
|
+
"get",
|
|
13
|
+
15000,
|
|
14
|
+
ApiGroups.BIGDATACLOUD
|
|
15
|
+
);
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
getDataByResponse(
|
|
19
|
+
response,
|
|
20
|
+
params = [],
|
|
21
|
+
subRequestIndex = 0,
|
|
22
|
+
iterationsData = []
|
|
23
|
+
) {
|
|
24
|
+
return response?.data && response.data?.ipString;
|
|
25
|
+
}
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
class TrackipIpAddressProvider extends ExternalApiProvider {
|
|
29
|
+
constructor() {
|
|
30
|
+
super("https://www.trackip.net/ip", "get", 15000, ApiGroups.TRACKIP);
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
getDataByResponse(
|
|
34
|
+
response,
|
|
35
|
+
params = [],
|
|
36
|
+
subRequestIndex = 0,
|
|
37
|
+
iterationsData = []
|
|
38
|
+
) {
|
|
39
|
+
return response?.data;
|
|
40
|
+
}
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
class IpifyV6IpAddressProvider extends ExternalApiProvider {
|
|
44
|
+
constructor() {
|
|
45
|
+
super(
|
|
46
|
+
"https://api6.ipify.org/?format=json",
|
|
47
|
+
"get",
|
|
48
|
+
15000,
|
|
49
|
+
ApiGroups.IPIFY
|
|
50
|
+
);
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
getDataByResponse(
|
|
54
|
+
response,
|
|
55
|
+
params = [],
|
|
56
|
+
subRequestIndex = 0,
|
|
57
|
+
iterationsData = []
|
|
58
|
+
) {
|
|
59
|
+
return response?.data && response.data?.ip;
|
|
60
|
+
}
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
class IpifyIpAddressProvider extends ExternalApiProvider {
|
|
64
|
+
constructor() {
|
|
65
|
+
super(
|
|
66
|
+
"https://api.ipify.org/?format=json",
|
|
67
|
+
"get",
|
|
68
|
+
15000,
|
|
69
|
+
ApiGroups.IPIFY
|
|
70
|
+
);
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
getDataByResponse(
|
|
74
|
+
response,
|
|
75
|
+
params = [],
|
|
76
|
+
subRequestIndex = 0,
|
|
77
|
+
iterationsData = []
|
|
78
|
+
) {
|
|
79
|
+
return response?.data && response.data?.ip;
|
|
80
|
+
}
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
class WhatismyipaddressIpAddressProvider extends ExternalApiProvider {
|
|
84
|
+
constructor() {
|
|
85
|
+
super(
|
|
86
|
+
"http://bot.whatismyipaddress.com/",
|
|
87
|
+
"get",
|
|
88
|
+
15000,
|
|
89
|
+
ApiGroups.WHATISMYIPADDRESS
|
|
90
|
+
);
|
|
91
|
+
}
|
|
92
|
+
|
|
93
|
+
getDataByResponse(
|
|
94
|
+
response,
|
|
95
|
+
params = [],
|
|
96
|
+
subRequestIndex = 0,
|
|
97
|
+
iterationsData = []
|
|
98
|
+
) {
|
|
99
|
+
return response?.data;
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
|
|
103
|
+
export class IpAddressProvider {
|
|
104
|
+
static externalIPAddressAPICaller =
|
|
105
|
+
new CachedRobustExternalApiCallerService(
|
|
106
|
+
"externalIPAddressAPICaller",
|
|
107
|
+
new Cache(EventBusInstance),
|
|
108
|
+
[
|
|
109
|
+
new BigdatacloudIpAddressProvider(),
|
|
110
|
+
new TrackipIpAddressProvider(),
|
|
111
|
+
new IpifyV6IpAddressProvider(),
|
|
112
|
+
new IpifyIpAddressProvider(),
|
|
113
|
+
new WhatismyipaddressIpAddressProvider(),
|
|
114
|
+
],
|
|
115
|
+
300_000
|
|
116
|
+
);
|
|
117
|
+
|
|
118
|
+
/**
|
|
119
|
+
* Returns current public IP address identified by one of external services.
|
|
120
|
+
*
|
|
121
|
+
* It is easier than manual identification and also (as ip needed for server side to check it) it saves us from
|
|
122
|
+
* issues related to changes of infrastructure configurations (like adding proxies etc.) so we should not configure
|
|
123
|
+
* anything on server side to get correct client's IP.
|
|
124
|
+
*
|
|
125
|
+
* @returns {Promise<String>} IP address
|
|
126
|
+
* @throws {Error} if fails to retrieve IP address from all the services
|
|
127
|
+
*/
|
|
128
|
+
static async getClientIpAddress() {
|
|
129
|
+
try {
|
|
130
|
+
return await this.externalIPAddressAPICaller.callExternalAPICached(
|
|
131
|
+
[],
|
|
132
|
+
7000
|
|
133
|
+
);
|
|
134
|
+
} catch (e) {
|
|
135
|
+
improveAndRethrow(e, "getClientIpAddress");
|
|
136
|
+
}
|
|
137
|
+
}
|
|
138
|
+
}
|