cd /news/artificial-intelligence/bangun-api-pendeteksi-gambar-ai-deng… · home topics artificial-intelligence article
[ARTICLE · art-4695] src=dev.to pub= topic=artificial-intelligence verified=true sentiment=· neutral

Bangun API Pendeteksi Gambar AI dengan C2PA + Klasifikasi

Tutorial on building a FastAPI service with a POST /verify endpoint that combines two independent signals—C2PA cryptographic provenance manifests and an AI image detection classifier—to assess whether an uploaded image was generated by AI or captured by a camera. The service reads C2PA content credentials for cryptographically signed metadata, calls an AI detection API to analyze pixel-level features, and returns a JSON verdict with confidence levels. The tutorial also covers designing an OpenAPI contract, testing with Apidog, and using the c2pa-python library to handle C2PA manifest validation.

read13 min views6 publishedMay 21, 2026

Seseorang mengunggah foto ke produk Anda dan mengklaim foto itu diambil dengan kamera. Backend Anda tidak bisa lagi hanya “percaya pada mata”, karena generator gambar sekarang dapat menghasilkan gambar yang tampak nyata. Pendekatan yang lebih aman adalah menggabungkan dua sinyal independen: manifes asal-usul kriptografis dari C2PA dan skor pengklasifikasi deteksi gambar AI.

Dalam tutorial ini, kita akan membangun layanan FastAPI dengan endpoint POST /verify

. Endpoint ini menerima gambar, membaca Kredensial Konten C2PA jika tersedia, memanggil API deteksi gambar AI sebagai sinyal kedua, lalu mengembalikan putusan JSON. Karena ini proyek API, kita juga akan mendesain kontrak OpenAPI terlebih dahulu dan mengujinya dengan Apidog.

TL;DR #

Anda akan membuat layanan FastAPI yang:

  • menerima unggahan gambar lewat POST /verify

, - membaca dan memvalidasi manifes C2PA dengan c2pa-python

, - memanggil pengklasifikasi deteksi AI yang di-host,

  • menggabungkan dua sinyal menjadi salah satu putusan: likely_authentic

likely_ai

uncertain

  • mengembalikan skor kepercayaan dan detail sinyal mentah,
  • mendesain kontrak OpenAPI dan menjalankan mock/test endpoint dengan Apidog.

Mengapa Menggunakan Dua Sinyal? #

Tidak ada satu properti file yang bisa membuktikan “gambar ini dibuat manusia” atau “gambar ini dibuat AI”. Yang tersedia adalah sinyal.

1. Sinyal asal-usul C2PA

C2PA adalah standar terbuka untuk menyertakan metadata yang ditandatangani secara kriptografis pada file media. Metadata ini disebut manifes, dan nama yang biasa dilihat pengguna adalah Kredensial Konten.

Jika kamera, editor, atau generator gambar mendukung C2PA, alat tersebut dapat menulis riwayat pembuatan atau pengeditan gambar dan menandatanganinya dengan sertifikat.

Kelemahannya:

  • C2PA bersifat opt-in.
  • Screenshot biasanya menghapus metadata.
  • Aplikasi pesan atau platform upload sering menghapus metadata.
  • Tidak adanya manifes bukan berarti gambar palsu atau asli.

2. Sinyal pengklasifikasi AI

Pengklasifikasi deteksi AI melihat piksel gambar dan mengembalikan kemungkinan bahwa gambar tersebut dihasilkan AI.

Kelemahannya:

  • hasilnya probabilistik,
  • bisa false positive,
  • bisa false negative,
  • akurasi dapat turun pada gambar yang dikompresi berat atau berasal dari generator baru.

Jadi strategi yang lebih jujur adalah:

“Inilah yang bisa dibuktikan secara kriptografis, inilah perkiraan model, dan inilah tingkat keyakinan gabungannya.”

Jika Anda ingin memahami mode kegagalan pendekatan satu sinyal, baca juga artikel tentang mengapa deteksi gambar AI gagal.

Arsitektur Layanan #

Layanan ini hanya memiliki satu endpoint dan dua sinyal downstream.

                ┌─────────────────────────────┐
   gambar ──▶   │   FastAPI POST /verify       │
                │                              │
                │   1. Validasi unggahan       │
                │   2. Baca manifes C2PA       │
                │   3. Panggil classifier AI   │
                │   4. Gabungkan putusan       │
                └─────────────────────────────┘
                              │
                              ▼
                   JSON verdict + confidence

Stack yang digunakan:

  • Python 3.10+
  • FastAPI
  • Uvicorn python-multipart

httpx

c2pa-python

Instal dependensi:

pip install fastapi "uvicorn[standard]" python-multipart httpx c2pa-python

Membaca Sinyal C2PA #

Pustaka c2pa-python

adalah binding Python untuk pustaka Rust c2pa-rs

. Kita akan menggunakannya untuk membaca manifes C2PA dari file gambar.

Buat file provenance.py

:

import json
import c2pa

def read_provenance(image_path: str) -> dict:
    """
    Baca dan validasi manifes C2PA dari gambar.
    Mengembalikan dict yang dinormalisasi.
    """
    try:
        with c2pa.Reader(image_path) as reader:
            manifest_store = json.loads(reader.json())

    except c2pa.C2paError as err:
        if str(err).startswith("ManifestNotFound"):
            return {
                "has_manifest": False,
                "validation": "none",
                "detail": "Tidak ada manifes C2PA di gambar ini.",
            }

        return {
            "has_manifest": True,
            "validation": "error",
            "detail": f"Tidak dapat mengurai manifes: {err}",
        }

    active_label = manifest_store.get("active_manifest")
    manifests = manifest_store.get("manifests", {})
    active = manifests.get(active_label, {})

    validation_status = manifest_store.get("validation_status", [])
    validation = "valid" if not validation_status else "invalid"

    claim_generator = active.get("claim_generator", "unknown")
    signature_issuer = active.get("signature_info", {}).get("issuer", "unknown")

    return {
        "has_manifest": True,
        "validation": validation,
        "claim_generator": claim_generator,
        "signature_issuer": signature_issuer,
        "validation_status": validation_status,
        "detail": "Manifes berhasil dibaca.",
    }

Hal penting:

ManifestNotFound

adalah kondisi normal. - Manifes hilang tidak boleh dianggap error. #

validation_status

kosong berarti tanda tangan dan hash valid. - validation_status

berisi data berarti manifes gagal validasi. - claim_generator

dapat membantu mengidentifikasi apakah alat pembuatnya kamera, editor, atau generator AI.

Memanggil Pengklasifikasi Deteksi AI #

Untuk sinyal kedua, gunakan API deteksi AI yang di-host. Contoh ini memakai Sightengine karena endpoint dan responsnya terdokumentasi dengan jelas. Polanya sama untuk vendor lain: ganti URL, parameter, dan field respons yang dibaca.

Endpoint Sightengine:

https://api.sightengine.com/1.0/check.json

Buat file classifier.py

:

import httpx

SIGHTENGINE_URL = "https://api.sightengine.com/1.0/check.json"

async def classify_image(
    image_bytes: bytes,
    filename: str,
    api_user: str,
    api_secret: str,
    timeout_seconds: float = 8.0,
) -> dict:
    """
    Kirim gambar ke detektor AI yang di-host.
    Mengembalikan skor AI-generated yang dinormalisasi.
    """
    data = {
        "models": "genai",
        "api_user": api_user,
        "api_secret": api_secret,
    }

    files = {
        "media": (filename, image_bytes)
    }

    try:
        async with httpx.AsyncClient(timeout=timeout_seconds) as client:
            response = await client.post(
                SIGHTENGINE_URL,
                data=data,
                files=files,
            )
            response.raise_for_status()
            payload = response.json()

    except httpx.TimeoutException:
        return {
            "available": False,
            "reason": "classifier_timeout",
        }

    except httpx.HTTPStatusError as err:
        return {
            "available": False,
            "reason": f"classifier_http_{err.response.status_code}",
        }

    except httpx.HTTPError as err:
        return {
            "available": False,
            "reason": f"classifier_error: {err}",
        }

    if payload.get("status") != "success":
        return {
            "available": False,
            "reason": payload.get("error", {}).get("message", "unknown_error"),
        }

    ai_score = payload.get("type", {}).get("ai_generated")

    if ai_score is None:
        return {
            "available": False,
            "reason": "missing_score_in_response",
        }

    return {
        "available": True,
        "ai_score": float(ai_score),
    }

Prinsip implementasinya:

  • Gunakan timeout eksplisit.
  • Jangan biarkan kegagalan vendor menjatuhkan endpoint.
  • Kembalikan available: false

jika classifier timeout atau error. - Perlakukan skor sebagai probabilitas, bukan fakta.

Untuk perbandingan vendor, lihat daftar API deteksi gambar AI terbaik. Untuk pendekatan manual dan teknis lainnya, lihat panduan cara memeriksa apakah gambar dihasilkan oleh AI.

Mendesain Kontrak POST /verify #

Sebelum menulis route FastAPI, desain kontrak endpoint. Ini membuat frontend bisa mulai integrasi lewat mock server sebelum backend selesai.

Dengan Apidog, Anda bisa:

  • mendesain endpoint secara visual,
  • mengimpor/menulis OpenAPI,
  • membuat mock server,
  • menyimpan skenario test endpoint,
  • menjalankan test terhadap backend asli.

Request

Endpoint menerima multipart/form-data

dengan satu field:

image: file

Response

Contoh response:

{
  "verdict": "likely_ai",
  "confidence": 0.86,
  "signals": {
    "provenance": {
      "has_manifest": true,
      "validation": "valid",
      "claim_generator": "SomeImageTool/2.1",
      "signature_issuer": "Some Issuing CA"
    },
    "classifier": {
      "available": true,
      "ai_score": 0.91
    }
  },
  "explanation": "Manifes C2PA yang valid menyebutkan alat gambar AI, dan pengklasifikasi menilai gambar tersebut kemungkinan dihasilkan AI.",
  "checked_at": "2026-05-21T09:30:00Z"
}

Nilai verdict

hanya boleh salah satu dari:

likely_authentic
likely_ai
uncertain

Gunakan uncertain

ketika sinyal lemah, hilang, atau bertentangan.

Skema OpenAPI

Tambahkan komponen response berikut ke spesifikasi OpenAPI Anda:

components:
  schemas:
    VerifyResponse:
      type: object
      required:
        - verdict
        - confidence
        - signals
        - checked_at
      properties:
        verdict:
          type: string
          enum:
            - likely_authentic
            - likely_ai
            - uncertain
        confidence:
          type: number
          format: float
          minimum: 0
          maximum: 1
        signals:
          type: object
          properties:
            provenance:
              type: object
              properties:
                has_manifest:
                  type: boolean
                validation:
                  type: string
                  enum:
                    - valid
                    - invalid
                    - error
                    - none
                claim_generator:
                  type: string
                signature_issuer:
                  type: string
            classifier:
              type: object
              properties:
                available:
                  type: boolean
                ai_score:
                  type: number
                  format: float
        explanation:
          type: string
        checked_at:
          type: string
          format: date-time

Jika Anda ingin menerapkan workflow spec-first, lihat panduan mode spec-first di Apidog.

Menggabungkan Dua Sinyal #

Buat file verdict.py

.

Logika berikut konservatif:

  • manifes C2PA valid lebih kuat daripada classifier,
  • manifes gagal validasi menghasilkan uncertain

, - classifier digunakan saat manifes tidak tersedia,

  • konflik sinyal menghasilkan uncertain

.


def combine_signals(provenance: dict, classifier: dict) -> dict:
    """
    Gabungkan sinyal asal-usul dan classifier menjadi satu putusan.
    """
    has_manifest = provenance.get("has_manifest", False)
    validation = provenance.get("validation", "none")
    generator = (provenance.get("claim_generator") or "").lower()

    classifier_ok = classifier.get("available", False)
    ai_score = classifier.get("ai_score")

    ai_keywords = (
        "firefly",
        "dall-e",
        "dalle",
        "midjourney",
        "stable",
        "gpt",
        "gemini",
        "imagen",
        "generat",
    )

    generator_looks_ai = any(keyword in generator for keyword in ai_keywords)

    if has_manifest and validation == "valid" and generator_looks_ai:
        return _verdict(
            "likely_ai",
            0.95,
            "Manifes C2PA yang valid menyebutkan alat gambar AI.",
        )

    if has_manifest and validation == "valid" and not generator_looks_ai:
        if classifier_ok and ai_score is not None and ai_score > 0.85:
            return _verdict(
                "uncertain",
                0.55,
                "Manifes terlihat otentik tetapi pengklasifikasi tidak setuju; sinyal bertentangan.",
            )

        return _verdict(
            "likely_authentic",
            0.9,
            "Manifes C2PA yang valid dari alat non-AI tersedia.",
        )

    if has_manifest and validation in ("invalid", "error"):
        return _verdict(
            "uncertain",
            0.6,
            "Gambar membawa manifes C2PA yang gagal divalidasi.",
        )

    if classifier_ok and ai_score is not None:
        if ai_score >= 0.7:
            return _verdict(
                "likely_ai",
                round(ai_score, 2),
                "Tidak ada data asal-usul; pengklasifikasi menilai gambar kemungkinan dihasilkan AI.",
            )

        if ai_score <= 0.3:
            return _verdict(
                "likely_authentic",
                round(1 - ai_score, 2),
                "Tidak ada data asal-usul; pengklasifikasi menilai gambar kemungkinan otentik.",
            )

        return _verdict(
            "uncertain",
            0.5,
            "Tidak ada data asal-usul dan skor pengklasifikasi tidak meyakinkan.",
        )

    return _verdict(
        "uncertain",
        0.0,
        "Tidak ada data asal-usul dan pengklasifikasi tidak tersedia.",
    )

def _verdict(verdict: str, confidence: float, explanation: str) -> dict:
    return {
        "verdict": verdict,
        "confidence": confidence,
        "explanation": explanation,
    }

Ambang batas 0.7

, 0.3

, dan 0.85

bukan angka universal. Sesuaikan berdasarkan toleransi risiko produk Anda.

Membuat Aplikasi FastAPI #

Buat file main.py

:

import os
import tempfile
from datetime import datetime, timezone

from fastapi import FastAPI, UploadFile, File, HTTPException
from fastapi.responses import JSONResponse

from provenance import read_provenance
from classifier import classify_image
from verdict import combine_signals

app = FastAPI(
    title="API Detektor Gambar AI",
    version="1.0.0",
)

ALLOWED_TYPES = {
    "image/jpeg",
    "image/png",
    "image/webp",
}

MAX_BYTES = 12 * 1024 * 1024  # 12 MB

SIGHTENGINE_USER = os.environ.get("SIGHTENGINE_API_USER", "")
SIGHTENGINE_SECRET = os.environ.get("SIGHTENGINE_API_SECRET", "")

@app.post("/verify")
async def verify(image: UploadFile = File(...)):
    if image.content_type not in ALLOWED_TYPES:
        raise HTTPException(
            status_code=415,
            detail=(
                f"Tipe {image.content_type} tidak didukung. "
                "Kirim JPEG, PNG, atau WebP."
            ),
        )

    image_bytes = await image.read()

    if len(image_bytes) == 0:
        raise HTTPException(
            status_code=400,
            detail="File kosong.",
        )

    if len(image_bytes) > MAX_BYTES:
        raise HTTPException(
            status_code=413,
            detail="File melebihi batas 12 MB.",
        )

    suffix = os.path.splitext(image.filename or "")[1] or ".img"

    with tempfile.NamedTemporaryFile(suffix=suffix, delete=False) as tmp:
        tmp.write(image_bytes)
        tmp_path = tmp.name

    try:
        provenance = read_provenance(tmp_path)
    finally:
        os.unlink(tmp_path)

    if SIGHTENGINE_USER and SIGHTENGINE_SECRET:
        classifier = await classify_image(
            image_bytes=image_bytes,
            filename=image.filename or "upload",
            api_user=SIGHTENGINE_USER,
            api_secret=SIGHTENGINE_SECRET,
        )
    else:
        classifier = {
            "available": False,
            "reason": "classifier_not_configured",
        }

    result = combine_signals(provenance, classifier)

    return JSONResponse(
        {
            "verdict": result["verdict"],
            "confidence": result["confidence"],
            "signals": {
                "provenance": {
                    key: provenance.get(key)
                    for key in (
                        "has_manifest",
                        "validation",
                        "claim_generator",
                        "signature_issuer",
                    )
                },
                "classifier": {
                    "available": classifier.get("available", False),
                    "ai_score": classifier.get("ai_score"),
                },
            },
            "explanation": result["explanation"],
            "checked_at": datetime.now(timezone.utc).isoformat(),
        }
    )

Jalankan server lokal:

uvicorn main:app --reload

Endpoint aktif di:

http://127.0.0.1:8000/verify

Contoh request dengan curl

:

curl -X POST http://127.0.0.1:8000/verify \
  -F "image=@sample.jpg"

Jika Anda menggunakan Sightengine, set environment variable terlebih dahulu:

export SIGHTENGINE_API_USER="your_user"
export SIGHTENGINE_API_SECRET="your_secret"

Desain seperti ini cocok untuk layanan API kecil yang fokus pada satu kapabilitas. Untuk konteks lebih luas tentang pola produk berbasis API, baca artikel tentang software yang menjadi headless.

Mocking dan Testing dengan Apidog #

Frontend tidak perlu menunggu backend selesai. Setelah kontrak OpenAPI tersedia, Anda bisa membuat mock server di Apidog.

1. Buat mock server

Di Apidog:

  • Buat project baru.
  • Tambahkan endpoint POST /verify

. - Set body sebagai multipart/form-data

. - Tambahkan field image

bertipe file. - Tambahkan response schema VerifyResponse

. - Buat contoh response untuk beberapa skenario.

Contoh skenario mock yang sebaiknya dibuat:

likely_authentic

dengan manifes kamera valid, - likely_ai

dengan manifes dari alat AI, - uncertain

ketika classifier tidak tersedia, - error 415

untuk tipe file tidak didukung, - error 413

untuk file terlalu besar.

Frontend dapat mengarahkan fetch

ke URL mock Apidog. Saat backend asli siap, ganti base URL saja.

2. Jalankan test endpoint

Setelah backend lokal berjalan:

  • Buat request POST /verify

di Apidog. - Set URL ke http://127.0.0.1:8000/verify

. - Di Body, pilih form-data

. - Tambahkan field image

. - Set tipe field ke File

. - Pilih gambar uji.

  • Kirim request.

Tambahkan assertion:

  • status response adalah 200

, - verdict

ada, - verdict

adalah salah satu darilikely_authentic

,likely_ai

,uncertain

, - confidence

angka antara0

dan1

, - signals.provenance.has_manifest

bertipe boolean, - signals.classifier.available

bertipe boolean.

Buat test suite kecil dengan beberapa file:

  • gambar dengan Kredensial Konten,
  • JPEG biasa tanpa manifes,
  • file terlalu besar,
  • file non-gambar yang diganti ekstensi menjadi .jpg

, - gambar yang memicu timeout/mock classifier unavailable.

Dengan cara ini, perubahan pada fungsi putusan bisa langsung diuji ulang.

Penguatan Implementasi #

Endpoint verifikasi menerima input yang bisa bersifat adversarial. Tambahkan perlindungan berikut sebelum production.

Validasi file sebenarnya

Content-Type bisa dipalsukan. Untuk validasi lebih kuat, decode gambar dengan Pillow:

pip install pillow

Contoh validasi:

from io import BytesIO
from PIL import Image, UnidentifiedImageError

def validate_image_bytes(image_bytes: bytes) -> None:
    try:
        with Image.open(BytesIO(image_bytes)) as img:
            img.verify()
    except UnidentifiedImageError:
        raise ValueError("File bukan gambar valid.")

Lalu panggil sebelum membaca C2PA atau classifier.

Jangan anggap manifes hilang sebagai error

Kasus paling umum adalah gambar tanpa manifes. Ini bukan 500

, bukan bukti palsu, dan bukan bukti asli.

Respons yang benar biasanya tetap 200

dengan sinyal:

{
  "has_manifest": false,
  "validation": "none"
}

Tangani timeout classifier

Classifier adalah dependency jaringan. Gunakan timeout pendek dan perlakukan kegagalan sebagai sinyal tidak tersedia.

{
  "available": false,
  "reason": "classifier_timeout"
}

Jangan biarkan vendor yang lambat menjatuhkan endpoint Anda.

Waspadai manifes palsu atau rusak

Manifes yang ada belum tentu valid. Selalu cek validation_status

.

  • kosong: valid,
  • berisi data: gagal validasi.

Manifes gagal validasi harus menghasilkan uncertain

, bukan likely_authentic

.

Batasi ukuran file

Contoh kode memakai batas 12 MB. Untuk production:

  • batasi ukuran request di reverse proxy,
  • batasi ukuran di aplikasi,
  • gunakan rate limiting,
  • logging minimal,
  • hindari menyimpan gambar pengguna lebih lama dari yang dibutuhkan.

Perhatikan privasi

Anda menerima gambar pengguna dan mungkin mengirimkannya ke vendor pihak ketiga. Pastikan:

  • tidak mencatat byte gambar,
  • file sementara dihapus,
  • kebijakan privasi menjelaskan pemrosesan pihak ketiga,
  • penggunaan vendor sesuai dengan kebutuhan produk Anda.

Apa yang Ditangkap dan Dilewatkan Tiap Sinyal #

Skenario Sinyal asal-usul C2PA Sinyal pengklasifikasi
Gambar AI dari alat yang menulis Kredensial Konten Menangkapnya: manifes menyebutkan generator Biasanya menangkapnya: artefak visual hadir
Gambar AI dengan metadata dihapus Melewatkannya: tidak ada manifes Menangkapnya: bekerja pada piksel
Foto asli dari kamera yang menandatangani Kredensial Konten Memverifikasi: manifes valid dan generator non-AI Bisa false positive pada kompresi/edit berat
Foto asli tanpa metadata Tidak ada sinyal Hanya estimasi probabilistik
Gambar dengan manifes palsu atau dirusak Menangkapnya lewat validation_status
Mungkin menangkap, mungkin tidak
Generator baru yang belum ada di data training Menangkap hanya jika alat menulis manifes Bisa melewatkan karena out-of-distribution
Foto asli dengan retouch AI Jika ada, manifes mencatat riwayat edit Ambigu; skor bisa berada di tengah

Kesimpulannya: C2PA kuat tetapi tidak selalu ada. Classifier selalu bisa dijalankan pada piksel, tetapi hasilnya tidak pasti. Gabungan keduanya lebih berguna daripada salah satu saja.

Kasus Penggunaan #

Pola POST /verify

ini cocok untuk:

Platform konten buatan pengguna

Tandai gambar yang kemungkinan AI atau memiliki manifes gagal validasi.Ruang berita dan fact-checking

Berikan editor sinyal asal-usul dan skor classifier dalam satu respons.Asuransi dan klaim

Flag bukti foto yang terlihat sintetis atau memiliki metadata rusak.Pipeline aset internal

Cegah gambar AI masuk ke library tanpa label.CMS sadar provenance

Tampilkan badge terverifikasi ketika Kredensial Konten valid tersedia.

Kesimpulan #

Deteksi gambar AI yang baik bukan tentang menemukan satu tes sempurna. Yang lebih realistis adalah menggabungkan sinyal independen dan menyatakan ketidakpastian secara eksplisit.

Ringkasnya:

  • C2PA memberi sinyal kriptografis yang kuat, tetapi sering tidak tersedia.
  • Classifier memberi sinyal universal, tetapi probabilistik.
  • FastAPI cukup untuk membangun layanan POST /verify

kecil dan fokus. - Putusan tiga nilai lebih jujur daripada boolean.

  • OpenAPI + Apidog membantu frontend dan backend bekerja paralel lewat mock server dan test suite.

Langkah berikutnya: desain skema /verify

, buat mock server di Apidog, jalankan test endpoint, lalu ganti mock URL ke backend asli saat implementasi siap.

── more in #artificial-intelligence 4 stories · sorted by recency
sponsored brought to you by zahid.host 4,200+ EU-deployed projects
reading about agents? ship yours in a single git push.

Run your AI side-project on zahid.host

EU-based hosting, git-push deploys, automatic HTTPS, no cold starts. Free tier with a custom domain — perfect for shipping the agent you just read about.

$git push zahid main
Live at https://your-agent.zahid.host
Get free account → Pricing
from €0/mo · no card required
LIVE [news/bangun-api-pendeteks…] indexed:0 read:13min 2026-05-21 ·