2024-01-14 19:54:53 +00:00
|
|
|
# Copyright (c) 2023 Lincoln D. Stein and the InvokeAI Development Team
|
|
|
|
|
|
|
|
"""
|
|
|
|
This module fetches model metadata objects from the Civitai model repository.
|
|
|
|
In addition to the `from_url()` and `from_id()` methods inherited from the
|
|
|
|
`ModelMetadataFetchBase` base class.
|
|
|
|
|
|
|
|
Civitai has two separate ID spaces: a model ID and a version ID. The
|
|
|
|
version ID corresponds to a specific model, and is the ID accepted by
|
|
|
|
`from_id()`. The model ID corresponds to a family of related models,
|
|
|
|
such as different training checkpoints or 16 vs 32-bit versions. The
|
|
|
|
`from_civitai_modelid()` method will accept a model ID and return the
|
|
|
|
metadata from the default version within this model set. The default
|
|
|
|
version is the same as what the user sees when they click on a model's
|
|
|
|
thumbnail.
|
|
|
|
|
|
|
|
Usage:
|
|
|
|
|
|
|
|
from invokeai.backend.model_manager.metadata.fetch import CivitaiMetadataFetch
|
|
|
|
|
|
|
|
fetcher = CivitaiMetadataFetch()
|
|
|
|
metadata = fetcher.from_url("https://civitai.com/models/206883/split")
|
|
|
|
print(metadata.trained_words)
|
|
|
|
"""
|
|
|
|
|
2024-03-04 08:17:01 +00:00
|
|
|
import json
|
2024-01-14 19:54:53 +00:00
|
|
|
import re
|
|
|
|
from pathlib import Path
|
2024-03-01 11:12:13 +00:00
|
|
|
from typing import Any, Optional
|
2024-01-14 19:54:53 +00:00
|
|
|
|
|
|
|
import requests
|
2024-03-04 08:17:01 +00:00
|
|
|
from pydantic import TypeAdapter, ValidationError
|
2024-01-14 19:54:53 +00:00
|
|
|
from pydantic.networks import AnyHttpUrl
|
|
|
|
from requests.sessions import Session
|
|
|
|
|
2024-03-01 11:12:13 +00:00
|
|
|
from invokeai.backend.model_manager.config import ModelRepoVariant
|
2024-02-06 02:55:11 +00:00
|
|
|
|
2024-01-14 19:54:53 +00:00
|
|
|
from ..metadata_base import (
|
|
|
|
AnyModelRepoMetadata,
|
|
|
|
CivitaiMetadata,
|
|
|
|
RemoteModelFile,
|
|
|
|
UnknownMetadataException,
|
|
|
|
)
|
|
|
|
from .fetch_base import ModelMetadataFetchBase
|
|
|
|
|
|
|
|
CIVITAI_MODEL_PAGE_RE = r"https?://civitai.com/models/(\d+)"
|
|
|
|
CIVITAI_VERSION_PAGE_RE = r"https?://civitai.com/models/(\d+)\?modelVersionId=(\d+)"
|
|
|
|
CIVITAI_DOWNLOAD_RE = r"https?://civitai.com/api/download/models/(\d+)"
|
|
|
|
|
|
|
|
CIVITAI_VERSION_ENDPOINT = "https://civitai.com/api/v1/model-versions/"
|
|
|
|
CIVITAI_MODEL_ENDPOINT = "https://civitai.com/api/v1/models/"
|
|
|
|
|
|
|
|
|
2024-03-01 11:12:13 +00:00
|
|
|
StringSetAdapter = TypeAdapter(set[str])
|
|
|
|
|
|
|
|
|
2024-01-14 19:54:53 +00:00
|
|
|
class CivitaiMetadataFetch(ModelMetadataFetchBase):
|
|
|
|
"""Fetch model metadata from Civitai."""
|
|
|
|
|
2024-03-04 08:17:01 +00:00
|
|
|
def __init__(self, session: Optional[Session] = None, api_key: Optional[str] = None):
|
2024-01-14 19:54:53 +00:00
|
|
|
"""
|
|
|
|
Initialize the fetcher with an optional requests.sessions.Session object.
|
|
|
|
|
|
|
|
By providing a configurable Session object, we can support unit tests on
|
|
|
|
this module without an internet connection.
|
|
|
|
"""
|
|
|
|
self._requests = session or requests.Session()
|
2024-03-04 08:17:01 +00:00
|
|
|
self._api_key = api_key
|
2024-01-14 19:54:53 +00:00
|
|
|
|
|
|
|
def from_url(self, url: AnyHttpUrl) -> AnyModelRepoMetadata:
|
|
|
|
"""
|
|
|
|
Given a URL to a CivitAI model or version page, return a ModelMetadata object.
|
|
|
|
|
|
|
|
In the event that the URL points to a model page without the particular version
|
|
|
|
indicated, the default model version is returned. Otherwise, the requested version
|
|
|
|
is returned.
|
|
|
|
"""
|
|
|
|
if match := re.match(CIVITAI_VERSION_PAGE_RE, str(url), re.IGNORECASE):
|
|
|
|
model_id = match.group(1)
|
|
|
|
version_id = match.group(2)
|
|
|
|
return self.from_civitai_versionid(int(version_id), int(model_id))
|
|
|
|
elif match := re.match(CIVITAI_MODEL_PAGE_RE, str(url), re.IGNORECASE):
|
|
|
|
model_id = match.group(1)
|
|
|
|
return self.from_civitai_modelid(int(model_id))
|
|
|
|
elif match := re.match(CIVITAI_DOWNLOAD_RE, str(url), re.IGNORECASE):
|
|
|
|
version_id = match.group(1)
|
|
|
|
return self.from_civitai_versionid(int(version_id))
|
|
|
|
raise UnknownMetadataException("The url '{url}' does not match any known Civitai URL patterns")
|
|
|
|
|
2024-02-06 02:55:11 +00:00
|
|
|
def from_id(self, id: str, variant: Optional[ModelRepoVariant] = None) -> AnyModelRepoMetadata:
|
2024-01-14 19:54:53 +00:00
|
|
|
"""
|
|
|
|
Given a Civitai model version ID, return a ModelRepoMetadata object.
|
|
|
|
|
2024-02-06 02:55:11 +00:00
|
|
|
:param id: An ID.
|
|
|
|
:param variant: A model variant from the ModelRepoVariant enum (currently ignored)
|
|
|
|
|
2024-01-14 19:54:53 +00:00
|
|
|
May raise an `UnknownMetadataException`.
|
|
|
|
"""
|
|
|
|
return self.from_civitai_versionid(int(id))
|
|
|
|
|
|
|
|
def from_civitai_modelid(self, model_id: int) -> CivitaiMetadata:
|
|
|
|
"""
|
|
|
|
Return metadata from the default version of the indicated model.
|
|
|
|
|
|
|
|
May raise an `UnknownMetadataException`.
|
|
|
|
"""
|
|
|
|
model_url = CIVITAI_MODEL_ENDPOINT + str(model_id)
|
2024-03-04 08:17:01 +00:00
|
|
|
model_json = self._requests.get(self._get_url_with_api_key(model_url)).json()
|
2024-03-01 11:12:13 +00:00
|
|
|
return self._from_api_response(model_json)
|
2024-01-14 19:54:53 +00:00
|
|
|
|
2024-03-01 11:12:13 +00:00
|
|
|
def _from_api_response(self, api_response: dict[str, Any], version_id: Optional[int] = None) -> CivitaiMetadata:
|
2024-01-14 19:54:53 +00:00
|
|
|
try:
|
2024-03-01 11:12:13 +00:00
|
|
|
version_id = version_id or api_response["modelVersions"][0]["id"]
|
2024-01-14 19:54:53 +00:00
|
|
|
except TypeError as excp:
|
|
|
|
raise UnknownMetadataException from excp
|
|
|
|
|
|
|
|
# loop till we find the section containing the version requested
|
2024-03-01 11:12:13 +00:00
|
|
|
version_sections = [x for x in api_response["modelVersions"] if x["id"] == version_id]
|
2024-01-14 19:54:53 +00:00
|
|
|
if not version_sections:
|
|
|
|
raise UnknownMetadataException(f"Version {version_id} not found in model metadata")
|
|
|
|
|
|
|
|
version_json = version_sections[0]
|
|
|
|
|
|
|
|
# Civitai has one "primary" file plus others such as VAEs. We only fetch the primary.
|
|
|
|
primary = [x for x in version_json["files"] if x.get("primary")]
|
|
|
|
assert len(primary) == 1
|
|
|
|
primary_file = primary[0]
|
|
|
|
|
|
|
|
url = primary_file["downloadUrl"]
|
|
|
|
if "?" not in url: # work around apparent bug in civitai api
|
|
|
|
metadata_string = ""
|
|
|
|
for key, value in primary_file["metadata"].items():
|
|
|
|
if not value:
|
|
|
|
continue
|
|
|
|
metadata_string += f"&{key}={value}"
|
|
|
|
url = url + f"?type={primary_file['type']}{metadata_string}"
|
|
|
|
model_files = [
|
|
|
|
RemoteModelFile(
|
2024-03-04 08:17:01 +00:00
|
|
|
url=self._get_url_with_api_key(url),
|
2024-01-14 19:54:53 +00:00
|
|
|
path=Path(primary_file["name"]),
|
|
|
|
size=int(primary_file["sizeKB"] * 1024),
|
|
|
|
sha256=primary_file["hashes"]["SHA256"],
|
|
|
|
)
|
|
|
|
]
|
2024-03-01 11:12:13 +00:00
|
|
|
|
|
|
|
try:
|
2024-03-05 00:10:18 +00:00
|
|
|
trigger_phrases = StringSetAdapter.validate_python(version_json.get("trainedWords"))
|
2024-03-04 08:17:01 +00:00
|
|
|
except ValidationError:
|
2024-03-05 00:10:18 +00:00
|
|
|
trigger_phrases: set[str] = set()
|
2024-03-01 11:12:13 +00:00
|
|
|
|
2024-03-04 08:17:01 +00:00
|
|
|
return CivitaiMetadata(
|
|
|
|
name=version_json["name"],
|
|
|
|
files=model_files,
|
2024-03-05 00:10:18 +00:00
|
|
|
trigger_phrases=trigger_phrases,
|
2024-03-04 08:17:01 +00:00
|
|
|
api_response=json.dumps(version_json),
|
|
|
|
)
|
2024-01-14 19:54:53 +00:00
|
|
|
|
|
|
|
def from_civitai_versionid(self, version_id: int, model_id: Optional[int] = None) -> CivitaiMetadata:
|
|
|
|
"""
|
|
|
|
Return a CivitaiMetadata object given a model version id.
|
|
|
|
|
|
|
|
May raise an `UnknownMetadataException`.
|
|
|
|
"""
|
|
|
|
if model_id is None:
|
|
|
|
version_url = CIVITAI_VERSION_ENDPOINT + str(version_id)
|
2024-03-04 08:17:01 +00:00
|
|
|
version = self._requests.get(self._get_url_with_api_key(version_url)).json()
|
2024-02-02 17:18:47 +00:00
|
|
|
if error := version.get("error"):
|
|
|
|
raise UnknownMetadataException(error)
|
2024-01-14 19:54:53 +00:00
|
|
|
model_id = version["modelId"]
|
|
|
|
|
|
|
|
model_url = CIVITAI_MODEL_ENDPOINT + str(model_id)
|
2024-03-04 08:17:01 +00:00
|
|
|
model_json = self._requests.get(self._get_url_with_api_key(model_url)).json()
|
2024-03-01 11:12:13 +00:00
|
|
|
return self._from_api_response(model_json, version_id)
|
2024-01-14 19:54:53 +00:00
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def from_json(cls, json: str) -> CivitaiMetadata:
|
|
|
|
"""Given the JSON representation of the metadata, return the corresponding Pydantic object."""
|
|
|
|
metadata = CivitaiMetadata.model_validate_json(json)
|
|
|
|
return metadata
|
2024-03-04 08:17:01 +00:00
|
|
|
|
|
|
|
def _get_url_with_api_key(self, url: str) -> str:
|
|
|
|
if not self._api_key:
|
|
|
|
return url
|
|
|
|
|
|
|
|
if "?" in url:
|
|
|
|
return f"{url}&token={self._api_key}"
|
|
|
|
|
|
|
|
return f"{url}?token={self._api_key}"
|