What attracted me, and a lot of Shopify Partners, to build on Shopify is that it is easy to extend. Whatever we can do through the merchant admin, we can do through the Admin API.
We build apps to help merchants get things done, enhancing core Shopify features. With data from the API, we can give merchants insights and tools to grow their businesses.
Until recently, though, our Partner Dashboard was closed off from our tinkering. Many other developers and I came up with scrappy solutions to build the tools and get the data we need to grow our businesses. We had to work with the manual CSV exports or do hacky scraping of the dashboard.
Shopify has launched a Partner API, bringing us a step closer to doing everything programmatically that we can do through the Partner Dashboard.
We’re super excited for Partner API and where it’s headed!
With this Partner API, we’re now able to replace our scrappy import scripts with some proper GraphQL. It also opens up opportunities for other products to integrate with our apps, themes, and jobs data.
First, I’ll walk you through how to get up and running with the Partner API using Ruby. Then we’ll see what others are saying about how they’ll use the API and what they're building.
How to work with the Partner API in Ruby
The Partner API uses GraphQL, which requires more work from the client than a REST API. In exchange, we get a richer way to query and return just the data that we need. What would be multiple REST queries can be a single GraphQL request.
There aren’t any client libraries for the Partner API like we have for the Admin API. I’ll walk you through how to work with the Partner API in Ruby.
We’ll start by adding the GraphQL gem, a gem that internal Shopify teams use.
# Gemfile
gem 'graphql'
$ bundle install
Then:
$ rails generate graphql:install
Since we’re not running a GraphQL server, we can delete the app/graphql
directory and remove this route from config/routes.rb
.
post "/graphql" to: "graphql#execute"
Shopify offers a GraphiQL Explorer through your Partner Dashboard, so you can also remove the one provided by the gem if you like. Delete this from your routes.rb
:
if Rails.env.development?
mount GraphiQL::Rails::Engine, at: "/graphiql", graphql_path: "/graphql"
end
Now let’s start building!
Our GraphQL::Client
needs an HTTP network adapter to make requests. We’ll create our HTTPClient
class for the Partner API specific parts.
All API requests go to a single URL. The path contains our organization ID: a unique identifier for our partner account, and the API version we want to query.
For example:
“https://partners.shopify.com/#{organization_id}/api/#{api_version}/graphql.json”
Authentication is with a custom HTTP header X-Shopify-Access-Token containing the access token we’ve created in our Partner Dashboard.
Putting this together, we have an HTTP Client like this:
module ShopifyPartnerAPI | |
class HTTPClient < GraphQL::Client::HTTP | |
def initialize | |
super("https://partners.shopify.com/") | |
end | |
def headers(context) | |
{ | |
"X-Shopify-Access-Token": context.fetch(:access_token) | |
} | |
end | |
def execute(document:, operation_name: nil, variables: {}, context: {}) | |
@uri = URI.parse("https://partners.shopify.com/#{context.fetch(:organization_id)}/api/unstable/graphql.json") | |
super(document: document, operation_name: operation_name, variables: variables, context: context) | |
end | |
end | |
end |
Core to GraphQL is that every API has a typed schema. Our client uses this schema to make requests to the API. Since the schema can get large and does not vary between requests, we’ll save it to disk once and cache it in memory at runtime.
Here’s the code:
require "graphql/client" | |
require "http_client" | |
module ShopifyPartnerAPI | |
class << self | |
delegate :parse, :query, to: :client | |
def client | |
initialize_client_cache | |
cached_client = @_client_cache | |
if cached_client != nil | |
cached_client | |
else | |
initialize_client | |
@_client_cache | |
end | |
end | |
def initialize_client | |
initialize_client_cache | |
http = ShopifyPartnerAPI::HTTPClient.new | |
# So the schema is not requested every time the client is initialized we store it on disk. | |
# If the schema changes, run GraphQL::Client.dump_schema(http, "config/partner-api-schema.json") | |
GraphQL::Client.dump_schema(http, "config/partner-api-schema.json") unless File.exists?("config/partner-api-schema.json") | |
schema = GraphQL::Client.load_schema("config/partner-api-schema.json") | |
client = GraphQL::Client.new(schema: schema, execute: http) | |
@_client_cache = client | |
end | |
private | |
def initialize_client_cache | |
@_client_cache ||= nil | |
end | |
end | |
end |
Once you’ve fetched partner-api-schema.json
, you can, of course, remove the GraphQL::Client.dump_schema
check altogether.
Now we’re ready to fetch the data we want from the Partner API. Take a look at the Partner API reference to see what’s available. Data-modifying operations are called mutations in GraphQL. The API is currently query-only; there are no mutations.
We’ll use querying all of our monthly and annual recurring app charges in this example.
You might also like: Getting Started with GraphQL.
Writing our query
We’ll start by going to the GraphiQL explorer. Here we can browse the schema in the Documentation Explorer and craft our query. Here’s how to fire it up:
1. From your Shopify Partner Dashboard, click Settings
2. At the bottom of the Settings page, click Manage Partner API clients
3. Push View GraphiQL Explorer next to the token you wish to use.
In the QueryRoot
, we find transactions
. Inside transactions
, we can query for a TransactionType
of AppSubscriptionSale
.
For our example, we want to know when a transaction was created, how much has been paid out to us, for which app, and for which store. With GraphQL, we can selectively pull just those fields.
query { | |
transactions(types: [APP_SUBSCRIPTION_SALE]) { | |
edges { | |
node { | |
id | |
createdAt | |
... on AppSubscriptionSale { | |
netAmount { | |
amount | |
} | |
app { | |
name | |
} | |
shop { | |
myshopifyDomain | |
} | |
} | |
} | |
} | |
} | |
} |
Run this query in your GraphiQL explorer to see the results.
GraphQL is typed. We’ve queried for the AppSubscriptionSale
transaction type and requested data from its fields. Other transaction types share some fields and add their own fields too.
We cannot simply leave off the type since not all fields are available on all types. Run this query, and you’ll get an error.
query { | |
transactions(types: [APP_SUBSCRIPTION_SALE]) { | |
edges { | |
node { | |
id | |
createdAt | |
netAmount { | |
amount | |
} | |
app { | |
name | |
} | |
shop { | |
myshopifyDomain | |
} | |
} | |
} | |
} | |
} |
We use inline fragments to query the fields we want on the types we want.
Let’s add the ServiceSale
type to our query. It doesn’t have an app field. We’ll omit that from its inline fragment.
query { | |
transactions(types: [APP_SUBSCRIPTION_SALE]) { | |
edges { | |
node { | |
id | |
createdAt | |
... on AppSubscriptionSale { | |
netAmount { | |
amount | |
} | |
app { | |
name | |
} | |
shop { | |
myshopifyDomain | |
} | |
} | |
... on ServiceSale { | |
netAmount { | |
amount | |
} | |
shop { | |
myshopifyDomain | |
} | |
} | |
} | |
} | |
} | |
} |
GraphQL paginates using a cursor. The query results contain a cursor string and a hasNextPage
boolean. If there’s another page, we pass the cursor string on the next query, and the next page is returned. We can request up to 100 results at a time.
We’ll add these pagination fields to our query.
query ($cursor: String) { | |
transactions(types: [APP_SUBSCRIPTION_SALE], after: $cursor, first: 100) { | |
edges { | |
cursor | |
node { | |
id | |
createdAt | |
... on AppSubscriptionSale { | |
netAmount { | |
amount | |
} | |
app { | |
name | |
} | |
shop { | |
myshopifyDomain | |
} | |
} | |
... on AppOneTimeSale { | |
netAmount { | |
amount | |
} | |
app { | |
name | |
} | |
shop { | |
myshopifyDomain | |
} | |
} | |
} | |
} | |
pageInfo { | |
hasNextPage | |
} | |
} | |
} |
Now the query is ready, let’s complete our Ruby code.
Putting it all together
Our query is parsed and validated against the Partner API schema at runtime. The GraphQL gem requires the parsed query to be a constant so that we don’t inefficiently parse and validate the query every time it’s executed.
TRANSACTIONS_QUERY = ShopifyPartnerAPI.client.parse <<-'GRAPHQL' | |
query($cursor: String) { | |
transactions(types: [APP_SUBSCRIPTION_SALE], after: $cursor, first: 100) { | |
edges { | |
cursor | |
node { | |
id, | |
createdAt, | |
... on AppSubscriptionSale { | |
netAmount { | |
amount | |
}, | |
app { | |
name | |
}, | |
shop { | |
myshopifyDomain | |
} | |
}, | |
... on ServiceSale { | |
netAmount { | |
amount | |
}, | |
shop { | |
myshopifyDomain | |
} | |
} | |
} | |
}, | |
pageInfo { | |
hasNextPage | |
} | |
} | |
} | |
GRAPHQL |
The API is rate limited to four requests per second per API client. We’ll write a simple throttler to make at most one call every 0.3 seconds.
def throttle(start_time) | |
stop_time = Time.zone.now | |
processing_duration = stop_time - start_time | |
wait_time = (0.3 - processing_duration).round(1) | |
Rails.logger.info("THROTTLING: #{wait_time}") | |
sleep wait_time if wait_time > 0.0 | |
Time.zone.now | |
end |
Here’s the complete code for a Rails class to query the Partner API, paging through results.
class Transaction < ActiveRecord::Base | |
include ShopifyPartnerAPI | |
THROTTLE_MIN_TIME_PER_CALL = 0.3 | |
TRANSACTIONS_QUERY = ShopifyPartnerAPI.client.parse <<-'GRAPHQL' | |
query($cursor: String) { | |
transactions(types: [APP_SUBSCRIPTION_SALE], after: $cursor, first: 100) { | |
edges { | |
cursor | |
node { | |
id, | |
createdAt, | |
... on AppSubscriptionSale { | |
netAmount { | |
amount | |
}, | |
app { | |
name | |
}, | |
shop { | |
myshopifyDomain | |
} | |
}, | |
... on ServiceSale { | |
netAmount { | |
amount | |
}, | |
shop { | |
myshopifyDomain | |
} | |
} | |
} | |
}, | |
pageInfo { | |
hasNextPage | |
} | |
} | |
} | |
GRAPHQL | |
class << self | |
def import_transactions(partner_api_access_token, partner_api_organization_id) | |
cursor = "" | |
has_next_page = true | |
throttle_start_time = Time.zone.now | |
while has_next_page == true | |
throttle_start_time = throttle(throttle_start_time) | |
transactions = [] | |
results = ShopifyPartnerAPI.client.query( | |
TRANSACTIONS_QUERY, | |
variables: {cursor: cursor}, | |
context: {access_token: partner_api_access_token, organization_id: partner_api_organization_id} | |
) | |
raise StandardError.new(results.errors.messages.map { |k, v| "#{k}=#{v}" }.join("&")) if results.errors.any? | |
return if results.data.nil? | |
transactions = results.data.transactions.edges | |
has_next_page = results.data.transactions.page_info.has_next_page | |
cursor = results.data.transactions.edges.last.cursor | |
transactions.each do |transaction| | |
node = transaction.node | |
Transaction.new(node) | |
end | |
end | |
rescue => e | |
Rails.logger.info(e.message) | |
Rails.logger.info(e.backtrace.join("\n")) | |
Rails.logger.info(transactions.to_json) if transactions.present? | |
raise e | |
end | |
private | |
def throttle(start_time) | |
stop_time = Time.zone.now | |
processing_duration = stop_time - start_time | |
wait_time = (THROTTLE_MIN_TIME_PER_CALL - processing_duration).round(1) | |
Rails.logger.info("THROTTLING: #{wait_time}") | |
sleep wait_time if wait_time > 0.0 | |
Time.zone.now | |
end | |
end | |
end |
And there you have it—how to work with the Partner API in Ruby. Now, when it comes to how this will impact partners, I’ve been hearing conversations in the partner communities on social media discussing the opportunities and the unknowns.
What’s the big deal with the Partner API?
A few partners on Twitter and Facebook have asked, What’s the big deal with the Partner API?” There are two opportunities with the Partner API:
- Internal tools
- Buy, don’t build
Partners have all kinds of interesting internal scripts and tools using CSVs and scraped data. They break when the CSV format changes or the Partner Dashboard is changed. A documented and supported Partner API makes these robust and much easier to build.
Then, it opens the possibility of buying tools instead of building our own. It can be fun to hack around on APIs and build admin scripts and dashboards. Sometimes though, I just want to pay for a useful tool and get back to investing time in my products.
We’ll see a crop of Shopify Partner tools start to grow because we can now give other products secure access to our data with scopes and revocable tokens.
There’s been a bit of chatter about what these internal and partner tools could be.
SaaS metrics
To grow our partner businesses, we need to be on top of our SaaS metrics to know where to focus our efforts. Getting accurate and timely numbers for monthly recurring revenue (MRR), churn, lifetime customer value (LTV), and average revenue per user (ARPU) has not been easy.
SaaS metrics products can now plug directly into the Partner API. Baremetrics is working on support. A free tool I contribute to, Partner Metrics, now supports the Partner API.
Forecasting lets us model how our businesses will go in the future, helping us plan. Partners are already asking for tools like Summit to add support.
You might also like: 8 Growth Metrics Every App Developer Should Track.
Marketing attribution
It hasn’t been easy to calculate how effective a marketing campaign is. Attributing specific merchants to a campaign, how much revenue they generate for us, and how much the campaign cost has been murky. Partners are looking for more visibility of return on marketing spend.
...Can’t wait to see what tools get made. My dream is some toolset that helps devs run effective paid user acquisition for their new app.
You might also like: How to Market an App: 11 Expert Tips.
Cash for partners
Shopify Capital gives merchants quick and easy access to cash. With the transparency the Partner API brings, will we see similar non-dilutive financing for Shopify App Developers?
Prediction: With the launch of Shopify’s Partner API, app developers will soon have easier access to cash.
And just five days later, Shopify Partner Eyal Toledano, Creator at Batch Commerce, responded that his team started building a “no nonsense CRM” called Partner CRM that connects into their app dashboard and will help them manage their merchant/client lifecycle.
Partner programs
App developers often partner up with their referral programs, offering a percentage of a merchant’s monthly spend for a referral. Until recently, that’s been hard to calculate accurately. With the Partner API, we can work out exactly how much a merchant has paid us and how much to pay for the referral.
Support and CRM
Partners use customer relationship management (CRM) software and helpdesk tools like HubSpot, Zendesk, and Gorgias to support merchants. When a ticket comes in, we can more effectively help merchants if we have a view of the apps they have installed, their history with our products, and billing information. CRM and helpdesk tools can query the Partner API directly to pull this detailed view of a merchant.
Building together with the Partner API
As you can tell, I’m excited about building on our partner platform, just like we’ve been able to develop on the merchant platform. We can now get deeper insights into our businesses to grow faster. Partners can build tools for partners.
"We can now get deeper insights into our businesses to grow faster."
The Shopify Partner community is remarkable in how we work together, and the Partner API gives us a way to strengthen our relationships even more.
Read more
- How to Upload Files with the Shopify GraphQL API and React
- How the Routes and Page_Image Liquid Objects Can Help You Build More Robust Themes
- How to Work with Shopify’s query Argument in GraphQL
- Shopify API Release: April 2022
- Free eBook] Become a Full-stack Freelancer With Grow Vol. 2
- How We’re Improving Discoverability On The Shopify App Store
- Build for the 20 Percent: How Cleverific Evolved to Meet Merchant Needs
- Shopify Fulfillment Orders API: A Better Fulfillment Experience