This guide shows you how to publish single images, videos, reels (single media posts), or posts containing multiple images and videos (carousel posts) on Instagram professional accounts using the Instagram Platform.
On March 24, 2025, we introduced the new alt_text
field for image posts on the /<INSTAGRAM_PROFESSIONAL_ACCOUNT_ID>/media
endpoint. Reels and stories are not supported.
This guide assumes you have read the Instagram Platform Overview and implemented the needed components for using this API, such as a Meta login flow and a webhooks server to receive notifications.
We cURL media used in publishing attempts, so the media must be hosted on a publicly accessible server at the time of the attempt.
An Instagram professional account connected to a Page that requires Page Publishing Authorization (PPA) cannot be published to until PPA has been completed.
It's possible that an app user may be able to perform Tasks on a Page that initially does not require PPA but later requires it. In this scenario, the app user would not be able to publish content to their Instagram professional account until completing PPA. Since there's no way for you to determine if an app user's Page requires PPA, we recommend that you advise app users to preemptively complete PPA.
You will need the following:
Instagram API with Instagram Login | Instagram API with Facebook Login | |
---|---|---|
Access Levels |
|
|
Access Tokens |
|
|
Host URL |
|
|
Login Type | Business Login for Instagram | Facebook Login for Business |
Permissions |
|
If the app user was granted a role on the Page connected to your app user's Instagram professional account via the Business Manager, your app will also need:
|
Webhooks |
/<IG_ID>/media
— Create media container and upload the media
upload_type=resumable
— Create a resumbable upload session to upload large videos from an area with frequent network interruptions or other transmission failures. Only for apps that have implemented Facebook Login for Business./<IG_ID>/media_publish
— publish uploaded media using their media containers./<IG_CONTAINER_ID>?fields=status_code
— check media container publishing eligibility and status./<IG_ID>/content_publishing_limit
— check app user's current publishing rate limit usage.
POST https://rupload.facebook.com/ig-api-upload/<IG_MEDIA_CONTAINER_ID>
— Upload the video to Meta servers
GET /<IG_MEDIA_CONTAINER_ID>?fields=status_code
— Check publishing eligibility and status of the video
user_tags=[{username:’ig_user_name’}]
is encoded to user_tags=%5B%7Busername:ig_user_name%7D%5D
where [
is encoded to %5B
and {
is encoded to %7B
. For more conversions, please refer to the HTML URL Encoding standard.For additional limitations, see each endpoint's reference.
Instagram accounts are limited to 50 API-published posts within a 24-hour moving period. Carousels count as a single post. This limit is enforced on the POST /<IG_ID>/media_publish
endpoint when attempting to publish a media container. We recommend that your app also enforce the publishing rate limit, especially if your app allows app users to schedule posts to be published in the future.
To check an Instagram professional account's current rate limit usage, query the GET /<IG_ID>/content_publishing_limit
endpoint.
In order to publish a media object, it must have a container. To create the media container and upload a media file, send a
Visit the Instagram User Media Endpoint Reference for additional optional parameters. | ![]() |
Formatted for readability.
curl -X POST "https://<HOST_URL>/<LATEST_API_VERSION>/<IG_ID>/media" -H "Content-Type: application/json" -H "Authorization: Bearer <ACCESS_TOKEN>" -d '{ "image_url":"https://www.example.com/images/bronz-fonz.jpg" }'
On success, your app receives a JSON response with the Instagram Container ID.
{ "id": "<IG_CONTAINER_ID>" }
To publish up to 10 images, videos, or a combination of the two, in a single post, a carousel post, you must create a carousel container. This carousel containter will contain a list of all media containers. To create the carousel container, send a
| ![]() |
Formatted for readability.
curl -X POST "https://graph.instagram.com/v22.0
/90010177253934/media"
-H "Content-Type: application/json"
-d '{
"caption":"Fruit%20candies"
"media_type":"CAROUSEL"
"children":"<IG_CONTAINER_ID_1>,<IG_CONTAINER_ID_2>,<IG_CONTAINER_ID_3>"
}'
On success, your app receives a JSON response with the Instagram Carousel Container ID.
{ "id": "<IG_CAROUSEL_CONTAINER_ID>" }
If you created a container for a resumable video upload in Step 1, your need to upload the video before it can be published.
Most API calls use the graph.facebook.com
host however, calls to upload videos for Reels use rupload.facebook.com
.
The following file sources are supported for uploaded video files:
To start the upload session, send a POST
request to the /<IG_MEDIA_CONTAINER_ID
endpoint on the rupload.facebook.com
host with the following parameters:
access_token
With the ig-container-id
returned from a resumable upload session call, upload the video.
rupload.facebook.com
.media_type
shares the same flow to upload the video.ig-container-id
is the ID returned from resumable upload session calls. access-token
is the same one used in previous steps.offset
is set to the first byte being upload, generally 0
.file_size
is set to the size of your file in bytes.Your_file_local_path
is set to the file path of your local file, for example, if uploading a file from, the Downloads folder on macOS, the path is @Downloads/example.mov.curl -X POST "https://rupload.facebook.com/ig-api-upload/<API_VERSION>/<IG_MEDIA_CONTAINER_ID>`" \ -H "Authorization: OAuth <ACCESS_TOKEN>" \ -H "offset: 0" \ -H "file_size: Your_file_size_in_bytes" \ --data-binary "@my_video_file.mp4"
curl -X POST "https://rupload.facebook.com/ig-api-upload/<API_VERSION>/<IG_MEDIA_CONTAINER_ID>`" \ -H "Authorization: OAuth <ACCESS_TOKEN>" \ -H "file_url: https://example_hosted_video.com"
// Success Response Message { "success":true, "message":"Upload successful." } // Failure Response Message { "debug_info":{ "retriable":false, "type":"ProcessingFailedError", "message":"{\"success\":false,\"error\":{\"message\":\"unauthorized user request\"}}" } }
To publish the media,
Send a POST
request to the /<IG_ID>/media_publish
endpoint with the following parameters:
creation_id
set to the container ID, either for a single media container or a carousel containerFormatted for readability.
curl -X POST "https://<HOST_URL>/<LATEST_API_VERSION>/<IG_ID>/media_publish" -H "Content-Type: application/json" -H "Authorization: Bearer <ACCESS_TOKEN>" -d '{ "creation_id":"<IG_CONTAINER_ID>" }'
On success, your app receives a JSON response with the Instagram Media ID.
{ "id": "<IG_MEDIA_ID>" }
Reels are short-form videos that appears in the Reels tab of the Instagram app. To publish a reel, create a container for the video and include the media_type=REELS
parameter along with the path to the video using the video_url
parameter.
If you publish a reel and then request its media_type
field, the value returned is VIDEO
. To determine if a published video has been designated as a reel, request its media_product_type
field instead.
You can use the code sample on GitHub (insta_reels_publishing_api_sample) to learn how to publish Reels to Instagram.
To publish a reel, create a container for the media object and include the media_type
parameter set to STORIES
.
If you publish a story and then request its media_type
field, the value will be returned as IMAGE/VIDEO
. To determine if a published image/video is a story, request its media_product_type
field instead.
If you are able to create a container for a video but the POST /<IG_ID>/media_publish
endpoint does not return the published media ID, you can get the container's publishing status by querying the GET /<IG_CONTAINER_ID>?fields=status_code
endpoint. This endpoint will return one of the following:
EXPIRED
— The container was not published within 24 hours and has expired.ERROR
— The container failed to complete the publishing process.FINISHED
— The container and its media object are ready to be published.IN_PROGRESS
— The container is still in the publishing process.PUBLISHED
— The container's media object has been published.We recommend querying a container's status once per minute, for no more than 5 minutes.
See the Error Codes reference.
Now that you have published to an Instagram professional account, learn how to moderate comments on your media.