· Nacho Coll · Tutorials  · 9 perc olvasás

IPFS Upload API — Teljes fejlesztői oktatóanyag

Tanuld meg, hogyan tölts fel fájlokat az IPFS-re REST API-n keresztül. Teljes kódminták JavaScriptben, Node.js-ben és curlban. Tölts fel JSON-t, képeket, használj aláírt tokeneket kliensoldali feltöltésekhez.

Tanuld meg, hogyan tölts fel fájlokat az IPFS-re REST API-n keresztül. Teljes kódminták JavaScriptben, Node.js-ben és curlban. Tölts fel JSON-t, képeket, használj aláírt tokeneket kliensoldali feltöltésekhez.

IPFS Upload API — Teljes fejlesztői oktatóanyag

Az IPFS-re való fájlfeltöltésnek olyan egyszerűnek kell lennie, mint egy POST kérés elküldésének. Ebben az oktatóanyagban pontosan ezt fogod tenni — JSON dokumentumokat, képeket és PDF-eket tölthetsz fel az IPFS-re mindössze fetch() és egy API kulcs segítségével. A végén lesz egy teljes Node.js szkripted, amely feltölti a tartalmat, listázza a fájlokat, lekéri a metaadatokat, és aláírt tokeneket generál a biztonságos kliensoldali feltöltésekhez. Új vagy az IPFS-ben? Kezd a Mi az IPFS Pinning? kérdéssel, hogy megértsd az alapokat, mielőtt belevágnál a kódba.

IPFS Ninja API keys management page

Mit fogsz építeni

  • Tölts fel egy JSON objektumot az IPFS-re, és kapj vissza egy tartalomazonosítót (CID).
  • Tölts fel egy bináris képet a lemezről.
  • Csatolj leírásokat és egyéni metaadatokat a feltöltésekhez.
  • Kérdezd le a feltöltési előzményeidet dátumtartomány szerint.
  • Kérj le részleteket egy adott fájlhoz.
  • Hozz létre időkorlátos aláírt tokeneket, hogy egy böngésző közvetlenül feltölthessen anélkül, hogy felfedné az API kulcsodat.
  • Kezeld a hibákat, és implementálj újrapróbálkozási logikát.

Minden egyetlen alap URL ellen fut: https://api.ipfs.ninja

1. Beállítás — Regisztráció és API kulcs beszerzése

  1. Regisztrálj egy ingyenes fiókért (nincs szükség hitelkártyára).
  2. Menj az API Keys-hez a dashboard oldalsávjában.
  3. Kattints a Create key-re, adj neki nevet, és azonnal másold ki a kulcsot — nem fog újra megjelenni.
API Keys page — create and manage your keys

A kulcsod úgy néz ki, mint bws_a1b2c3d4e5f6a7b8c9d0e1f2a3b4c5d6. Minden alábbi példa az X-Api-Key fejlécben küldi el.

2. JSON feltöltése az IPFS-re

A legegyszerűbb feltöltés: adj át egy egyszerű JavaScript objektumot content-ként, és az API szerializálja, az IPFS-re pinneli, és visszaadja a CID-t.

// upload-json.mjs
const API = "https://api.ipfs.ninja";
const API_KEY = "bws_your_key_here";

const payload = {
  content: {
    name: "Galactic Badge #42",
    description: "Proof of attendance — Galactic Meetup 2026",
    attributes: [
      { trait_type: "Event", value: "Galactic Meetup" },
      { trait_type: "Year", value: 2026 }
    ]
  }
};

const res = await fetch(`${API}/upload/new`, {
  method: "POST",
  headers: {
    "Content-Type": "application/json",
    "X-Api-Key": API_KEY
  },
  body: JSON.stringify(payload)
});

const data = await res.json();
console.log("CID :", data.cid);
console.log("Size:", data.sizeMB, "MB");
console.log("IPFS:", data.uris.ipfs);
console.log("URL :", data.uris.url);

Futtasd a node upload-json.mjs-szel. Egy sikeres válasz így néz ki:

{
  "cid": "bafkreigx7gq...",
  "sizeMB": 0.0004,
  "uris": {
    "ipfs": "ipfs://bafkreigx7gq...",
    "url": "https://ipfs.ninja/ipfs/bafkreigx7gq..."
  }
}

Az url mező egy nyilvános HTTP átjáróra mutat, így a tartalom azonnal elérhető bármely böngészőben.

3. Kép feltöltése

A bináris fájlok (képek, PDF-ek, hang) base64 kódolt karakterláncokként kerülnek elküldésre a content mezőben.

// upload-image.mjs
import { readFileSync } from "node:fs";

const API = "https://api.ipfs.ninja";
const API_KEY = "bws_your_key_here";

const imageBuffer = readFileSync("./photo.png");
const base64 = imageBuffer.toString("base64");

const res = await fetch(`${API}/upload/new`, {
  method: "POST",
  headers: {
    "Content-Type": "application/json",
    "X-Api-Key": API_KEY
  },
  body: JSON.stringify({ content: base64 })
});

const data = await res.json();
console.log("Image CID:", data.cid);
console.log("Gateway  :", data.uris.url);

Az API automatikusan érzékeli a MIME típust — a PNG, JPEG, WebP, GIF és PDF mind támogatottak. Nincs szükség extra fejlécre vagy content-type felülírásra.

A curl esetén ugyanaz a művelet így néz ki:

BASE64=$(base64 -w 0 photo.png)

curl -X POST https://api.ipfs.ninja/upload/new \
  -H "Content-Type: application/json" \
  -H "X-Api-Key: bws_your_key_here" \
  -d "{\"content\": \"$BASE64\"}"

4. Feltöltés metaadatokkal

Minden feltöltés két opcionális mezőt fogad el: description (szabad szöveges címke) és metadata (tetszőleges kulcs-érték párok). Mindkettő a CID mellett kerül tárolásra, és visszaadásra kerül, amikor később listázod vagy lekéred a fájlt.

// upload-with-metadata.mjs
const API = "https://api.ipfs.ninja";
const API_KEY = "bws_your_key_here";

const res = await fetch(`${API}/upload/new`, {
  method: "POST",
  headers: {
    "Content-Type": "application/json",
    "X-Api-Key": API_KEY
  },
  body: JSON.stringify({
    content: { title: "Meeting Notes", body: "Q1 roadmap recap..." },
    description: "Q1 2026 planning meeting notes",
    metadata: {
      project: "acme-app",
      author: "dana",
      version: "1"
    }
  })
});

const data = await res.json();
console.log("CID:", data.cid);

A metaadatok megkönnyítik a fájlok szűrését és rendszerezését a te oldaladon anélkül, hogy magát az IPFS tartalmat kellene elemezni.

5. Egy meglévő CID pinnelése

Ha már van tartalmad az IPFS-en, és biztosítani szeretnéd, hogy elérhető maradjon, pinneld CID szerint:

// pin-cid.mjs
const API = "https://api.ipfs.ninja";
const API_KEY = "bws_your_key_here";

const res = await fetch(`${API}/pin`, {
  method: "POST",
  headers: {
    "Content-Type": "application/json",
    "X-Api-Key": API_KEY
  },
  body: JSON.stringify({
    cid: "bafkreigx7gq...",
    description: "Pinned from external source"
  })
});

const data = await res.json();
console.log("Pinned:", data.cid);

6. Listázd a fájljaidat

Kérd le az összes fájlt, amelyet egy időablakon belül feltöltöttél. A from és to lekérdezési paraméterek Unix időbélyegek millimásodpercekben.

// list-files.mjs
const API = "https://api.ipfs.ninja";
const API_KEY = "bws_your_key_here";

const now = Date.now();
const oneWeekAgo = now - 7 * 24 * 60 * 60 * 1000;

const url = `${API}/upload/list?from=${oneWeekAgo}&to=${now}`;

const res = await fetch(url, {
  headers: { "X-Api-Key": API_KEY }
});

const files = await res.json();

for (const file of files) {
  console.log(`${file.cid}  ${file.description ?? "(no description)"}`);
}

curl-lel:

FROM=$(($(date +%s) * 1000 - 604800000))
TO=$(($(date +%s) * 1000))

curl -s "https://api.ipfs.ninja/upload/list?from=$FROM&to=$TO" \
  -H "X-Api-Key: bws_your_key_here" | jq .

7. Fájl részleteinek lekérése

Kérd le a teljes rekordot egy adott CID-hez, beleértve a metaadatokat, méretet és időbélyegeket:

// get-file.mjs
const API = "https://api.ipfs.ninja";
const API_KEY = "bws_your_key_here";
const CID = "bafkreigx7gq...";

const res = await fetch(`${API}/file/${CID}`, {
  headers: { "X-Api-Key": API_KEY }
});

const details = await res.json();
console.log(JSON.stringify(details, null, 2));

8. Kliensoldali feltöltések aláírt tokenekkel

Egy API kulcs beágyazása egy böngésző csomagba biztonsági kockázatot jelent. Helyette generálj egy rövid életű aláírt tokent a szervereden, és add át a kliensnek.

Szerver (Express)

// server.mjs
import express from "express";

const app = express();
const API = "https://api.ipfs.ninja";
const API_KEY = process.env.IPFS_API_KEY;

app.post("/api/upload-token", async (req, res) => {
  const response = await fetch(`${API}/upload/signed-url`, {
    method: "POST",
    headers: {
      "Content-Type": "application/json",
      "X-Api-Key": API_KEY
    },
    body: JSON.stringify({
      name: "browser-upload",
      expiresIn: 600          // token valid for 10 minutes
    })
  });

  const { token, tokenId, expiresAt } = await response.json();
  res.json({ token, tokenId, expiresAt });
});

app.listen(3000, () => console.log("Server running on :3000"));

Böngésző kliens

<!-- upload.html -->
<input type="file" id="filePicker" />
<button id="uploadBtn">Upload to IPFS</button>
<pre id="result"></pre>

<script>
  const API = "https://api.ipfs.ninja";

  document.getElementById("uploadBtn").addEventListener("click", async () => {
    // 1. Get a signed token from your own backend
    const tokenRes = await fetch("/api/upload-token", { method: "POST" });
    const { token } = await tokenRes.json();

    // 2. Read the selected file as base64
    const file = document.getElementById("filePicker").files[0];
    if (!file) return alert("Pick a file first");

    const base64 = await new Promise((resolve) => {
      const reader = new FileReader();
      reader.onload = () => resolve(reader.result.split(",")[1]);
      reader.readAsDataURL(file);
    });

    // 3. Upload directly to IPFS using the signed token
    const uploadRes = await fetch(`${API}/upload/new`, {
      method: "POST",
      headers: {
        "Content-Type": "application/json",
        "Authorization": `Signed ${token}`
      },
      body: JSON.stringify({
        content: base64,
        description: file.name
      })
    });

    const data = await uploadRes.json();
    document.getElementById("result").textContent = JSON.stringify(data, null, 2);
  });
</script>

A böngésző soha nem látja az API kulcsodat. Az aláírt token automatikusan lejár. A tokenek többszöri használatra valók — többször is használhatók, amíg le nem járnak vagy vissza nem vonják őket. A useCount követve van, de nem érvényesítve határként.

9. Hibakezelés

Az API hagyományos HTTP állapotkódokat használ. Itt vannak azok, amelyeket explicit módon kezelned kell:

StatusJelentésMit kell tenni
400Bad request — hiányzó vagy érvénytelen mezőkEllenőrizd, hogy a content jelen van-e és érvényes
401Unauthorized — rossz vagy hiányzó API kulcsEllenőrizd az X-Api-Key fejlécet
413A Payload túl nagyCsökkentsd a fájlméretet vagy oszd fel a tartalmat
429Sebesség korlátozvaVonulj vissza és próbáld újra exponenciális késleltetéssel
500SzerverhibaPróbáld újra egy rövid késleltetés után

Egy rugalmas feltöltő függvény exponenciális backoff-fal:

// resilient-upload.mjs
const API = "https://api.ipfs.ninja";
const API_KEY = "bws_your_key_here";

async function uploadWithRetry(content, retries = 3) {
  for (let attempt = 1; attempt <= retries; attempt++) {
    const res = await fetch(`${API}/upload/new`, {
      method: "POST",
      headers: {
        "Content-Type": "application/json",
        "X-Api-Key": API_KEY
      },
      body: JSON.stringify({ content })
    });

    if (res.ok) return await res.json();

    const body = await res.text();

    if (res.status === 401) {
      throw new Error(`Authentication failed: ${body}`);
    }

    if (res.status === 413) {
      throw new Error(`Payload too large — reduce file size. ${body}`);
    }

    if (res.status === 429 || res.status >= 500) {
      const delay = Math.pow(2, attempt) * 1000;    // 2s, 4s, 8s
      console.warn(`Attempt ${attempt} failed (${res.status}). Retrying in ${delay}ms...`);
      await new Promise((r) => setTimeout(r, delay));
      continue;
    }

    throw new Error(`Upload failed (${res.status}): ${body}`);
  }

  throw new Error("Upload failed after all retries");
}

// Usage
const data = await uploadWithRetry({ hello: "world" });
console.log("CID:", data.cid);

10. Teljes működő példa

Egy egyetlen szkript, amely gyakorolja az ebben az oktatóanyagban tárgyalt összes végpontot. Mentsd ipfs-demo.mjs néven, és futtasd a node ipfs-demo.mjs-szel.

// ipfs-demo.mjs
import { readFileSync } from "node:fs";

const API = "https://api.ipfs.ninja";
const API_KEY = "bws_your_key_here";

const headers = {
  "Content-Type": "application/json",
  "X-Api-Key": API_KEY
};

async function request(method, path, body) {
  const opts = { method, headers };
  if (body) opts.body = JSON.stringify(body);
  const res = await fetch(`${API}${path}`, opts);
  if (!res.ok) throw new Error(`${method} ${path} → ${res.status}: ${await res.text()}`);
  return res.json();
}

// --- 1. Upload JSON ---
console.log("--- Upload JSON ---");
const jsonResult = await request("POST", "/upload/new", {
  content: { name: "Demo Token", edition: 1 },
  description: "Tutorial demo — JSON upload",
  metadata: { tutorial: "true" }
});
console.log("CID:", jsonResult.cid);
console.log("URL:", jsonResult.uris.url);

// --- 2. Upload an image (if photo.png exists) ---
try {
  const img = readFileSync("./photo.png");
  console.log("\n--- Upload Image ---");
  const imgResult = await request("POST", "/upload/new", {
    content: img.toString("base64"),
    description: "Tutorial demo — image upload"
  });
  console.log("CID:", imgResult.cid);
  console.log("URL:", imgResult.uris.url);
} catch {
  console.log("\n--- Skipping image upload (no photo.png found) ---");
}

// --- 3. Pin the JSON CID ---
console.log("\n--- Pin CID ---");
const pinResult = await request("POST", "/pin", {
  cid: jsonResult.cid,
  description: "Pinned from tutorial"
});
console.log("Pinned:", pinResult.cid);

// --- 4. Get file details ---
console.log("\n--- File Details ---");
const details = await request("GET", `/file/${jsonResult.cid}`);
console.log(JSON.stringify(details, null, 2));

// --- 5. List recent files ---
console.log("\n--- Recent Files ---");
const now = Date.now();
const oneHourAgo = now - 60 * 60 * 1000;
const files = await request("GET", `/upload/list?from=${oneHourAgo}&to=${now}`);
console.log(`Found ${files.length} file(s) in the last hour`);
for (const f of files) {
  console.log(` - ${f.cid}  ${f.description ?? ""}`);
}

// --- 6. Create a signed upload token ---
console.log("\n--- Signed Token ---");
const tokenResult = await request("POST", "/upload/signed-url", {
  name: "demo-token",
  expiresIn: 300
});
console.log("Token ID :", tokenResult.tokenId);
console.log("Expires  :", new Date(tokenResult.expiresAt).toISOString());

// --- 7. Upload using the signed token ---
console.log("\n--- Upload with Signed Token ---");
const signedRes = await fetch(`${API}/upload/new`, {
  method: "POST",
  headers: {
    "Content-Type": "application/json",
    "Authorization": `Signed ${tokenResult.token}`
  },
  body: JSON.stringify({
    content: { note: "Uploaded with a signed token" }
  })
});
const signedData = await signedRes.json();
console.log("CID:", signedData.cid);

console.log("\nDone.");

Cseréld le a bws_your_key_here-t egy igazi kulcsra, és futtasd a szkriptet. Minden lépés kiírja az eredményét, hogy követhesd.

11. Következő lépések

Most már tudod, hogyan tölts fel, pinnelj, listázz és kérj le fájlokat az IPFS Upload API-n keresztül, és hogyan tartsd biztonságban a böngésző feltöltéseket aláírt tokenekkel. Egy szélesebb áttekintéshez, beleértve a dashboard UI-t és a Python példákat, lásd Hogyan tölts fel fájlokat az IPFS-re. Innen hova menj:

  • API referencia — Teljes végpont dokumentáció az ipfs.ninja/docs címen.
  • Egyedi átjárók — Szolgáld ki az IPFS tartalmat a saját domainedről. Lásd az átjáró beállítási útmutatót.
  • Analitika — Kövesd a feltöltési mennyiséget, sávszélességet és pin számokat a dashboardon.
  • HTTP kliens — Nem szükséges SDK. Használhatsz szabványos fetch()-et vagy bármilyen HTTP klienst az API-val való interakcióhoz.

Ha problémákba ütközöl, nyiss egy jegyet a support.ipfs.ninja címen, vagy csatlakozz a közösséghez a Discordon.

Vissza a Blogra