shopify_api 1.0.4 → 12.3.0
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.
- checksums.yaml +7 -0
- data/.github/CODEOWNERS +1 -0
- data/.github/ISSUE_TEMPLATE.md +35 -0
- data/.github/dependabot.yml +20 -0
- data/.github/pull_request_template.md +20 -0
- data/.github/workflows/build.yml +42 -0
- data/.github/workflows/cla.yml +22 -0
- data/.github/workflows/close-waiting-for-response-issues.yml +20 -0
- data/.github/workflows/remove-labels-on-activity.yml +16 -0
- data/.github/workflows/stale.yml +32 -0
- data/.rubocop.yml +59 -0
- data/BREAKING_CHANGES_FOR_OLDER_VERSIONS.md +110 -0
- data/CHANGELOG.md +574 -0
- data/CONTRIBUTING.md +9 -0
- data/Gemfile +13 -0
- data/Gemfile.lock +159 -0
- data/LICENSE +2 -2
- data/README.md +138 -0
- data/RELEASING.md +19 -0
- data/Rakefile +16 -50
- data/SECURITY.md +59 -0
- data/bin/tapioca +29 -0
- data/dev.yml +32 -0
- data/docs/README.md +13 -0
- data/docs/getting_started.md +53 -0
- data/docs/issues.md +39 -0
- data/docs/usage/graphql.md +115 -0
- data/docs/usage/graphql_storefront.md +42 -0
- data/docs/usage/oauth.md +104 -0
- data/docs/usage/rest.md +137 -0
- data/docs/usage/session_storage.md +46 -0
- data/docs/usage/webhooks.md +98 -0
- data/lib/shopify_api/admin_versions.rb +19 -0
- data/lib/shopify_api/auth/associated_user.rb +36 -0
- data/lib/shopify_api/auth/auth_scopes.rb +75 -0
- data/lib/shopify_api/auth/file_session_storage.rb +72 -0
- data/lib/shopify_api/auth/jwt_payload.rb +83 -0
- data/lib/shopify_api/auth/oauth/auth_query.rb +47 -0
- data/lib/shopify_api/auth/oauth/session_cookie.rb +28 -0
- data/lib/shopify_api/auth/oauth.rb +135 -0
- data/lib/shopify_api/auth/session.rb +119 -0
- data/lib/shopify_api/auth/session_storage.rb +30 -0
- data/lib/shopify_api/auth.rb +26 -0
- data/lib/shopify_api/clients/graphql/admin.rb +15 -0
- data/lib/shopify_api/clients/graphql/client.rb +40 -0
- data/lib/shopify_api/clients/graphql/storefront.rb +35 -0
- data/lib/shopify_api/clients/http_client.rb +100 -0
- data/lib/shopify_api/clients/http_request.rb +35 -0
- data/lib/shopify_api/clients/http_response.rb +66 -0
- data/lib/shopify_api/clients/rest/admin.rb +118 -0
- data/lib/shopify_api/context.rb +196 -0
- data/lib/shopify_api/errors/context_not_setup_error.rb +9 -0
- data/lib/shopify_api/errors/cookie_not_found_error.rb +9 -0
- data/lib/shopify_api/errors/feature_deprecated_error.rb +9 -0
- data/lib/shopify_api/errors/http_response_error.rb +23 -0
- data/lib/shopify_api/errors/invalid_graphql_request_error.rb +9 -0
- data/lib/shopify_api/errors/invalid_http_request_error.rb +9 -0
- data/lib/shopify_api/errors/invalid_jwt_token_error.rb +9 -0
- data/lib/shopify_api/errors/invalid_oauth_error.rb +9 -0
- data/lib/shopify_api/errors/invalid_webhook_error.rb +9 -0
- data/lib/shopify_api/errors/invalid_webhook_registration_error.rb +9 -0
- data/lib/shopify_api/errors/log_level_not_found_error.rb +9 -0
- data/lib/shopify_api/errors/max_http_retries_exceeded_error.rb +9 -0
- data/lib/shopify_api/errors/missing_jwt_token_error.rb +9 -0
- data/lib/shopify_api/errors/missing_required_argument_error.rb +9 -0
- data/lib/shopify_api/errors/no_active_session_error.rb +9 -0
- data/lib/shopify_api/errors/no_session_cookie_error.rb +9 -0
- data/lib/shopify_api/errors/no_webhook_handler.rb +9 -0
- data/lib/shopify_api/errors/private_app_error.rb +9 -0
- data/lib/shopify_api/errors/request_access_token_error.rb +9 -0
- data/lib/shopify_api/errors/session_not_found_error.rb +9 -0
- data/lib/shopify_api/errors/session_storage_error.rb +9 -0
- data/lib/shopify_api/errors/unsupported_oauth_error.rb +9 -0
- data/lib/shopify_api/errors/unsupported_version_error.rb +9 -0
- data/lib/shopify_api/errors/webhook_registration_error.rb +9 -0
- data/lib/shopify_api/inflector.rb +17 -0
- data/lib/shopify_api/logger.rb +82 -0
- data/lib/shopify_api/rest/base.rb +387 -0
- data/lib/shopify_api/rest/base_errors.rb +32 -0
- data/lib/shopify_api/rest/resources/2022_01/abandoned_checkout.rb +190 -0
- data/lib/shopify_api/rest/resources/2022_01/access_scope.rb +58 -0
- data/lib/shopify_api/rest/resources/2022_01/android_pay_key.rb +77 -0
- data/lib/shopify_api/rest/resources/2022_01/apple_pay_certificate.rb +105 -0
- data/lib/shopify_api/rest/resources/2022_01/application_charge.rb +104 -0
- data/lib/shopify_api/rest/resources/2022_01/application_credit.rb +87 -0
- data/lib/shopify_api/rest/resources/2022_01/article.rb +265 -0
- data/lib/shopify_api/rest/resources/2022_01/asset.rb +118 -0
- data/lib/shopify_api/rest/resources/2022_01/assigned_fulfillment_order.rb +79 -0
- data/lib/shopify_api/rest/resources/2022_01/balance.rb +50 -0
- data/lib/shopify_api/rest/resources/2022_01/blog.rb +162 -0
- data/lib/shopify_api/rest/resources/2022_01/cancellation_request.rb +83 -0
- data/lib/shopify_api/rest/resources/2022_01/carrier_service.rb +116 -0
- data/lib/shopify_api/rest/resources/2022_01/checkout.rb +209 -0
- data/lib/shopify_api/rest/resources/2022_01/collect.rb +142 -0
- data/lib/shopify_api/rest/resources/2022_01/collection.rb +110 -0
- data/lib/shopify_api/rest/resources/2022_01/collection_listing.rb +155 -0
- data/lib/shopify_api/rest/resources/2022_01/comment.rb +283 -0
- data/lib/shopify_api/rest/resources/2022_01/country.rb +137 -0
- data/lib/shopify_api/rest/resources/2022_01/currency.rb +57 -0
- data/lib/shopify_api/rest/resources/2022_01/custom_collection.rb +187 -0
- data/lib/shopify_api/rest/resources/2022_01/customer.rb +318 -0
- data/lib/shopify_api/rest/resources/2022_01/customer_address.rb +201 -0
- data/lib/shopify_api/rest/resources/2022_01/customer_saved_search.rb +169 -0
- data/lib/shopify_api/rest/resources/2022_01/deprecated_api_call.rb +57 -0
- data/lib/shopify_api/rest/resources/2022_01/discount_code.rb +219 -0
- data/lib/shopify_api/rest/resources/2022_01/dispute.rb +111 -0
- data/lib/shopify_api/rest/resources/2022_01/draft_order.rb +275 -0
- data/lib/shopify_api/rest/resources/2022_01/event.rb +148 -0
- data/lib/shopify_api/rest/resources/2022_01/fulfillment.rb +268 -0
- data/lib/shopify_api/rest/resources/2022_01/fulfillment_event.rb +166 -0
- data/lib/shopify_api/rest/resources/2022_01/fulfillment_order.rb +281 -0
- data/lib/shopify_api/rest/resources/2022_01/fulfillment_request.rb +87 -0
- data/lib/shopify_api/rest/resources/2022_01/fulfillment_service.rb +130 -0
- data/lib/shopify_api/rest/resources/2022_01/gift_card.rb +215 -0
- data/lib/shopify_api/rest/resources/2022_01/gift_card_adjustment.rb +118 -0
- data/lib/shopify_api/rest/resources/2022_01/image.rb +157 -0
- data/lib/shopify_api/rest/resources/2022_01/inventory_item.rb +108 -0
- data/lib/shopify_api/rest/resources/2022_01/inventory_level.rb +179 -0
- data/lib/shopify_api/rest/resources/2022_01/location.rb +167 -0
- data/lib/shopify_api/rest/resources/2022_01/locations_for_move.rb +56 -0
- data/lib/shopify_api/rest/resources/2022_01/marketing_event.rb +209 -0
- data/lib/shopify_api/rest/resources/2022_01/metafield.rb +349 -0
- data/lib/shopify_api/rest/resources/2022_01/mobile_platform_application.rb +110 -0
- data/lib/shopify_api/rest/resources/2022_01/order.rb +473 -0
- data/lib/shopify_api/rest/resources/2022_01/order_risk.rb +135 -0
- data/lib/shopify_api/rest/resources/2022_01/page.rb +194 -0
- data/lib/shopify_api/rest/resources/2022_01/payment.rb +140 -0
- data/lib/shopify_api/rest/resources/2022_01/payment_gateway.rb +143 -0
- data/lib/shopify_api/rest/resources/2022_01/payment_transaction.rb +107 -0
- data/lib/shopify_api/rest/resources/2022_01/payout.rb +97 -0
- data/lib/shopify_api/rest/resources/2022_01/policy.rb +69 -0
- data/lib/shopify_api/rest/resources/2022_01/price_rule.rb +223 -0
- data/lib/shopify_api/rest/resources/2022_01/product.rb +223 -0
- data/lib/shopify_api/rest/resources/2022_01/product_listing.rb +196 -0
- data/lib/shopify_api/rest/resources/2022_01/product_resource_feedback.rb +88 -0
- data/lib/shopify_api/rest/resources/2022_01/province.rb +132 -0
- data/lib/shopify_api/rest/resources/2022_01/recurring_application_charge.rb +167 -0
- data/lib/shopify_api/rest/resources/2022_01/redirect.rb +139 -0
- data/lib/shopify_api/rest/resources/2022_01/refund.rb +151 -0
- data/lib/shopify_api/rest/resources/2022_01/report.rb +121 -0
- data/lib/shopify_api/rest/resources/2022_01/resource_feedback.rb +73 -0
- data/lib/shopify_api/rest/resources/2022_01/script_tag.rb +155 -0
- data/lib/shopify_api/rest/resources/2022_01/shipping_zone.rb +83 -0
- data/lib/shopify_api/rest/resources/2022_01/shop.rb +218 -0
- data/lib/shopify_api/rest/resources/2022_01/smart_collection.rb +216 -0
- data/lib/shopify_api/rest/resources/2022_01/storefront_access_token.rb +87 -0
- data/lib/shopify_api/rest/resources/2022_01/tender_transaction.rb +93 -0
- data/lib/shopify_api/rest/resources/2022_01/theme.rb +120 -0
- data/lib/shopify_api/rest/resources/2022_01/transaction.rb +181 -0
- data/lib/shopify_api/rest/resources/2022_01/usage_charge.rb +97 -0
- data/lib/shopify_api/rest/resources/2022_01/user.rb +138 -0
- data/lib/shopify_api/rest/resources/2022_01/variant.rb +212 -0
- data/lib/shopify_api/rest/resources/2022_01/webhook.rb +168 -0
- data/lib/shopify_api/rest/resources/2022_04/abandoned_checkout.rb +190 -0
- data/lib/shopify_api/rest/resources/2022_04/access_scope.rb +58 -0
- data/lib/shopify_api/rest/resources/2022_04/android_pay_key.rb +77 -0
- data/lib/shopify_api/rest/resources/2022_04/apple_pay_certificate.rb +105 -0
- data/lib/shopify_api/rest/resources/2022_04/application_charge.rb +104 -0
- data/lib/shopify_api/rest/resources/2022_04/application_credit.rb +87 -0
- data/lib/shopify_api/rest/resources/2022_04/article.rb +265 -0
- data/lib/shopify_api/rest/resources/2022_04/asset.rb +118 -0
- data/lib/shopify_api/rest/resources/2022_04/assigned_fulfillment_order.rb +79 -0
- data/lib/shopify_api/rest/resources/2022_04/balance.rb +50 -0
- data/lib/shopify_api/rest/resources/2022_04/blog.rb +162 -0
- data/lib/shopify_api/rest/resources/2022_04/cancellation_request.rb +83 -0
- data/lib/shopify_api/rest/resources/2022_04/carrier_service.rb +116 -0
- data/lib/shopify_api/rest/resources/2022_04/checkout.rb +209 -0
- data/lib/shopify_api/rest/resources/2022_04/collect.rb +142 -0
- data/lib/shopify_api/rest/resources/2022_04/collection.rb +110 -0
- data/lib/shopify_api/rest/resources/2022_04/collection_listing.rb +155 -0
- data/lib/shopify_api/rest/resources/2022_04/comment.rb +283 -0
- data/lib/shopify_api/rest/resources/2022_04/country.rb +137 -0
- data/lib/shopify_api/rest/resources/2022_04/currency.rb +57 -0
- data/lib/shopify_api/rest/resources/2022_04/custom_collection.rb +187 -0
- data/lib/shopify_api/rest/resources/2022_04/customer.rb +321 -0
- data/lib/shopify_api/rest/resources/2022_04/customer_address.rb +201 -0
- data/lib/shopify_api/rest/resources/2022_04/customer_saved_search.rb +169 -0
- data/lib/shopify_api/rest/resources/2022_04/deprecated_api_call.rb +57 -0
- data/lib/shopify_api/rest/resources/2022_04/discount_code.rb +219 -0
- data/lib/shopify_api/rest/resources/2022_04/dispute.rb +111 -0
- data/lib/shopify_api/rest/resources/2022_04/draft_order.rb +275 -0
- data/lib/shopify_api/rest/resources/2022_04/event.rb +148 -0
- data/lib/shopify_api/rest/resources/2022_04/fulfillment.rb +268 -0
- data/lib/shopify_api/rest/resources/2022_04/fulfillment_event.rb +166 -0
- data/lib/shopify_api/rest/resources/2022_04/fulfillment_order.rb +284 -0
- data/lib/shopify_api/rest/resources/2022_04/fulfillment_request.rb +87 -0
- data/lib/shopify_api/rest/resources/2022_04/fulfillment_service.rb +130 -0
- data/lib/shopify_api/rest/resources/2022_04/gift_card.rb +215 -0
- data/lib/shopify_api/rest/resources/2022_04/gift_card_adjustment.rb +118 -0
- data/lib/shopify_api/rest/resources/2022_04/image.rb +157 -0
- data/lib/shopify_api/rest/resources/2022_04/inventory_item.rb +108 -0
- data/lib/shopify_api/rest/resources/2022_04/inventory_level.rb +179 -0
- data/lib/shopify_api/rest/resources/2022_04/location.rb +167 -0
- data/lib/shopify_api/rest/resources/2022_04/locations_for_move.rb +56 -0
- data/lib/shopify_api/rest/resources/2022_04/marketing_event.rb +209 -0
- data/lib/shopify_api/rest/resources/2022_04/metafield.rb +342 -0
- data/lib/shopify_api/rest/resources/2022_04/mobile_platform_application.rb +110 -0
- data/lib/shopify_api/rest/resources/2022_04/order.rb +473 -0
- data/lib/shopify_api/rest/resources/2022_04/order_risk.rb +135 -0
- data/lib/shopify_api/rest/resources/2022_04/page.rb +194 -0
- data/lib/shopify_api/rest/resources/2022_04/payment.rb +140 -0
- data/lib/shopify_api/rest/resources/2022_04/payment_gateway.rb +143 -0
- data/lib/shopify_api/rest/resources/2022_04/payment_transaction.rb +107 -0
- data/lib/shopify_api/rest/resources/2022_04/payout.rb +97 -0
- data/lib/shopify_api/rest/resources/2022_04/policy.rb +69 -0
- data/lib/shopify_api/rest/resources/2022_04/price_rule.rb +223 -0
- data/lib/shopify_api/rest/resources/2022_04/product.rb +223 -0
- data/lib/shopify_api/rest/resources/2022_04/product_listing.rb +196 -0
- data/lib/shopify_api/rest/resources/2022_04/product_resource_feedback.rb +88 -0
- data/lib/shopify_api/rest/resources/2022_04/province.rb +132 -0
- data/lib/shopify_api/rest/resources/2022_04/recurring_application_charge.rb +167 -0
- data/lib/shopify_api/rest/resources/2022_04/redirect.rb +139 -0
- data/lib/shopify_api/rest/resources/2022_04/refund.rb +151 -0
- data/lib/shopify_api/rest/resources/2022_04/report.rb +121 -0
- data/lib/shopify_api/rest/resources/2022_04/resource_feedback.rb +73 -0
- data/lib/shopify_api/rest/resources/2022_04/script_tag.rb +155 -0
- data/lib/shopify_api/rest/resources/2022_04/shipping_zone.rb +83 -0
- data/lib/shopify_api/rest/resources/2022_04/shop.rb +218 -0
- data/lib/shopify_api/rest/resources/2022_04/smart_collection.rb +216 -0
- data/lib/shopify_api/rest/resources/2022_04/storefront_access_token.rb +87 -0
- data/lib/shopify_api/rest/resources/2022_04/tender_transaction.rb +93 -0
- data/lib/shopify_api/rest/resources/2022_04/theme.rb +120 -0
- data/lib/shopify_api/rest/resources/2022_04/transaction.rb +181 -0
- data/lib/shopify_api/rest/resources/2022_04/usage_charge.rb +97 -0
- data/lib/shopify_api/rest/resources/2022_04/user.rb +138 -0
- data/lib/shopify_api/rest/resources/2022_04/variant.rb +212 -0
- data/lib/shopify_api/rest/resources/2022_04/webhook.rb +168 -0
- data/lib/shopify_api/rest/resources/2022_07/abandoned_checkout.rb +190 -0
- data/lib/shopify_api/rest/resources/2022_07/access_scope.rb +58 -0
- data/lib/shopify_api/rest/resources/2022_07/android_pay_key.rb +77 -0
- data/lib/shopify_api/rest/resources/2022_07/apple_pay_certificate.rb +105 -0
- data/lib/shopify_api/rest/resources/2022_07/application_charge.rb +104 -0
- data/lib/shopify_api/rest/resources/2022_07/application_credit.rb +87 -0
- data/lib/shopify_api/rest/resources/2022_07/article.rb +265 -0
- data/lib/shopify_api/rest/resources/2022_07/asset.rb +118 -0
- data/lib/shopify_api/rest/resources/2022_07/assigned_fulfillment_order.rb +79 -0
- data/lib/shopify_api/rest/resources/2022_07/balance.rb +50 -0
- data/lib/shopify_api/rest/resources/2022_07/blog.rb +162 -0
- data/lib/shopify_api/rest/resources/2022_07/cancellation_request.rb +83 -0
- data/lib/shopify_api/rest/resources/2022_07/carrier_service.rb +113 -0
- data/lib/shopify_api/rest/resources/2022_07/checkout.rb +209 -0
- data/lib/shopify_api/rest/resources/2022_07/collect.rb +142 -0
- data/lib/shopify_api/rest/resources/2022_07/collection.rb +110 -0
- data/lib/shopify_api/rest/resources/2022_07/collection_listing.rb +155 -0
- data/lib/shopify_api/rest/resources/2022_07/comment.rb +283 -0
- data/lib/shopify_api/rest/resources/2022_07/country.rb +137 -0
- data/lib/shopify_api/rest/resources/2022_07/currency.rb +57 -0
- data/lib/shopify_api/rest/resources/2022_07/custom_collection.rb +187 -0
- data/lib/shopify_api/rest/resources/2022_07/customer.rb +321 -0
- data/lib/shopify_api/rest/resources/2022_07/customer_address.rb +201 -0
- data/lib/shopify_api/rest/resources/2022_07/customer_saved_search.rb +169 -0
- data/lib/shopify_api/rest/resources/2022_07/deprecated_api_call.rb +57 -0
- data/lib/shopify_api/rest/resources/2022_07/discount_code.rb +219 -0
- data/lib/shopify_api/rest/resources/2022_07/dispute.rb +111 -0
- data/lib/shopify_api/rest/resources/2022_07/dispute_evidence.rb +117 -0
- data/lib/shopify_api/rest/resources/2022_07/dispute_file_upload.rb +81 -0
- data/lib/shopify_api/rest/resources/2022_07/draft_order.rb +275 -0
- data/lib/shopify_api/rest/resources/2022_07/event.rb +148 -0
- data/lib/shopify_api/rest/resources/2022_07/fulfillment.rb +221 -0
- data/lib/shopify_api/rest/resources/2022_07/fulfillment_event.rb +166 -0
- data/lib/shopify_api/rest/resources/2022_07/fulfillment_order.rb +310 -0
- data/lib/shopify_api/rest/resources/2022_07/fulfillment_request.rb +87 -0
- data/lib/shopify_api/rest/resources/2022_07/fulfillment_service.rb +130 -0
- data/lib/shopify_api/rest/resources/2022_07/gift_card.rb +215 -0
- data/lib/shopify_api/rest/resources/2022_07/gift_card_adjustment.rb +118 -0
- data/lib/shopify_api/rest/resources/2022_07/image.rb +157 -0
- data/lib/shopify_api/rest/resources/2022_07/inventory_item.rb +108 -0
- data/lib/shopify_api/rest/resources/2022_07/inventory_level.rb +179 -0
- data/lib/shopify_api/rest/resources/2022_07/location.rb +167 -0
- data/lib/shopify_api/rest/resources/2022_07/locations_for_move.rb +56 -0
- data/lib/shopify_api/rest/resources/2022_07/marketing_event.rb +209 -0
- data/lib/shopify_api/rest/resources/2022_07/metafield.rb +342 -0
- data/lib/shopify_api/rest/resources/2022_07/mobile_platform_application.rb +110 -0
- data/lib/shopify_api/rest/resources/2022_07/order.rb +473 -0
- data/lib/shopify_api/rest/resources/2022_07/order_risk.rb +135 -0
- data/lib/shopify_api/rest/resources/2022_07/page.rb +194 -0
- data/lib/shopify_api/rest/resources/2022_07/payment.rb +140 -0
- data/lib/shopify_api/rest/resources/2022_07/payment_gateway.rb +143 -0
- data/lib/shopify_api/rest/resources/2022_07/payment_transaction.rb +107 -0
- data/lib/shopify_api/rest/resources/2022_07/payout.rb +97 -0
- data/lib/shopify_api/rest/resources/2022_07/policy.rb +69 -0
- data/lib/shopify_api/rest/resources/2022_07/price_rule.rb +223 -0
- data/lib/shopify_api/rest/resources/2022_07/product.rb +223 -0
- data/lib/shopify_api/rest/resources/2022_07/product_listing.rb +196 -0
- data/lib/shopify_api/rest/resources/2022_07/product_resource_feedback.rb +88 -0
- data/lib/shopify_api/rest/resources/2022_07/province.rb +132 -0
- data/lib/shopify_api/rest/resources/2022_07/recurring_application_charge.rb +167 -0
- data/lib/shopify_api/rest/resources/2022_07/redirect.rb +139 -0
- data/lib/shopify_api/rest/resources/2022_07/refund.rb +151 -0
- data/lib/shopify_api/rest/resources/2022_07/report.rb +121 -0
- data/lib/shopify_api/rest/resources/2022_07/resource_feedback.rb +73 -0
- data/lib/shopify_api/rest/resources/2022_07/script_tag.rb +155 -0
- data/lib/shopify_api/rest/resources/2022_07/shipping_zone.rb +83 -0
- data/lib/shopify_api/rest/resources/2022_07/shop.rb +218 -0
- data/lib/shopify_api/rest/resources/2022_07/smart_collection.rb +216 -0
- data/lib/shopify_api/rest/resources/2022_07/storefront_access_token.rb +87 -0
- data/lib/shopify_api/rest/resources/2022_07/tender_transaction.rb +93 -0
- data/lib/shopify_api/rest/resources/2022_07/theme.rb +120 -0
- data/lib/shopify_api/rest/resources/2022_07/transaction.rb +181 -0
- data/lib/shopify_api/rest/resources/2022_07/usage_charge.rb +97 -0
- data/lib/shopify_api/rest/resources/2022_07/user.rb +138 -0
- data/lib/shopify_api/rest/resources/2022_07/variant.rb +212 -0
- data/lib/shopify_api/rest/resources/2022_07/webhook.rb +168 -0
- data/lib/shopify_api/rest/resources/2022_10/abandoned_checkout.rb +190 -0
- data/lib/shopify_api/rest/resources/2022_10/access_scope.rb +58 -0
- data/lib/shopify_api/rest/resources/2022_10/android_pay_key.rb +77 -0
- data/lib/shopify_api/rest/resources/2022_10/apple_pay_certificate.rb +105 -0
- data/lib/shopify_api/rest/resources/2022_10/application_charge.rb +104 -0
- data/lib/shopify_api/rest/resources/2022_10/application_credit.rb +87 -0
- data/lib/shopify_api/rest/resources/2022_10/article.rb +265 -0
- data/lib/shopify_api/rest/resources/2022_10/asset.rb +118 -0
- data/lib/shopify_api/rest/resources/2022_10/assigned_fulfillment_order.rb +79 -0
- data/lib/shopify_api/rest/resources/2022_10/balance.rb +50 -0
- data/lib/shopify_api/rest/resources/2022_10/blog.rb +162 -0
- data/lib/shopify_api/rest/resources/2022_10/cancellation_request.rb +83 -0
- data/lib/shopify_api/rest/resources/2022_10/carrier_service.rb +113 -0
- data/lib/shopify_api/rest/resources/2022_10/checkout.rb +209 -0
- data/lib/shopify_api/rest/resources/2022_10/collect.rb +142 -0
- data/lib/shopify_api/rest/resources/2022_10/collection.rb +110 -0
- data/lib/shopify_api/rest/resources/2022_10/collection_listing.rb +155 -0
- data/lib/shopify_api/rest/resources/2022_10/comment.rb +283 -0
- data/lib/shopify_api/rest/resources/2022_10/country.rb +137 -0
- data/lib/shopify_api/rest/resources/2022_10/currency.rb +57 -0
- data/lib/shopify_api/rest/resources/2022_10/custom_collection.rb +187 -0
- data/lib/shopify_api/rest/resources/2022_10/customer.rb +321 -0
- data/lib/shopify_api/rest/resources/2022_10/customer_address.rb +201 -0
- data/lib/shopify_api/rest/resources/2022_10/customer_saved_search.rb +169 -0
- data/lib/shopify_api/rest/resources/2022_10/deprecated_api_call.rb +57 -0
- data/lib/shopify_api/rest/resources/2022_10/discount_code.rb +219 -0
- data/lib/shopify_api/rest/resources/2022_10/dispute.rb +111 -0
- data/lib/shopify_api/rest/resources/2022_10/dispute_evidence.rb +117 -0
- data/lib/shopify_api/rest/resources/2022_10/dispute_file_upload.rb +81 -0
- data/lib/shopify_api/rest/resources/2022_10/draft_order.rb +275 -0
- data/lib/shopify_api/rest/resources/2022_10/event.rb +148 -0
- data/lib/shopify_api/rest/resources/2022_10/fulfillment.rb +221 -0
- data/lib/shopify_api/rest/resources/2022_10/fulfillment_event.rb +166 -0
- data/lib/shopify_api/rest/resources/2022_10/fulfillment_order.rb +310 -0
- data/lib/shopify_api/rest/resources/2022_10/fulfillment_request.rb +87 -0
- data/lib/shopify_api/rest/resources/2022_10/fulfillment_service.rb +130 -0
- data/lib/shopify_api/rest/resources/2022_10/gift_card.rb +215 -0
- data/lib/shopify_api/rest/resources/2022_10/gift_card_adjustment.rb +118 -0
- data/lib/shopify_api/rest/resources/2022_10/image.rb +157 -0
- data/lib/shopify_api/rest/resources/2022_10/inventory_item.rb +108 -0
- data/lib/shopify_api/rest/resources/2022_10/inventory_level.rb +179 -0
- data/lib/shopify_api/rest/resources/2022_10/location.rb +167 -0
- data/lib/shopify_api/rest/resources/2022_10/locations_for_move.rb +56 -0
- data/lib/shopify_api/rest/resources/2022_10/marketing_event.rb +209 -0
- data/lib/shopify_api/rest/resources/2022_10/metafield.rb +342 -0
- data/lib/shopify_api/rest/resources/2022_10/mobile_platform_application.rb +110 -0
- data/lib/shopify_api/rest/resources/2022_10/order.rb +476 -0
- data/lib/shopify_api/rest/resources/2022_10/order_risk.rb +135 -0
- data/lib/shopify_api/rest/resources/2022_10/page.rb +194 -0
- data/lib/shopify_api/rest/resources/2022_10/payment.rb +140 -0
- data/lib/shopify_api/rest/resources/2022_10/payment_gateway.rb +143 -0
- data/lib/shopify_api/rest/resources/2022_10/payment_transaction.rb +107 -0
- data/lib/shopify_api/rest/resources/2022_10/payout.rb +97 -0
- data/lib/shopify_api/rest/resources/2022_10/policy.rb +69 -0
- data/lib/shopify_api/rest/resources/2022_10/price_rule.rb +223 -0
- data/lib/shopify_api/rest/resources/2022_10/product.rb +223 -0
- data/lib/shopify_api/rest/resources/2022_10/product_listing.rb +196 -0
- data/lib/shopify_api/rest/resources/2022_10/product_resource_feedback.rb +88 -0
- data/lib/shopify_api/rest/resources/2022_10/province.rb +132 -0
- data/lib/shopify_api/rest/resources/2022_10/recurring_application_charge.rb +167 -0
- data/lib/shopify_api/rest/resources/2022_10/redirect.rb +139 -0
- data/lib/shopify_api/rest/resources/2022_10/refund.rb +151 -0
- data/lib/shopify_api/rest/resources/2022_10/report.rb +121 -0
- data/lib/shopify_api/rest/resources/2022_10/resource_feedback.rb +73 -0
- data/lib/shopify_api/rest/resources/2022_10/script_tag.rb +155 -0
- data/lib/shopify_api/rest/resources/2022_10/shipping_zone.rb +83 -0
- data/lib/shopify_api/rest/resources/2022_10/shop.rb +221 -0
- data/lib/shopify_api/rest/resources/2022_10/smart_collection.rb +216 -0
- data/lib/shopify_api/rest/resources/2022_10/storefront_access_token.rb +87 -0
- data/lib/shopify_api/rest/resources/2022_10/tender_transaction.rb +93 -0
- data/lib/shopify_api/rest/resources/2022_10/theme.rb +120 -0
- data/lib/shopify_api/rest/resources/2022_10/transaction.rb +181 -0
- data/lib/shopify_api/rest/resources/2022_10/usage_charge.rb +97 -0
- data/lib/shopify_api/rest/resources/2022_10/user.rb +138 -0
- data/lib/shopify_api/rest/resources/2022_10/variant.rb +212 -0
- data/lib/shopify_api/rest/resources/2022_10/webhook.rb +168 -0
- data/lib/shopify_api/utils/graphql_proxy.rb +52 -0
- data/lib/shopify_api/utils/hmac_validator.rb +44 -0
- data/lib/shopify_api/utils/http_utils.rb +17 -0
- data/lib/shopify_api/utils/session_utils.rb +152 -0
- data/lib/shopify_api/utils/verifiable_query.rb +18 -0
- data/lib/shopify_api/version.rb +6 -0
- data/lib/shopify_api/webhooks/handler.rb +15 -0
- data/lib/shopify_api/webhooks/register_result.rb +14 -0
- data/lib/shopify_api/webhooks/registration.rb +73 -0
- data/lib/shopify_api/webhooks/registrations/event_bridge.rb +61 -0
- data/lib/shopify_api/webhooks/registrations/http.rb +72 -0
- data/lib/shopify_api/webhooks/registrations/pub_sub.rb +65 -0
- data/lib/shopify_api/webhooks/registry.rb +215 -0
- data/lib/shopify_api/webhooks/request.rb +56 -0
- data/lib/shopify_api.rb +23 -472
- data/service.yml +1 -0
- data/shipit.rubygems.yml +1 -0
- data/shopify_api.gemspec +43 -102
- data/sorbet/config +3 -0
- data/sorbet/rbi/gems/activesupport@7.0.1.rbi +654 -0
- data/sorbet/rbi/gems/addressable@2.8.0.rbi +290 -0
- data/sorbet/rbi/gems/ast@2.4.2.rbi +54 -0
- data/sorbet/rbi/gems/coderay@1.1.3.rbi +8 -0
- data/sorbet/rbi/gems/concurrent-ruby@1.1.9.rbi +2401 -0
- data/sorbet/rbi/gems/crack@0.4.5.rbi +57 -0
- data/sorbet/rbi/gems/diff-lcs@1.5.0.rbi +185 -0
- data/sorbet/rbi/gems/fakefs@1.4.1.rbi +571 -0
- data/sorbet/rbi/gems/hash_diff@1.0.0.rbi +47 -0
- data/sorbet/rbi/gems/hashdiff@1.0.1.rbi +82 -0
- data/sorbet/rbi/gems/httparty@0.20.0.rbi +573 -0
- data/sorbet/rbi/gems/i18n@1.8.11.rbi +25 -0
- data/sorbet/rbi/gems/jwt@2.3.0.rbi +437 -0
- data/sorbet/rbi/gems/method_source@1.0.0.rbi +8 -0
- data/sorbet/rbi/gems/mime-types-data@3.2022.0105.rbi +73 -0
- data/sorbet/rbi/gems/mime-types@3.4.1.rbi +295 -0
- data/sorbet/rbi/gems/minitest@5.15.0.rbi +541 -0
- data/sorbet/rbi/gems/mocha@1.13.0.rbi +986 -0
- data/sorbet/rbi/gems/multi_xml@0.6.0.rbi +36 -0
- data/sorbet/rbi/gems/oj@3.13.11.rbi +274 -0
- data/sorbet/rbi/gems/openssl@3.0.0.rbi +581 -0
- data/sorbet/rbi/gems/parallel@1.21.0.rbi +113 -0
- data/sorbet/rbi/gems/parser@3.1.0.0.rbi +1741 -0
- data/sorbet/rbi/gems/pry@0.14.1.rbi +8 -0
- data/sorbet/rbi/gems/public_suffix@4.0.6.rbi +145 -0
- data/sorbet/rbi/gems/rainbow@3.1.1.rbi +157 -0
- data/sorbet/rbi/gems/rake@13.0.6.rbi +814 -0
- data/sorbet/rbi/gems/rbi@0.0.11.rbi +1646 -0
- data/sorbet/rbi/gems/regexp_parser@2.2.0.rbi +1130 -0
- data/sorbet/rbi/gems/rexml@3.2.5.rbi +709 -0
- data/sorbet/rbi/gems/rubocop-ast@1.15.1.rbi +1921 -0
- data/sorbet/rbi/gems/rubocop-shopify@2.4.0.rbi +8 -0
- data/sorbet/rbi/gems/rubocop-sorbet@0.6.5.rbi +295 -0
- data/sorbet/rbi/gems/rubocop@1.25.1.rbi +13507 -0
- data/sorbet/rbi/gems/ruby-progressbar@1.11.0.rbi +405 -0
- data/sorbet/rbi/gems/securerandom@0.1.1.rbi +10 -0
- data/sorbet/rbi/gems/spoom@1.1.8.rbi +1252 -0
- data/sorbet/rbi/gems/tapioca@0.6.3.rbi +1238 -0
- data/sorbet/rbi/gems/thor@1.2.1.rbi +844 -0
- data/sorbet/rbi/gems/tzinfo@2.0.4.rbi +858 -0
- data/sorbet/rbi/gems/unicode-display_width@2.1.0.rbi +26 -0
- data/sorbet/rbi/gems/unparser@0.6.3.rbi +1816 -0
- data/sorbet/rbi/gems/webmock@3.14.0.rbi +683 -0
- data/sorbet/rbi/gems/webrick@1.7.0.rbi +601 -0
- data/sorbet/rbi/gems/yard-sorbet@0.6.1.rbi +199 -0
- data/sorbet/rbi/gems/yard@0.9.27.rbi +4145 -0
- data/sorbet/rbi/gems/zeitwerk@2.5.4.rbi +200 -0
- data/sorbet/rbi/shims/fakefs.rbi +1 -0
- data/sorbet/rbi/shims/openssl.rb +3 -0
- data/sorbet/rbi/todo.rbi +8 -0
- data/sorbet/tapioca/config.yml +4 -0
- data/sorbet/tapioca/require.rb +20 -0
- metadata +736 -80
- data/.document +0 -5
- data/.gitignore +0 -5
- data/CHANGELOG +0 -5
- data/README.rdoc +0 -55
- data/VERSION +0 -1
- data/test/order_test.rb +0 -48
- data/test/shopify_api_test.rb +0 -21
- data/test/test_helper.rb +0 -28
|
@@ -0,0 +1,72 @@
|
|
|
1
|
+
# typed: strict
|
|
2
|
+
# frozen_string_literal: true
|
|
3
|
+
|
|
4
|
+
require "fileutils"
|
|
5
|
+
|
|
6
|
+
module ShopifyAPI
|
|
7
|
+
module Auth
|
|
8
|
+
class FileSessionStorage
|
|
9
|
+
extend T::Sig
|
|
10
|
+
extend T::Helpers
|
|
11
|
+
include ShopifyAPI::Auth::SessionStorage
|
|
12
|
+
|
|
13
|
+
sig { returns(String) }
|
|
14
|
+
attr_accessor :path
|
|
15
|
+
|
|
16
|
+
sig { params(path: String).void }
|
|
17
|
+
def initialize(path: "/tmp/shopify_api_sessions")
|
|
18
|
+
@path = path
|
|
19
|
+
FileUtils.mkdir_p(path) unless Dir.exist?(path)
|
|
20
|
+
end
|
|
21
|
+
|
|
22
|
+
sig do
|
|
23
|
+
override.params(session: Session)
|
|
24
|
+
.returns(T::Boolean)
|
|
25
|
+
end
|
|
26
|
+
def store_session(session)
|
|
27
|
+
File.write(session_file_path(session.id), session.serialize) > 0
|
|
28
|
+
end
|
|
29
|
+
|
|
30
|
+
sig do
|
|
31
|
+
override.params(id: String)
|
|
32
|
+
.returns(T.nilable(Session))
|
|
33
|
+
end
|
|
34
|
+
def load_session(id)
|
|
35
|
+
session_path = session_file_path(id)
|
|
36
|
+
if File.exist?(session_path)
|
|
37
|
+
ShopifyAPI::Auth::Session.deserialize(File.read(session_path))
|
|
38
|
+
end
|
|
39
|
+
end
|
|
40
|
+
|
|
41
|
+
sig do
|
|
42
|
+
override.params(id: String)
|
|
43
|
+
.returns(T::Boolean)
|
|
44
|
+
end
|
|
45
|
+
def delete_session(id)
|
|
46
|
+
session_path = session_file_path(id)
|
|
47
|
+
File.delete(session_path) if File.exist?(session_path)
|
|
48
|
+
true
|
|
49
|
+
end
|
|
50
|
+
|
|
51
|
+
alias_method :eql?, :==
|
|
52
|
+
sig { params(other: T.nilable(FileSessionStorage)).returns(T::Boolean) }
|
|
53
|
+
def ==(other)
|
|
54
|
+
if other
|
|
55
|
+
@path == other.path
|
|
56
|
+
else
|
|
57
|
+
false
|
|
58
|
+
end
|
|
59
|
+
end
|
|
60
|
+
|
|
61
|
+
private
|
|
62
|
+
|
|
63
|
+
sig do
|
|
64
|
+
params(id: String)
|
|
65
|
+
.returns(String)
|
|
66
|
+
end
|
|
67
|
+
def session_file_path(id)
|
|
68
|
+
"#{@path}/#{id}"
|
|
69
|
+
end
|
|
70
|
+
end
|
|
71
|
+
end
|
|
72
|
+
end
|
|
@@ -0,0 +1,83 @@
|
|
|
1
|
+
# typed: strict
|
|
2
|
+
# frozen_string_literal: true
|
|
3
|
+
|
|
4
|
+
module ShopifyAPI
|
|
5
|
+
module Auth
|
|
6
|
+
class JwtPayload
|
|
7
|
+
extend T::Sig
|
|
8
|
+
|
|
9
|
+
JWT_EXPIRATION_LEEWAY = 10
|
|
10
|
+
|
|
11
|
+
sig { returns(String) }
|
|
12
|
+
attr_reader :iss, :dest, :aud, :sub, :jti, :sid
|
|
13
|
+
|
|
14
|
+
sig { returns(Integer) }
|
|
15
|
+
attr_reader :exp, :nbf, :iat
|
|
16
|
+
|
|
17
|
+
sig { params(token: String).void }
|
|
18
|
+
def initialize(token)
|
|
19
|
+
payload_hash = begin
|
|
20
|
+
decode_token(token, Context.api_secret_key)
|
|
21
|
+
rescue ShopifyAPI::Errors::InvalidJwtTokenError
|
|
22
|
+
raise unless Context.old_api_secret_key
|
|
23
|
+
|
|
24
|
+
decode_token(token, T.must(Context.old_api_secret_key))
|
|
25
|
+
end
|
|
26
|
+
|
|
27
|
+
@iss = T.let(payload_hash["iss"], String)
|
|
28
|
+
@dest = T.let(payload_hash["dest"], String)
|
|
29
|
+
@aud = T.let(payload_hash["aud"], String)
|
|
30
|
+
@sub = T.let(payload_hash["sub"], String)
|
|
31
|
+
@exp = T.let(payload_hash["exp"], Integer)
|
|
32
|
+
@nbf = T.let(payload_hash["nbf"], Integer)
|
|
33
|
+
@iat = T.let(payload_hash["iat"], Integer)
|
|
34
|
+
@jti = T.let(payload_hash["jti"], String)
|
|
35
|
+
@sid = T.let(payload_hash["sid"], String)
|
|
36
|
+
|
|
37
|
+
raise ShopifyAPI::Errors::InvalidJwtTokenError,
|
|
38
|
+
"Session token had invalid API key" unless @aud == Context.api_key
|
|
39
|
+
end
|
|
40
|
+
|
|
41
|
+
sig { returns(String) }
|
|
42
|
+
def shop
|
|
43
|
+
@dest.gsub("https://", "")
|
|
44
|
+
end
|
|
45
|
+
|
|
46
|
+
# TODO: Remove before releasing v11
|
|
47
|
+
sig { params(shop: String).returns(T::Boolean) }
|
|
48
|
+
def validate_shop(shop)
|
|
49
|
+
Context.logger.warn(
|
|
50
|
+
"Deprecation notice: ShopifyAPI::Auth::JwtPayload.validate_shop no longer checks the given shop and always " \
|
|
51
|
+
"returns true. It will be removed in v11.",
|
|
52
|
+
)
|
|
53
|
+
true
|
|
54
|
+
end
|
|
55
|
+
|
|
56
|
+
alias_method :eql?, :==
|
|
57
|
+
sig { params(other: T.nilable(JwtPayload)).returns(T::Boolean) }
|
|
58
|
+
def ==(other)
|
|
59
|
+
return false unless other
|
|
60
|
+
|
|
61
|
+
iss == other.iss &&
|
|
62
|
+
dest == other.dest &&
|
|
63
|
+
aud == other.aud &&
|
|
64
|
+
sub == other.sub &&
|
|
65
|
+
exp == other.exp &&
|
|
66
|
+
nbf == other.nbf &&
|
|
67
|
+
iat == other.iat &&
|
|
68
|
+
jti == other.jti &&
|
|
69
|
+
sid == other.sid
|
|
70
|
+
end
|
|
71
|
+
|
|
72
|
+
private
|
|
73
|
+
|
|
74
|
+
sig { params(token: String, api_secret_key: String).returns(T::Hash[String, T.untyped]) }
|
|
75
|
+
def decode_token(token, api_secret_key)
|
|
76
|
+
JWT.decode(token, api_secret_key, true,
|
|
77
|
+
{ exp_leeway: JWT_EXPIRATION_LEEWAY, algorithm: "HS256" })[0]
|
|
78
|
+
rescue
|
|
79
|
+
raise ShopifyAPI::Errors::InvalidJwtTokenError, "Failed to parse session token '#{token}'"
|
|
80
|
+
end
|
|
81
|
+
end
|
|
82
|
+
end
|
|
83
|
+
end
|
|
@@ -0,0 +1,47 @@
|
|
|
1
|
+
# typed: strict
|
|
2
|
+
# frozen_string_literal: true
|
|
3
|
+
|
|
4
|
+
module ShopifyAPI
|
|
5
|
+
module Auth
|
|
6
|
+
module Oauth
|
|
7
|
+
class AuthQuery
|
|
8
|
+
extend T::Sig
|
|
9
|
+
include Utils::VerifiableQuery
|
|
10
|
+
|
|
11
|
+
sig { returns(String) }
|
|
12
|
+
attr_reader :code, :host, :hmac, :shop, :state, :timestamp
|
|
13
|
+
|
|
14
|
+
sig do
|
|
15
|
+
params(
|
|
16
|
+
code: String,
|
|
17
|
+
shop: String,
|
|
18
|
+
timestamp: String,
|
|
19
|
+
state: String,
|
|
20
|
+
host: String,
|
|
21
|
+
hmac: String,
|
|
22
|
+
).void
|
|
23
|
+
end
|
|
24
|
+
def initialize(code:, shop:, timestamp:, state:, host:, hmac:)
|
|
25
|
+
@code = code
|
|
26
|
+
@shop = shop
|
|
27
|
+
@timestamp = timestamp
|
|
28
|
+
@state = state
|
|
29
|
+
@host = host
|
|
30
|
+
@hmac = hmac
|
|
31
|
+
end
|
|
32
|
+
|
|
33
|
+
sig { override.returns(String) }
|
|
34
|
+
def to_signable_string
|
|
35
|
+
params = {
|
|
36
|
+
code: code,
|
|
37
|
+
host: host,
|
|
38
|
+
shop: shop,
|
|
39
|
+
state: state,
|
|
40
|
+
timestamp: timestamp,
|
|
41
|
+
}
|
|
42
|
+
URI.encode_www_form(params)
|
|
43
|
+
end
|
|
44
|
+
end
|
|
45
|
+
end
|
|
46
|
+
end
|
|
47
|
+
end
|
|
@@ -0,0 +1,28 @@
|
|
|
1
|
+
# typed: strict
|
|
2
|
+
# frozen_string_literal: true
|
|
3
|
+
|
|
4
|
+
module ShopifyAPI
|
|
5
|
+
module Auth
|
|
6
|
+
module Oauth
|
|
7
|
+
class SessionCookie < T::Struct
|
|
8
|
+
extend T::Sig
|
|
9
|
+
|
|
10
|
+
SESSION_COOKIE_NAME = "shopify_app_session"
|
|
11
|
+
|
|
12
|
+
const :name, String, default: SESSION_COOKIE_NAME
|
|
13
|
+
const :value, String
|
|
14
|
+
const :expires, T.nilable(Time)
|
|
15
|
+
|
|
16
|
+
alias_method :eql?, :==
|
|
17
|
+
sig { params(other: T.nilable(SessionCookie)).returns(T::Boolean) }
|
|
18
|
+
def ==(other)
|
|
19
|
+
return false unless other
|
|
20
|
+
|
|
21
|
+
name == other.name &&
|
|
22
|
+
value == other.value &&
|
|
23
|
+
expires == other.expires
|
|
24
|
+
end
|
|
25
|
+
end
|
|
26
|
+
end
|
|
27
|
+
end
|
|
28
|
+
end
|
|
@@ -0,0 +1,135 @@
|
|
|
1
|
+
# typed: strict
|
|
2
|
+
# frozen_string_literal: true
|
|
3
|
+
|
|
4
|
+
module ShopifyAPI
|
|
5
|
+
module Auth
|
|
6
|
+
module Oauth
|
|
7
|
+
extend T::Sig
|
|
8
|
+
|
|
9
|
+
NONCE_LENGTH = 15
|
|
10
|
+
|
|
11
|
+
class << self
|
|
12
|
+
extend T::Sig
|
|
13
|
+
|
|
14
|
+
sig do
|
|
15
|
+
params(
|
|
16
|
+
shop: String,
|
|
17
|
+
redirect_path: String,
|
|
18
|
+
is_online: T.nilable(T::Boolean),
|
|
19
|
+
scope_override: T.nilable(T.any(ShopifyAPI::Auth::AuthScopes, T::Array[String], String)),
|
|
20
|
+
).returns(T::Hash[Symbol, T.any(String, SessionCookie)])
|
|
21
|
+
end
|
|
22
|
+
def begin_auth(shop:, redirect_path:, is_online: true, scope_override: nil)
|
|
23
|
+
scope = if scope_override.nil?
|
|
24
|
+
ShopifyAPI::Context.scope
|
|
25
|
+
elsif scope_override.is_a?(ShopifyAPI::Auth::AuthScopes)
|
|
26
|
+
scope_override
|
|
27
|
+
else
|
|
28
|
+
ShopifyAPI::Auth::AuthScopes.new(scope_override)
|
|
29
|
+
end
|
|
30
|
+
|
|
31
|
+
unless Context.setup?
|
|
32
|
+
raise Errors::ContextNotSetupError, "ShopifyAPI::Context not setup, please call ShopifyAPI::Context.setup"
|
|
33
|
+
end
|
|
34
|
+
raise Errors::UnsupportedOauthError, "Cannot perform OAuth for private apps." if Context.private?
|
|
35
|
+
|
|
36
|
+
state = SecureRandom.alphanumeric(NONCE_LENGTH)
|
|
37
|
+
|
|
38
|
+
cookie = SessionCookie.new(value: state, expires: Time.now + 60)
|
|
39
|
+
|
|
40
|
+
query = {
|
|
41
|
+
client_id: ShopifyAPI::Context.api_key,
|
|
42
|
+
scope: scope.to_s,
|
|
43
|
+
redirect_uri: "#{ShopifyAPI::Context.host}#{redirect_path}",
|
|
44
|
+
state: state,
|
|
45
|
+
"grant_options[]": is_online ? "per-user" : "",
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
query_string = URI.encode_www_form(query)
|
|
49
|
+
|
|
50
|
+
auth_route = "https://#{shop}/admin/oauth/authorize?#{query_string}"
|
|
51
|
+
{ auth_route: auth_route, cookie: cookie }
|
|
52
|
+
end
|
|
53
|
+
|
|
54
|
+
sig do
|
|
55
|
+
params(
|
|
56
|
+
cookies: T::Hash[String, String],
|
|
57
|
+
auth_query: AuthQuery,
|
|
58
|
+
).returns(T::Hash[Symbol, T.any(Session, SessionCookie)])
|
|
59
|
+
end
|
|
60
|
+
def validate_auth_callback(cookies:, auth_query:)
|
|
61
|
+
unless Context.setup?
|
|
62
|
+
raise Errors::ContextNotSetupError, "ShopifyAPI::Context not setup, please call ShopifyAPI::Context.setup"
|
|
63
|
+
end
|
|
64
|
+
raise Errors::InvalidOauthError, "Invalid OAuth callback." unless Utils::HmacValidator.validate(auth_query)
|
|
65
|
+
raise Errors::UnsupportedOauthError, "Cannot perform OAuth for private apps." if Context.private?
|
|
66
|
+
|
|
67
|
+
state = cookies[SessionCookie::SESSION_COOKIE_NAME]
|
|
68
|
+
raise Errors::NoSessionCookieError unless state
|
|
69
|
+
|
|
70
|
+
raise Errors::InvalidOauthError,
|
|
71
|
+
"Invalid state in OAuth callback." unless state == auth_query.state
|
|
72
|
+
|
|
73
|
+
# TODO: replace this call with the HTTP client once it is built
|
|
74
|
+
body = { client_id: Context.api_key, client_secret: Context.api_secret_key, code: auth_query.code }
|
|
75
|
+
response = HTTParty.post("https://#{auth_query.shop}/admin/oauth/access_token", body: body)
|
|
76
|
+
unless response.ok?
|
|
77
|
+
raise Errors::RequestAccessTokenError,
|
|
78
|
+
"Cannot complete OAuth process. Received a #{response.code} error while requesting access token."
|
|
79
|
+
end
|
|
80
|
+
session_params = response.to_h
|
|
81
|
+
|
|
82
|
+
session = create_new_session(session_params, auth_query.shop)
|
|
83
|
+
|
|
84
|
+
cookie = if Context.embedded?
|
|
85
|
+
SessionCookie.new(
|
|
86
|
+
value: "",
|
|
87
|
+
expires: Time.now,
|
|
88
|
+
)
|
|
89
|
+
else
|
|
90
|
+
SessionCookie.new(
|
|
91
|
+
value: session.id,
|
|
92
|
+
expires: session.online? ? session.expires : nil,
|
|
93
|
+
)
|
|
94
|
+
end
|
|
95
|
+
|
|
96
|
+
Context.session_storage&.store_session(session)
|
|
97
|
+
|
|
98
|
+
{ session: session, cookie: cookie }
|
|
99
|
+
end
|
|
100
|
+
|
|
101
|
+
private
|
|
102
|
+
|
|
103
|
+
sig { params(session_params: T::Hash[String, T.untyped], shop: String).returns(Session) }
|
|
104
|
+
def create_new_session(session_params, shop)
|
|
105
|
+
session_params = session_params.to_h { |k, v| [k.to_sym, v] }
|
|
106
|
+
|
|
107
|
+
scope = session_params[:scope]
|
|
108
|
+
|
|
109
|
+
is_online = !session_params[:associated_user].nil?
|
|
110
|
+
|
|
111
|
+
if is_online
|
|
112
|
+
associated_user = AssociatedUser.new(session_params[:associated_user].to_h { |k, v| [k.to_sym, v] })
|
|
113
|
+
expires = Time.now + session_params[:expires_in].to_i
|
|
114
|
+
associated_user_scope = session_params[:associated_user_scope]
|
|
115
|
+
id = "#{shop}_#{associated_user.id}"
|
|
116
|
+
else
|
|
117
|
+
id = "offline_#{shop}"
|
|
118
|
+
end
|
|
119
|
+
|
|
120
|
+
Session.new(
|
|
121
|
+
id: id,
|
|
122
|
+
shop: shop,
|
|
123
|
+
access_token: session_params[:access_token],
|
|
124
|
+
scope: scope,
|
|
125
|
+
is_online: is_online,
|
|
126
|
+
associated_user_scope: associated_user_scope,
|
|
127
|
+
associated_user: associated_user,
|
|
128
|
+
expires: expires,
|
|
129
|
+
shopify_session_id: session_params[:session],
|
|
130
|
+
)
|
|
131
|
+
end
|
|
132
|
+
end
|
|
133
|
+
end
|
|
134
|
+
end
|
|
135
|
+
end
|
|
@@ -0,0 +1,119 @@
|
|
|
1
|
+
# typed: strict
|
|
2
|
+
# frozen_string_literal: true
|
|
3
|
+
|
|
4
|
+
module ShopifyAPI
|
|
5
|
+
module Auth
|
|
6
|
+
class Session
|
|
7
|
+
extend T::Sig
|
|
8
|
+
|
|
9
|
+
sig { returns(String) }
|
|
10
|
+
attr_reader :id
|
|
11
|
+
|
|
12
|
+
sig { returns(T.nilable(String)) }
|
|
13
|
+
attr_accessor :state, :access_token
|
|
14
|
+
|
|
15
|
+
sig { returns(String) }
|
|
16
|
+
attr_accessor :shop
|
|
17
|
+
|
|
18
|
+
sig { returns(AuthScopes) }
|
|
19
|
+
attr_accessor :scope
|
|
20
|
+
|
|
21
|
+
sig { returns(T.nilable(AuthScopes)) }
|
|
22
|
+
attr_accessor :associated_user_scope
|
|
23
|
+
|
|
24
|
+
sig { returns(T.nilable(Time)) }
|
|
25
|
+
attr_accessor :expires
|
|
26
|
+
|
|
27
|
+
sig { returns(T.nilable(AssociatedUser)) }
|
|
28
|
+
attr_accessor :associated_user
|
|
29
|
+
|
|
30
|
+
sig { returns(T.nilable(String)) }
|
|
31
|
+
attr_accessor :shopify_session_id
|
|
32
|
+
|
|
33
|
+
sig { returns(T::Boolean) }
|
|
34
|
+
def online?
|
|
35
|
+
@is_online
|
|
36
|
+
end
|
|
37
|
+
|
|
38
|
+
sig do
|
|
39
|
+
params(
|
|
40
|
+
shop: String,
|
|
41
|
+
id: T.nilable(String),
|
|
42
|
+
state: T.nilable(String),
|
|
43
|
+
access_token: T.nilable(String),
|
|
44
|
+
scope: T.any(T::Array[String], String),
|
|
45
|
+
associated_user_scope: T.nilable(T.any(T::Array[String], String)),
|
|
46
|
+
expires: T.nilable(Time),
|
|
47
|
+
is_online: T.nilable(T::Boolean),
|
|
48
|
+
associated_user: T.nilable(AssociatedUser),
|
|
49
|
+
shopify_session_id: T.nilable(String),
|
|
50
|
+
).void
|
|
51
|
+
end
|
|
52
|
+
def initialize(shop:, id: nil, state: nil, access_token: "", scope: [], associated_user_scope: nil, expires: nil,
|
|
53
|
+
is_online: nil, associated_user: nil, shopify_session_id: nil)
|
|
54
|
+
@id = T.let(id || SecureRandom.uuid, String)
|
|
55
|
+
@shop = shop
|
|
56
|
+
@state = state
|
|
57
|
+
@access_token = access_token
|
|
58
|
+
@scope = T.let(AuthScopes.new(scope), AuthScopes)
|
|
59
|
+
@associated_user_scope = T.let(
|
|
60
|
+
associated_user_scope.nil? ? nil : AuthScopes.new(associated_user_scope), T.nilable(AuthScopes)
|
|
61
|
+
)
|
|
62
|
+
@expires = expires
|
|
63
|
+
@associated_user = associated_user
|
|
64
|
+
@is_online = T.let(is_online || !associated_user.nil?, T::Boolean)
|
|
65
|
+
@shopify_session_id = shopify_session_id
|
|
66
|
+
end
|
|
67
|
+
|
|
68
|
+
class << self
|
|
69
|
+
extend T::Sig
|
|
70
|
+
|
|
71
|
+
sig do
|
|
72
|
+
params(shop: String, access_token: String,
|
|
73
|
+
blk: T.proc.params(arg0: Session).returns(T.untyped)).returns(T.untyped)
|
|
74
|
+
end
|
|
75
|
+
def temp(shop:, access_token:, &blk)
|
|
76
|
+
original_session = Context.active_session
|
|
77
|
+
temp_session = Session.new(shop: shop, access_token: access_token)
|
|
78
|
+
|
|
79
|
+
begin
|
|
80
|
+
Context.activate_session(temp_session)
|
|
81
|
+
yield temp_session
|
|
82
|
+
ensure
|
|
83
|
+
Context.activate_session(original_session)
|
|
84
|
+
end
|
|
85
|
+
end
|
|
86
|
+
|
|
87
|
+
sig { params(str: String).returns(Session) }
|
|
88
|
+
def deserialize(str)
|
|
89
|
+
Oj.load(str)
|
|
90
|
+
end
|
|
91
|
+
end
|
|
92
|
+
|
|
93
|
+
sig { returns(String) }
|
|
94
|
+
def serialize
|
|
95
|
+
Oj.dump(self)
|
|
96
|
+
end
|
|
97
|
+
|
|
98
|
+
alias_method :eql?, :==
|
|
99
|
+
sig { params(other: T.nilable(Session)).returns(T::Boolean) }
|
|
100
|
+
def ==(other)
|
|
101
|
+
if other
|
|
102
|
+
(
|
|
103
|
+
id == other.id &&
|
|
104
|
+
shop == other.shop &&
|
|
105
|
+
state == other.state &&
|
|
106
|
+
scope == other.scope &&
|
|
107
|
+
associated_user_scope == other.associated_user_scope &&
|
|
108
|
+
(!(expires.nil? ^ other.expires.nil?) && (expires.nil? || expires.to_i == other.expires.to_i)) &&
|
|
109
|
+
online? == other.online? &&
|
|
110
|
+
associated_user == other.associated_user &&
|
|
111
|
+
shopify_session_id == other.shopify_session_id
|
|
112
|
+
)
|
|
113
|
+
else
|
|
114
|
+
false
|
|
115
|
+
end
|
|
116
|
+
end
|
|
117
|
+
end
|
|
118
|
+
end
|
|
119
|
+
end
|
|
@@ -0,0 +1,30 @@
|
|
|
1
|
+
# typed: strict
|
|
2
|
+
# frozen_string_literal: true
|
|
3
|
+
|
|
4
|
+
module ShopifyAPI
|
|
5
|
+
module Auth
|
|
6
|
+
module SessionStorage
|
|
7
|
+
extend T::Sig
|
|
8
|
+
extend T::Helpers
|
|
9
|
+
interface!
|
|
10
|
+
|
|
11
|
+
sig do
|
|
12
|
+
abstract.params(session: Session)
|
|
13
|
+
.returns(T::Boolean)
|
|
14
|
+
end
|
|
15
|
+
def store_session(session); end
|
|
16
|
+
|
|
17
|
+
sig do
|
|
18
|
+
abstract.params(id: String)
|
|
19
|
+
.returns(T.nilable(Session))
|
|
20
|
+
end
|
|
21
|
+
def load_session(id); end
|
|
22
|
+
|
|
23
|
+
sig do
|
|
24
|
+
abstract.params(id: String)
|
|
25
|
+
.returns(T::Boolean)
|
|
26
|
+
end
|
|
27
|
+
def delete_session(id); end
|
|
28
|
+
end
|
|
29
|
+
end
|
|
30
|
+
end
|
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
# typed: strict
|
|
2
|
+
# frozen_string_literal: true
|
|
3
|
+
|
|
4
|
+
module ShopifyAPI
|
|
5
|
+
module Auth
|
|
6
|
+
extend T::Sig
|
|
7
|
+
|
|
8
|
+
class << self
|
|
9
|
+
extend T::Sig
|
|
10
|
+
|
|
11
|
+
sig { params(host: T.nilable(String)).returns(String) }
|
|
12
|
+
def embedded_app_url(host)
|
|
13
|
+
unless Context.setup?
|
|
14
|
+
raise Errors::ContextNotSetupError, "ShopifyAPI::Context not setup, please call ShopifyAPI::Context.setup"
|
|
15
|
+
end
|
|
16
|
+
|
|
17
|
+
unless host
|
|
18
|
+
raise Errors::MissingRequiredArgumentError, "host argument is required"
|
|
19
|
+
end
|
|
20
|
+
|
|
21
|
+
decoded_host = Base64.decode64(host)
|
|
22
|
+
"https://#{decoded_host}/apps/#{Context.api_key}"
|
|
23
|
+
end
|
|
24
|
+
end
|
|
25
|
+
end
|
|
26
|
+
end
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
# typed: strict
|
|
2
|
+
# frozen_string_literal: true
|
|
3
|
+
|
|
4
|
+
module ShopifyAPI
|
|
5
|
+
module Clients
|
|
6
|
+
module Graphql
|
|
7
|
+
class Admin < Client
|
|
8
|
+
sig { params(session: T.nilable(Auth::Session)).void }
|
|
9
|
+
def initialize(session:)
|
|
10
|
+
super(session: session, base_path: "/admin/api")
|
|
11
|
+
end
|
|
12
|
+
end
|
|
13
|
+
end
|
|
14
|
+
end
|
|
15
|
+
end
|
|
@@ -0,0 +1,40 @@
|
|
|
1
|
+
# typed: strict
|
|
2
|
+
# frozen_string_literal: true
|
|
3
|
+
|
|
4
|
+
module ShopifyAPI
|
|
5
|
+
module Clients
|
|
6
|
+
module Graphql
|
|
7
|
+
class Client
|
|
8
|
+
extend T::Sig
|
|
9
|
+
|
|
10
|
+
sig { params(session: T.nilable(Auth::Session), base_path: String).void }
|
|
11
|
+
def initialize(session:, base_path:)
|
|
12
|
+
@http_client = T.let(HttpClient.new(session: session, base_path: base_path), HttpClient)
|
|
13
|
+
end
|
|
14
|
+
|
|
15
|
+
sig do
|
|
16
|
+
params(
|
|
17
|
+
query: String,
|
|
18
|
+
variables: T.nilable(T::Hash[T.any(Symbol, String), T.untyped]),
|
|
19
|
+
headers: T.nilable(T::Hash[T.any(Symbol, String), T.untyped]),
|
|
20
|
+
tries: Integer,
|
|
21
|
+
).returns(HttpResponse)
|
|
22
|
+
end
|
|
23
|
+
def query(query:, variables: nil, headers: nil, tries: 1)
|
|
24
|
+
body = { query: query, variables: variables }
|
|
25
|
+
@http_client.request(
|
|
26
|
+
HttpRequest.new(
|
|
27
|
+
http_method: :post,
|
|
28
|
+
path: "#{Context.api_version}/graphql.json",
|
|
29
|
+
body: body,
|
|
30
|
+
query: nil,
|
|
31
|
+
extra_headers: headers,
|
|
32
|
+
body_type: "application/json",
|
|
33
|
+
tries: tries,
|
|
34
|
+
),
|
|
35
|
+
)
|
|
36
|
+
end
|
|
37
|
+
end
|
|
38
|
+
end
|
|
39
|
+
end
|
|
40
|
+
end
|
|
@@ -0,0 +1,35 @@
|
|
|
1
|
+
# typed: strict
|
|
2
|
+
# frozen_string_literal: true
|
|
3
|
+
|
|
4
|
+
module ShopifyAPI
|
|
5
|
+
module Clients
|
|
6
|
+
module Graphql
|
|
7
|
+
class Storefront < Client
|
|
8
|
+
sig { params(shop: String, storefront_access_token: String).void }
|
|
9
|
+
def initialize(shop, storefront_access_token)
|
|
10
|
+
session = Auth::Session.new(
|
|
11
|
+
id: shop,
|
|
12
|
+
shop: shop,
|
|
13
|
+
access_token: "",
|
|
14
|
+
is_online: false,
|
|
15
|
+
)
|
|
16
|
+
super(session: session, base_path: "/api")
|
|
17
|
+
@storefront_access_token = storefront_access_token
|
|
18
|
+
end
|
|
19
|
+
|
|
20
|
+
sig do
|
|
21
|
+
params(
|
|
22
|
+
query: String,
|
|
23
|
+
variables: T.nilable(T::Hash[T.any(Symbol, String), T.untyped]),
|
|
24
|
+
headers: T.nilable(T::Hash[T.any(Symbol, String), T.untyped]),
|
|
25
|
+
tries: Integer,
|
|
26
|
+
).returns(HttpResponse)
|
|
27
|
+
end
|
|
28
|
+
def query(query:, variables: nil, headers: {}, tries: 1)
|
|
29
|
+
T.must(headers).merge!({ "X-Shopify-Storefront-Access-Token": @storefront_access_token })
|
|
30
|
+
super(query: query, variables: variables, headers: headers, tries: tries)
|
|
31
|
+
end
|
|
32
|
+
end
|
|
33
|
+
end
|
|
34
|
+
end
|
|
35
|
+
end
|