vse_toolbox/resources/trackers/kitsu.py

383 lines
12 KiB
Python
Raw Normal View History

2023-03-14 13:38:04 +01:00
import bpy
import os
import re
import urllib3
import traceback
import time
2023-04-20 00:12:39 +02:00
import uuid
2023-04-21 21:44:05 +02:00
from pprint import pprint
2023-03-14 13:38:04 +01:00
from bpy.props import PointerProperty, StringProperty
from pathlib import Path
from vse_toolbox.file_utils import install_module, norm_str
from vse_toolbox.resources.trackers.tracker import Tracker
2023-04-14 18:55:00 +02:00
try:
gazu = install_module('gazu')
except Exception as e:
print('Could not install gazu')
print(e)
2023-04-20 00:12:39 +02:00
LOGIN = None
2023-03-14 13:38:04 +01:00
class Kitsu(Tracker):
2023-04-20 00:12:39 +02:00
2023-03-14 13:38:04 +01:00
url: StringProperty()
login: StringProperty()
password: StringProperty(subtype='PASSWORD')
2023-04-20 00:12:39 +02:00
def connect(self, url=None, login=None, password=None):
'''Connect to kitsu api using provided url, login and password'''
global LOGIN
urllib3.disable_warnings()
if url is None:
url = self.url
if login is None:
login = self.login
if password is None:
password = self.password
if not url:
print(f'Kitsu Url: {self.url} is empty')
return
if login == LOGIN:
return
if not url.endswith('/api'):
url += '/api'
print(f'Info: Setting Host for kitsu {url}')
gazu.client.set_host(url)
if not gazu.client.host_is_up():
print('Error: Kitsu Host is down')
try:
print(f'Info: Log in to kitsu as {login}')
res = gazu.log_in(login, password)
LOGIN = login
print(f'Info: Sucessfully login to Kitsu as {res["user"]["full_name"]}')
return res['user']
except Exception as e:
print(f'Error: {traceback.format_exc()}')
def get_id(self, data):
if isinstance(data, str):
if self.is_id(data):
return data
#return None
elif isinstance(data, dict):
return data['id']
elif data: # Should be a Class
return data.id
def is_id(self, id):
if not isinstance(id, str):
return False
try:
uuid.UUID(id)
return True
except ValueError:
return False
def get_project(self, project=None):
if project:
project_id = self.get_id(project)
if project_id:
return project_id
return gazu.project.get_project_by_name(project)
if os.environ.get('TRACKER_PROJECT_ID'):
return os.environ['TRACKER_PROJECT_ID']
elif os.environ.get('TRACKER_PROJECT_NAME'):
return os.environ['TRACKER_PROJECT_NAME']
2023-03-14 13:38:04 +01:00
def get_projects(self):
return gazu.project.all_open_projects()
2023-04-20 00:12:39 +02:00
def get_episode(self, episode, project=None):
project = self.get_project(project)
if episode:
episode_id = self.get_id(episode)
if episode_id:
return episode_id
return gazu.shot.get_episode_by_name(project, episode)
if os.environ.get('TRACKER_EPISODE_ID'):
return os.environ['TRACKER_EPISODE_ID']
elif os.environ.get('TRACKER_EPISODE_NAME'):
return os.environ['TRACKER_EPISODE_NAME']
2023-03-14 13:38:04 +01:00
def get_episodes(self, project):
return gazu.shot.all_episodes_for_project(project)
2023-04-20 00:12:39 +02:00
def get_task_type(self, task_type=None):
2023-04-20 18:16:34 +02:00
task_type_id = self.get_id(task_type)
if task_type_id:
return task_type_id
2023-04-20 00:12:39 +02:00
return gazu.task.get_task_type_by_name(task_type)
def get_shot_task_types(self, project=None):
project = self.get_project(project)
task_types = gazu.task.all_task_types_for_project(project)
return [t for t in task_types if t['for_entity'].lower() == 'shot']
def get_shots_metadata(self, project=None):
project = self.get_project(project)
metadatas = []
2023-03-14 13:38:04 +01:00
2023-04-20 00:12:39 +02:00
for metadata in gazu.project.all_metadata_descriptors(project):
if metadata['entity_type'] == 'Shot' and metadata['name']:
metadatas.append(metadata)
return metadatas
def get_task_status(self, status=None):
status_id = self.get_id(status)
if status_id:
return status_id
return gazu.client.fetch_first('task-status', {"short_name": status.lower()})
def get_task_statuses(self, project=None):
project = self.get_project(project)
return gazu.task.all_task_statuses_for_project(project)
def get_asset_types(self, project=None):
project = self.get_project(project)
2023-04-04 12:21:45 +02:00
return gazu.asset.all_asset_types_for_project(project)
2023-03-17 20:03:38 +01:00
2023-04-20 00:12:39 +02:00
def get_sequence(self, sequence, project=None):
print(f'get_sequence({sequence=}, {project=})')
project = self.get_project(project)
sequence_id = self.get_id(sequence)
if sequence_id:
return sequence_id
return gazu.shot.get_sequence_by_name(project, sequence)
def get_shot(self, shot, sequence, project=None):
print(f'get_shot({shot=}, {sequence=}, {project=})')
project = self.get_project(project)
sequence = self.get_sequence(sequence, project)
if not sequence:
return
shot_id = self.get_id(shot)
if shot_id:
return shot_id
return gazu.shot.get_shot_by_name(sequence, shot)
2023-04-21 21:44:05 +02:00
def get_asset(self, asset, asset_type=None, project=None):
#print('get_asset', "name", name, 'asset_type', asset_type)
asset_id = self.get_id(asset)
if asset_id:
return asset_id
project = self.get_project(project)
asset_type = self.get_id(asset_type)
asset = gazu.asset.get_asset_by_name(project, asset, asset_type)
return asset
2023-04-20 00:12:39 +02:00
def get_assets(self, project=None):
project = self.get_project(project)
2023-03-17 20:03:38 +01:00
assets = gazu.asset.all_assets_for_project(project)
2023-04-04 12:21:45 +02:00
entity_types = self.get_asset_types(project)
entity_types_ids = {e['id']: e['name'] for e in entity_types}
for asset_data in assets:
asset_data['asset_type'] = entity_types_ids[asset_data.pop('entity_type_id')]
2023-03-17 20:03:38 +01:00
return assets
2023-03-14 13:38:04 +01:00
2023-04-20 00:12:39 +02:00
def get_last_comment(self, task):
task = self.get_id(task)
return gazu.task.get_last_comment_for_task(task)
2023-04-14 18:55:00 +02:00
2023-04-20 00:12:39 +02:00
def get_task(self, task=None, entity=None):
entity = self.get_id(entity)
2023-04-14 18:55:00 +02:00
2023-04-20 00:12:39 +02:00
task_type = self.get_task_type(task)
task = gazu.task.get_task_by_name(entity, task_type)
2023-04-20 18:16:34 +02:00
if not task:
return
2023-04-20 00:12:39 +02:00
#task = gazu.task.get_task(task['id'])
2023-03-20 19:05:22 +01:00
2023-04-20 00:12:39 +02:00
task['last_comment'] = self.get_last_comment(task)
2023-03-20 19:05:22 +01:00
2023-04-20 00:12:39 +02:00
return task
2023-03-14 13:38:04 +01:00
2023-04-20 18:16:34 +02:00
def new_preview(self, task, comment, preview, set_main_preview=False):
#print('new_preview', task, comment, preview, set_main_preview)
task = self.get_id(task)
comment = self.get_id(comment)
preview_data = gazu.task.add_preview(
task=task,
comment=comment,
preview_file_path=preview )
if set_main_preview:
gazu.task.set_main_preview(preview_data)
return preview_data
2023-04-20 00:12:39 +02:00
def new_comment(self, task, status=None, comment='', preview=None, set_main_preview=False):
#task = self.get_task(task)
2023-03-14 13:38:04 +01:00
2023-04-20 00:12:39 +02:00
#print('Add Comment', status)
if status is None:
#print(task)
status = {'id' : task['task_status_id']}
else:
status = self.get_task_status(status)
2023-03-14 13:38:04 +01:00
2023-04-20 00:12:39 +02:00
comment = gazu.task.add_comment(
task=task,
task_status=status,
comment=comment )
if preview:
2023-04-20 18:16:34 +02:00
#logging.info(f'Adding Preview to Kitsu {preview}')
preview = self.new_preview(
2023-04-20 00:12:39 +02:00
task=task,
comment=comment,
2023-04-20 18:16:34 +02:00
preview=str(preview),
set_main_preview=set_main_preview )
2023-03-14 13:38:04 +01:00
2023-04-20 00:12:39 +02:00
return comment
2023-04-20 18:16:34 +02:00
def remove_comment(self, comment):
return gazu.task.remove_comment(comment)
2023-04-20 00:12:39 +02:00
def download_preview(self, preview_id, filepath):
if isinstance(filepath, str):
filepath = Path(filepath)
2023-03-14 13:38:04 +01:00
2023-04-20 00:12:39 +02:00
if filepath.exists():
return
filepath.parent.mkdir(parents=True, exist_ok=True)
gazu.files.download_preview_file_thumbnail(preview_id, filepath.as_posix())
2023-03-14 13:38:04 +01:00
2023-04-20 18:16:34 +02:00
def new_task(self, entity, task_type, status=None, project=None):
task_type = self.get_task_type(task_type)
if status:
status = self.get_task_status(status)
return gazu.task.new_task(entity, task_type=task_type, task_status=status)
2023-04-20 00:12:39 +02:00
def new_sequence(self, sequence, episode=None, project=None):
project = self.get_project(project)
2023-03-14 13:38:04 +01:00
2023-04-20 00:12:39 +02:00
params = dict(name=sequence, project=project)
episode = self.get_episode(episode)
if episode:
params['episode'] = episode
2023-03-14 13:38:04 +01:00
2023-04-20 00:12:39 +02:00
return gazu.shot.new_sequence(**params)
2023-03-14 13:38:04 +01:00
2023-04-20 00:12:39 +02:00
def new_shot(self, shot, sequence, nb_frames=None, frame_in=None,
2023-04-20 18:16:34 +02:00
frame_out=None, description='', custom_data=None, with_tasks=True, project=None):
2023-03-14 13:38:04 +01:00
2023-04-20 00:12:39 +02:00
project = self.get_project(project)
sequence = self.get_sequence(sequence)
2023-03-14 13:38:04 +01:00
2023-04-20 00:12:39 +02:00
custom_data = custom_data or {}
2023-03-14 13:38:04 +01:00
2023-04-20 00:12:39 +02:00
if frame_in is not None:
custom_data["frame_in"] = frame_in
if frame_out is not None:
custom_data["frame_out"] = frame_out
2023-03-14 13:38:04 +01:00
2023-04-20 00:12:39 +02:00
params = dict(name=shot, data=custom_data,
sequence_id=self.get_id(sequence), description=description)
2023-03-14 13:38:04 +01:00
2023-04-20 00:12:39 +02:00
if nb_frames is not None:
params["nb_frames"] = nb_frames
2023-03-14 13:38:04 +01:00
2023-04-20 00:12:39 +02:00
shot = self.get_shot(shot=shot, sequence=sequence)
2023-04-20 18:16:34 +02:00
if not shot:
2023-04-20 00:12:39 +02:00
path = f"data/projects/{self.get_id(project)}/shots"
2023-04-20 18:16:34 +02:00
shot = gazu.client.post(path, params)
if with_tasks:
for task_type in self.get_shot_task_types(project=project):
self.new_task(shot, task_type)
return shot
def update_data(self, entity, data, name=None, description=None, frames=None, clear=False):
if isinstance(entity, dict):
entity_id = entity['id']
else:
entity_id = self.get_id(entity)
entity = gazu.client.fetch_one('entities', entity_id)
if data.get('custom_data'):
data['data'] = data.pop('custom_data')
if name:
entity['name'] = name
if description:
entity['description'] = description
if frames:
entity['nb_frames'] = frames
if clear or not entity['data']:
entity['data'] = data
else:
entity['data'].update(data)
2023-04-21 21:44:05 +02:00
#pprint(entity)
2023-04-20 18:16:34 +02:00
entity_data = gazu.client.put(f"data/entities/{entity_id}", entity)
2023-04-21 21:44:05 +02:00
2023-04-20 18:16:34 +02:00
return entity_data['data']
2023-03-14 13:38:04 +01:00
2023-04-21 21:44:05 +02:00
def get_casting(self, shot, project=None):
project = self.get_project(project)
project_id = self.get_id(project)
return gazu.casting.get_shot_casting({'id': self.get_id(shot), 'project_id': project_id})
def update_casting(self, shot, casting, clear=True, project=None):
project = self.get_project(project)
shot_id = self.get_id(shot)
norm_casting = []
if clear is False:
norm_casting += self.get_casting(shot, project)
for asset in casting:
if isinstance(asset, dict) and 'asset_id' in asset: # It's an asset instance
asset_id = asset['asset_id']
nb_occurences = asset['nb_occurences']
else: # It's an asset
asset = self.get_asset(asset)
nb_occurences = 1
cast = next((c for c in norm_casting if c['asset_id'] == asset_id), None)
if cast:
cast['nb_occurences'] += 1
else:
norm_casting.append({'asset_id': asset_id, 'nb_occurences': nb_occurences})
return gazu.casting.update_shot_casting(project, shot_id, norm_casting)
2023-04-20 00:12:39 +02:00
def draw_prefs(self, layout):
layout.prop(self, 'url', text='Url')
layout.prop(self, 'login', text='Login')
layout.prop(self, 'password', text='Password')