Skills

Create

Upload a new skill as a .zip or .skill archive.

Skills are uploaded as .zip or .skill files via multipart form data. See Package requirements for the archive format.

POST /api/public/v1/artifacts/skills

Example

curl -X POST https://api.nairi.ai/api/public/v1/artifacts/skills \
  -H "Authorization: Bearer $NAIRI_API_KEY" \
  -F "file=@my-skill.zip"
import { readFile } from "node:fs/promises";

const buffer = await readFile("my-skill.zip");
const form = new FormData();
form.append("file", new Blob([buffer]), "my-skill.zip");

const res = await fetch("https://api.nairi.ai/api/public/v1/artifacts/skills", {
  method: "POST",
  headers: {
    Authorization: `Bearer ${process.env.NAIRI_API_KEY}`,
  },
  body: form,
});
const data = (await res.json()) as { id: string };
require "net/http"
require "json"
require "uri"
require "securerandom"

uri = URI("https://api.nairi.ai/api/public/v1/artifacts/skills")
boundary = "----RubyMultipart#{SecureRandom.hex(8)}"
file_path = "my-skill.zip"
file_body = File.binread(file_path)

body = String.new
body << "--#{boundary}\r\n"
body << %(Content-Disposition: form-data; name="file"; filename="#{File.basename(file_path)}"\r\n)
body << "Content-Type: application/zip\r\n\r\n"
body << file_body
body << "\r\n--#{boundary}--\r\n"

req = Net::HTTP::Post.new(uri)
req["Authorization"] = "Bearer #{ENV['NAIRI_API_KEY']}"
req["Content-Type"] = "multipart/form-data; boundary=#{boundary}"
req.body = body

res = Net::HTTP.start(uri.host, uri.port, use_ssl: true) { |h| h.request(req) }
data = JSON.parse(res.body)
import os
import requests

with open("my-skill.zip", "rb") as f:
    res = requests.post(
        "https://api.nairi.ai/api/public/v1/artifacts/skills",
        headers={"Authorization": f"Bearer {os.environ['NAIRI_API_KEY']}"},
        files={"file": ("my-skill.zip", f, "application/zip")},
    )
data = res.json()
package main

import (
	"bytes"
	"encoding/json"
	"fmt"
	"io"
	"mime/multipart"
	"net/http"
	"os"
)

func main() {
	file, _ := os.Open("my-skill.zip")
	defer file.Close()

	var body bytes.Buffer
	writer := multipart.NewWriter(&body)
	part, _ := writer.CreateFormFile("file", "my-skill.zip")
	io.Copy(part, file)
	writer.Close()

	req, _ := http.NewRequest("POST", "https://api.nairi.ai/api/public/v1/artifacts/skills", &body)
	req.Header.Set("Authorization", "Bearer "+os.Getenv("NAIRI_API_KEY"))
	req.Header.Set("Content-Type", writer.FormDataContentType())
	res, _ := http.DefaultClient.Do(req)
	defer res.Body.Close()
	raw, _ := io.ReadAll(res.Body)
	var data map[string]any
	json.Unmarshal(raw, &data)
	fmt.Println(data)
}

Response: 201 Created

{
  "id": "ask_01KQ9T51QVMZPQKBZ4G56735V9",
  "name": "code-reviewer",
  "description": "Reviews code for best practices",
  "warnings": ["unsupported subdirectory ignored: examples/"],
  "created_at": "2026-04-28T10:28:57.000Z",
  "updated_at": "2026-04-28T10:28:57.000Z"
}

warnings only appears when the skill parser produced non-fatal issues during upload (missing optional fields, schema deviations, unsupported subdirectories, etc.). On a clean upload the field is omitted entirely.

On this page