From e092d83101692672225a40678e4590e8e0f48634 Mon Sep 17 00:00:00 2001 From: Philip Blyth Date: Thu, 12 Mar 2015 21:24:57 -0400 Subject: [PATCH 1/3] For #28441: added support + tests for 'include_template_projects' flag --- shotgun_api3/shotgun.py | 72 +++++++++++++++++------ tests/test_api.py | 4 +- tests/test_flags.py | 127 ++++++++++++++++++++++++++++++++++++++++ 3 files changed, 184 insertions(+), 19 deletions(-) create mode 100644 tests/test_flags.py diff --git a/shotgun_api3/shotgun.py b/shotgun_api3/shotgun.py index fbe6c6c0e..b6eb79d97 100755 --- a/shotgun_api3/shotgun.py +++ b/shotgun_api3/shotgun.py @@ -171,6 +171,13 @@ def ensure_include_archived_projects(self): 'label': 'include_archived_projects parameter' }) + def ensure_include_template_projects(self): + """Wrapper for ensure_support""" + self._ensure_support({ + 'version': (6, 0, 0), + 'label': 'include_template_projects parameter' + }) + def ensure_per_project_customization(self): """Wrapper for ensure_support""" return self._ensure_support({ @@ -485,7 +492,8 @@ def info(self): return self._call_rpc("info", None, include_auth_params=False) def find_one(self, entity_type, filters, fields=None, order=None, - filter_operator=None, retired_only=False, include_archived_projects=True): + filter_operator=None, retired_only=False, + include_archived_projects=True, include_template_projects=False): """Calls the find() method and returns the first result, or None. :param entity_type: Required, entity type (string) to find. @@ -504,16 +512,26 @@ def find_one(self, entity_type, filters, fields=None, order=None, :param limit: Optional, number of entities to return per page. Defaults to 0 which returns all entities that match. - :param page: Optional, page of results to return. By default all - results are returned. Use together with limit. - :param retired_only: Optional, flag to return only entities that have been retried. Defaults to False which returns only entities which have not been retired. + + :param page: Optional, page of results to return. By default all + results are returned. Use together with limit. + + :param include_archived_projects: Optional, flag to include entities + whose projects have been archived. Default: True + + :param include_template_projects: Optional, flag to include entities + belonging to template projects. Default: False + + :returns: Result """ results = self.find(entity_type, filters, fields, order, - filter_operator, 1, retired_only, include_archived_projects=include_archived_projects) + filter_operator, 1, retired_only, + include_archived_projects=include_archived_projects, + include_template_projects=include_template_projects) if results: return results[0] @@ -521,7 +539,7 @@ def find_one(self, entity_type, filters, fields=None, order=None, def find(self, entity_type, filters, fields=None, order=None, filter_operator=None, limit=0, retired_only=False, page=0, - include_archived_projects=True): + include_archived_projects=True, include_template_projects=False): """Find entities matching the given filters. :param entity_type: Required, entity type (string) to find. @@ -548,7 +566,10 @@ def find(self, entity_type, filters, fields=None, order=None, have not been retired. :param include_archived_projects: Optional, flag to include entities - whose projects have been archived + whose projects have been archived. Default: True + + :param include_template_projects: Optional, flag to include entities + belonging to template projects. Default: False :returns: list of the dicts for each entity with the requested fields, and their id and type. @@ -572,13 +593,19 @@ def find(self, entity_type, filters, fields=None, order=None, # So we only need to check the server version if it is False self.server_caps.ensure_include_archived_projects() + if include_template_projects: + # This defaults to False on the server (no argument is sent) + # So we only need to check the server version if it is True + self.server_caps.ensure_include_template_projects() + params = self._construct_read_parameters(entity_type, fields, filters, retired_only, order, - include_archived_projects) + include_archived_projects, + include_template_projects) if limit and limit <= self.config.records_per_page: params["paging"]["entities_per_page"] = limit @@ -622,7 +649,8 @@ def _construct_read_parameters(self, filters, retired_only, order, - include_archived_projects): + include_archived_projects, + include_template_projects): params = {} params["type"] = entity_type params["return_fields"] = fields or ["id"] @@ -636,6 +664,10 @@ def _construct_read_parameters(self, # Defaults to True on the server, so only pass it if it's False params["include_archived_projects"] = False + if include_template_projects is True: + # Defaults to False on the server, so only pass it if it's True + params["include_template_projects"] = True + if order: sort_list = [] for sort in order: @@ -665,7 +697,8 @@ def summarize(self, summary_fields, filter_operator=None, grouping=None, - include_archived_projects=True): + include_archived_projects=True, + include_template_projects=False): """ Return group and summary information for entity_type for summary_fields based on the given filters. @@ -678,19 +711,24 @@ def summarize(self, if isinstance(filters, (list, tuple)): filters = _translate_filters(filters, filter_operator) - if not include_archived_projects: - # This defaults to True on the server (no argument is sent) - # So we only need to check the server version if it is False - self.server_caps.ensure_include_archived_projects() - params = {"type": entity_type, "summaries": summary_fields, "filters": filters} - if include_archived_projects is False: - # Defaults to True on the server, so only pass it if it's False + if not include_archived_projects: + # This defaults to True on the server (no argument is sent) + # So we only need to check the server version if it is False + self.server_caps.ensure_include_archived_projects() + # Only pass it if it's False params["include_archived_projects"] = False + if include_template_projects: + # This defaults to False on the server (no argument is sent) + # So we only need to check the server version if it is True + self.server_caps.ensure_include_template_projects() + # Only pass it if it's True + params["include_template_projects"] = True + if grouping != None: params['grouping'] = grouping diff --git a/tests/test_api.py b/tests/test_api.py index f4a004c59..eccae70dd 100644 --- a/tests/test_api.py +++ b/tests/test_api.py @@ -439,7 +439,7 @@ def test_simple_summary(self): assert(result['groups'][0]['summaries']) assert(result['summaries']) - def test_summary_include_archived_projects(self): + def _test_summary_include_archived_projects(self): if self.sg.server_caps.version > (5, 3, 13): # archive project self.sg.update('Project', self.project['id'], {'archived':True}) @@ -1345,7 +1345,7 @@ def test_zero_is_not_none(self): result = self.sg.find_one( 'Asset', [['id','is',self.asset['id']],[num_field, 'is_not', None]] ,[num_field] ) self.assertFalse(result == None) - def test_include_archived_projects(self): + def _test_include_archived_projects(self): if self.sg.server_caps.version > (5, 3, 13): # Ticket #25082 result = self.sg.find_one('Shot', [['id','is',self.shot['id']]]) diff --git a/tests/test_flags.py b/tests/test_flags.py new file mode 100644 index 000000000..403a05e18 --- /dev/null +++ b/tests/test_flags.py @@ -0,0 +1,127 @@ +"""Test using the Shotgun API flags.""" + +import shotgun_api3 +from shotgun_api3.lib.httplib2 import Http + +import base + +class TestFlags(base.LiveTestBase): + def setUp(self): + super(TestFlags, self).setUp() + # We will need the created_at field for the shot + fields = self.shot.keys()[:] + fields.append('created_at') + self.shot = self.sg.find_one('Shot', [['id', 'is', self.shot['id']]], fields) + # We will need the uuid field for our LocalStorage + fields = self.local_storage.keys()[:] + fields.append('uuid') + self.local_storage = self.sg.find_one('LocalStorage', [['id', 'is', self.local_storage['id']]], fields) + + def test_summary_include_archived_projects(self): + """Test summary with 'include_archived_projects'""" + + if self.sg.server_caps.version > (5, 3, 13): + # Ticket #25082 ability to hide archived projects in summary + + # archive project + self.sg.update('Project', self.project['id'], {'archived':True}) + + summaries = [{'field': 'id', 'type': 'count'}] + grouping = [{'direction': 'asc', 'field': 'id', 'type': 'exact'}] + filters = [['project', 'is', self.project]] + + # setting defaults to False, so we should get result + result = self.sg.summarize('Shot', + filters=filters, + summary_fields=summaries, + grouping=grouping) + self.assertEqual(result['summaries']['id'], 1) + + # should get no result + result = self.sg.summarize('Shot', + filters=filters, + summary_fields=summaries, + grouping=grouping, + include_archived_projects=False) + self.assertEqual(result['summaries']['id'], 0) + + # reset project + self.sg.update('Project', self.project['id'], {'archived':False}) + + def test_summary_include_template_projects(self): + """Test summary with 'include_template_projects'""" + + if self.sg.server_caps.version > (6, 0, 0): + # Ticket #28441 + + # set as template project + self.sg.update('Project', self.project['id'], {'template':True}) + + summaries = [{'field': 'id', 'type': 'count'}] + grouping = [{'direction': 'asc', 'field': 'id', 'type': 'exact'}] + filters = [['project', 'is', self.project]] + + # setting defaults to False, so we should get no result + result = self.sg.summarize('Shot', + filters=filters, + summary_fields=summaries, + grouping=grouping) + self.assertEqual(result['summaries']['id'], 0) + + # should get result + result = self.sg.summarize('Shot', + filters=filters, + summary_fields=summaries, + grouping=grouping, + include_template_projects=False) + self.assertEqual(result['summaries']['id'], 1) + + # reset project + self.sg.update('Project', self.project['id'], {'template':False}) + + def test_include_archived_projects(self): + """Test find with 'include_archived_projects'""" + + if self.sg.server_caps.version > (5, 3, 13): + # Ticket #25082 + + result = self.sg.find_one('Shot', [['id','is',self.shot['id']]]) + self.assertEquals(self.shot['id'], result['id']) + + # archive project + self.sg.update('Project', self.project['id'], {'archived':True}) + + # setting defaults to True, so we should get result + result = self.sg.find_one('Shot', [['id','is',self.shot['id']]]) + self.assertEquals(self.shot['id'], result['id']) + + # should get no result + result = self.sg.find_one('Shot', [['id','is',self.shot['id']]], include_archived_projects=False) + self.assertEquals(None, result) + + # reset project + self.sg.update('Project', self.project['id'], {'archived':False}) + + def test_include_template_projects(self): + """Test find with 'include_template_projects'""" + + if self.sg.server_caps.version > (6, 0, 0): + # Ticket #28441 + + result = self.sg.find_one('Shot', [['id','is',self.shot['id']]]) + self.assertEquals(self.shot['id'], result['id']) + + # set as template project + self.sg.update('Project', self.project['id'], {'is_template':True}) + + # setting defaults to False, so we should not get result + result = self.sg.find_one('Shot', [['id','is',self.shot['id']]]) + self.assertEquals(None, result) + + # should get result + result = self.sg.find_one('Shot', [['id','is',self.shot['id']]], include_template_projects=True) + self.assertEquals(self.shot['id'], result['id']) + + # reset project + self.sg.update('Project', self.project['id'], {'is_template':False}) + From d5914e49b00ef605ce213d3a174004091088499b Mon Sep 17 00:00:00 2001 From: Philip Blyth Date: Mon, 16 Mar 2015 17:57:08 -0400 Subject: [PATCH 2/3] For #28441: refactored to clean up duplication in flag checks --- shotgun_api3/shotgun.py | 97 ++++++++++++++++++++--------------------- tests/test_client.py | 8 ++-- 2 files changed, 51 insertions(+), 54 deletions(-) diff --git a/shotgun_api3/shotgun.py b/shotgun_api3/shotgun.py index b6eb79d97..eed5536d8 100755 --- a/shotgun_api3/shotgun.py +++ b/shotgun_api3/shotgun.py @@ -134,7 +134,7 @@ def __init__(self, host, meta): self.is_dev = False self.version = tuple(self.version[:3]) - self._ensure_json_supported() + self.ensure_json_supported() def _ensure_support(self, feature, raise_hell=True): @@ -157,23 +157,23 @@ def _ensure_support(self, feature, raise_hell=True): return True - def _ensure_json_supported(self): + def ensure_json_supported(self): """Wrapper for ensure_support""" - self._ensure_support({ + return self._ensure_support({ 'version': (2, 4, 0), 'label': 'JSON API' }) def ensure_include_archived_projects(self): """Wrapper for ensure_support""" - self._ensure_support({ + return self._ensure_support({ 'version': (5, 3, 14), 'label': 'include_archived_projects parameter' }) def ensure_include_template_projects(self): """Wrapper for ensure_support""" - self._ensure_support({ + return self._ensure_support({ 'version': (6, 0, 0), 'label': 'include_template_projects parameter' }) @@ -525,7 +525,7 @@ def find_one(self, entity_type, filters, fields=None, order=None, :param include_template_projects: Optional, flag to include entities belonging to template projects. Default: False - :returns: Result + :returns: dict of requested entity's fields, or None if not found. """ results = self.find(entity_type, filters, fields, order, @@ -588,22 +588,13 @@ def find(self, entity_type, filters, fields=None, order=None, raise ShotgunError("Deprecated: Use of filter_operator for find()" " is not valid any more. See the documentation on find()") - if not include_archived_projects: - # This defaults to True on the server (no argument is sent) - # So we only need to check the server version if it is False - self.server_caps.ensure_include_archived_projects() - - if include_template_projects: - # This defaults to False on the server (no argument is sent) - # So we only need to check the server version if it is True - self.server_caps.ensure_include_template_projects() - - params = self._construct_read_parameters(entity_type, fields, filters, retired_only, - order, + order) + + params = self._construct_flag_parameters(params, include_archived_projects, include_template_projects) @@ -642,31 +633,24 @@ def find(self, entity_type, filters, fields=None, order=None, return self._parse_records(records) - def _construct_read_parameters(self, entity_type, fields, filters, retired_only, - order, - include_archived_projects, - include_template_projects): - params = {} - params["type"] = entity_type - params["return_fields"] = fields or ["id"] - params["filters"] = filters - params["return_only"] = (retired_only and 'retired') or "active" - params["return_paging_info"] = True - params["paging"] = { "entities_per_page": self.config.records_per_page, - "current_page": 1 } - - if include_archived_projects is False: - # Defaults to True on the server, so only pass it if it's False - params["include_archived_projects"] = False + order): - if include_template_projects is True: - # Defaults to False on the server, so only pass it if it's True - params["include_template_projects"] = True + params = { + "type": entity_type, + "return_fields": fields or ["id"], + "filters": filters, + "return_only": (retired_only and 'retired') or "active", + "return_paging_info": True, + "paging": { + "entities_per_page": self.config.records_per_page, + "current_page": 1 + } + } if order: sort_list = [] @@ -680,6 +664,29 @@ def _construct_read_parameters(self, 'direction' : sort['direction'] }) params['sorts'] = sort_list + + return params + + + def _construct_flag_parameters(self, + params, + include_archived_projects, + include_template_projects): + + if not include_archived_projects: + # This defaults to True on the server (no argument is sent) + # So we only need to check the server version if it's False + self.server_caps.ensure_include_archived_projects() + # Only pass it if it's False + params["include_archived_projects"] = False + + if include_template_projects: + # This defaults to False on the server (no argument is sent) + # So we only need to check the server version if it's True + self.server_caps.ensure_include_template_projects() + # Only pass it if it's True + params["include_template_projects"] = True + return params @@ -715,19 +722,9 @@ def summarize(self, "summaries": summary_fields, "filters": filters} - if not include_archived_projects: - # This defaults to True on the server (no argument is sent) - # So we only need to check the server version if it is False - self.server_caps.ensure_include_archived_projects() - # Only pass it if it's False - params["include_archived_projects"] = False - - if include_template_projects: - # This defaults to False on the server (no argument is sent) - # So we only need to check the server version if it is True - self.server_caps.ensure_include_template_projects() - # Only pass it if it's True - params["include_template_projects"] = True + params = self._construct_flag_parameters(params, + include_archived_projects, + include_template_projects) if grouping != None: params['grouping'] = grouping diff --git a/tests/test_client.py b/tests/test_client.py index 7460b5970..21a5182f6 100644 --- a/tests/test_client.py +++ b/tests/test_client.py @@ -74,18 +74,18 @@ def test_server_version_json(self): sc = ServerCapabilities("foo", {"version" : (2,4,0)}) sc.version = (2,3,99) - self.assertRaises(api.ShotgunError, sc._ensure_json_supported) + self.assertRaises(api.ShotgunError, sc.ensure_json_supported) self.assertRaises(api.ShotgunError, ServerCapabilities, "foo", {"version" : (2,2,0)}) sc.version = (0,0,0) - self.assertRaises(api.ShotgunError, sc._ensure_json_supported) + self.assertRaises(api.ShotgunError, sc.ensure_json_supported) sc.version = (2,4,0) - sc._ensure_json_supported() + sc.ensure_json_supported() sc.version = (2,5,0) - sc._ensure_json_supported() + sc.ensure_json_supported() def test_session_uuid(self): From 6a74c73cdc3a83cd31ebcd76ab2e489d84d25480 Mon Sep 17 00:00:00 2001 From: Philip Blyth Date: Tue, 24 Mar 2015 15:59:48 -0400 Subject: [PATCH 3/3] For #28441: FIXED version checks, better API flag tests --- shotgun_api3/shotgun.py | 34 +++---- tests/base.py | 24 ++++- tests/test_flags.py | 199 +++++++++++++++++++++++++++++++--------- 3 files changed, 193 insertions(+), 64 deletions(-) diff --git a/shotgun_api3/shotgun.py b/shotgun_api3/shotgun.py index eed5536d8..323c6da66 100755 --- a/shotgun_api3/shotgun.py +++ b/shotgun_api3/shotgun.py @@ -164,19 +164,23 @@ def ensure_json_supported(self): 'label': 'JSON API' }) - def ensure_include_archived_projects(self): + def ensure_include_archived_projects(self, value=True): """Wrapper for ensure_support""" + # This defaults to True on the server + # So we only need to raise a version error if it's False return self._ensure_support({ 'version': (5, 3, 14), 'label': 'include_archived_projects parameter' - }) + }, (value == False)) - def ensure_include_template_projects(self): + def ensure_include_template_projects(self, value=False): """Wrapper for ensure_support""" + # This defaults to False on the server + # So we only need to raise a version error if it's True return self._ensure_support({ 'version': (6, 0, 0), 'label': 'include_template_projects parameter' - }) + }, (value == True)) def ensure_per_project_customization(self): """Wrapper for ensure_support""" @@ -673,19 +677,11 @@ def _construct_flag_parameters(self, include_archived_projects, include_template_projects): - if not include_archived_projects: - # This defaults to True on the server (no argument is sent) - # So we only need to check the server version if it's False - self.server_caps.ensure_include_archived_projects() - # Only pass it if it's False - params["include_archived_projects"] = False + if self.server_caps.ensure_include_archived_projects(include_archived_projects): + params["include_archived_projects"] = include_archived_projects - if include_template_projects: - # This defaults to False on the server (no argument is sent) - # So we only need to check the server version if it's True - self.server_caps.ensure_include_template_projects() - # Only pass it if it's True - params["include_template_projects"] = True + if self.server_caps.ensure_include_template_projects(include_template_projects): + params["include_template_projects"] = include_template_projects return params @@ -1659,6 +1655,7 @@ def _call_rpc(self, method, params, include_auth_params=True, first=False): """ + log_time = datetime.datetime.now() LOG.debug("Starting rpc call to %s with params %s" % ( method, params)) @@ -1673,7 +1670,10 @@ def _call_rpc(self, method, params, include_auth_params=True, first=False): } http_status, resp_headers, body = self._make_call("POST", self.config.api_path, encoded_payload, req_headers) - LOG.debug("Completed rpc call to %s" % (method)) + + log_time = datetime.datetime.now() - log_time + LOG.debug("Completed rpc call to %s in %s" % (method, str(log_time))) + try: self._parse_http_status(http_status) except ProtocolError, e: diff --git a/tests/base.py b/tests/base.py index 201844b4f..39e0ebc75 100644 --- a/tests/base.py +++ b/tests/base.py @@ -3,13 +3,14 @@ import unittest from ConfigParser import ConfigParser - import mock import shotgun_api3 as api from shotgun_api3.shotgun import json from shotgun_api3.shotgun import ServerCapabilities +import logging + CONFIG_PATH = 'tests/config' class TestBase(unittest.TestCase): @@ -35,6 +36,10 @@ def __init__(self, *args, **kws): def setUp(self, auth_mode='ApiUser'): + + self.LOG = logging.getLogger("shotgun_api3") + self.LOG.setLevel(logging.WARN) + self.config = SgTestConfig() self.config.read_config(CONFIG_PATH) self.human_login = self.config.human_login @@ -185,10 +190,14 @@ def _setup_mock_data(self): class LiveTestBase(TestBase): '''Test base for tests relying on connection to server.''' + def setUp(self, auth_mode='ApiUser'): super(LiveTestBase, self).setUp(auth_mode) + self.sg_version = self.sg.info()['version'][:3] + self._setup_db(self.config) + if self.sg.server_caps.version and \ self.sg.server_caps.version >= (3, 3, 0) and \ (self.sg.server_caps.host.startswith('0.0.0.0') or \ @@ -197,17 +206,22 @@ def setUp(self, auth_mode='ApiUser'): else: self.server_address = self.sg.server_caps.host + def _setup_db(self, config): data = {'name':self.config.project_name} self.project = _find_or_create_entity(self.sg, 'Project', data) + self.template_project = _find_or_create_entity(self.sg, 'Project', { + 'name': 'Template Project', + 'is_template': True + }) + data = {'name':self.config.human_name, 'login':self.config.human_login, 'password_proxy':self.config.human_password} if self.sg_version >= (3, 0, 0): data['locked_until'] = None - self.human_user = _find_or_create_entity(self.sg, 'HumanUser', data) data = {'code':self.config.asset_code, @@ -256,6 +270,12 @@ def _setup_db(self, config): keys = ['title','project', 'sg_priority'] self.ticket = _find_or_create_entity(self.sg, 'Ticket', data, keys) + data = {'project': self.template_project, + 'title': self.config.ticket_title, + 'sg_priority': '1'} + keys = ['title', 'project', 'sg_priority'] + self.template_ticket = _find_or_create_entity(self.sg, 'Ticket', data, keys) + keys = ['code'] data = {'code':'api wrapper test storage', 'mac_path':'nowhere', diff --git a/tests/test_flags.py b/tests/test_flags.py index 403a05e18..38f320be1 100644 --- a/tests/test_flags.py +++ b/tests/test_flags.py @@ -1,21 +1,23 @@ """Test using the Shotgun API flags.""" import shotgun_api3 +from shotgun_api3 import * from shotgun_api3.lib.httplib2 import Http import base +import logging + class TestFlags(base.LiveTestBase): + def setUp(self): super(TestFlags, self).setUp() + # We will need the created_at field for the shot fields = self.shot.keys()[:] fields.append('created_at') self.shot = self.sg.find_one('Shot', [['id', 'is', self.shot['id']]], fields) - # We will need the uuid field for our LocalStorage - fields = self.local_storage.keys()[:] - fields.append('uuid') - self.local_storage = self.sg.find_one('LocalStorage', [['id', 'is', self.local_storage['id']]], fields) + def test_summary_include_archived_projects(self): """Test summary with 'include_archived_projects'""" @@ -23,105 +25,212 @@ def test_summary_include_archived_projects(self): if self.sg.server_caps.version > (5, 3, 13): # Ticket #25082 ability to hide archived projects in summary - # archive project - self.sg.update('Project', self.project['id'], {'archived':True}) - summaries = [{'field': 'id', 'type': 'count'}] grouping = [{'direction': 'asc', 'field': 'id', 'type': 'exact'}] filters = [['project', 'is', self.project]] - # setting defaults to False, so we should get result + # archive project + self.sg.update('Project', self.project['id'], {'archived':True}) + + # should get no result result = self.sg.summarize('Shot', filters=filters, summary_fields=summaries, - grouping=grouping) - self.assertEqual(result['summaries']['id'], 1) + grouping=grouping, + include_archived_projects=False) + self.assertEquals(result['summaries']['id'], 0) - # should get no result + # should get result result = self.sg.summarize('Shot', filters=filters, summary_fields=summaries, grouping=grouping, - include_archived_projects=False) - self.assertEqual(result['summaries']['id'], 0) + include_archived_projects=True) + self.assertEquals(result['summaries']['id'], 1) + + # setting defaults to True, should get result + result = self.sg.summarize('Shot', + filters=filters, + summary_fields=summaries, + grouping=grouping) + self.assertEquals(result['summaries']['id'], 1) # reset project self.sg.update('Project', self.project['id'], {'archived':False}) + def test_summary_include_template_projects(self): """Test summary with 'include_template_projects'""" - if self.sg.server_caps.version > (6, 0, 0): - # Ticket #28441 + # Ticket #28441 - # set as template project - self.sg.update('Project', self.project['id'], {'template':True}) + self.LOG.setLevel(logging.DEBUG) - summaries = [{'field': 'id', 'type': 'count'}] - grouping = [{'direction': 'asc', 'field': 'id', 'type': 'exact'}] - filters = [['project', 'is', self.project]] + summaries = [{'field': 'id', 'type': 'count'}] + grouping = [{'direction': 'asc', 'field': 'id', 'type': 'exact'}] + filters = [['project', 'is', self.project]] - # setting defaults to False, so we should get no result + # control + result = self.sg.summarize('Shot', + filters=filters, + summary_fields=summaries, + grouping=grouping) + self.assertEquals(result['summaries']['id'], 1) + + # backwards-compatibility + if self.sg.server_caps.version < (6, 0, 0): + + # flag should not be passed, should get result + self.assertRaises(ShotgunError, self.sg.summarize, 'Shot', + filters=filters, + summary_fields=summaries, + grouping=grouping, + include_template_projects=True) + + # test new features + if self.sg.server_caps.version >= (6, 0, 0): + # set as template project + self.sg.update('Project', self.project['id'], {'is_template':True}) + + # should get result result = self.sg.summarize('Shot', filters=filters, summary_fields=summaries, - grouping=grouping) - self.assertEqual(result['summaries']['id'], 0) + grouping=grouping, + include_template_projects=True) + self.assertEquals(result['summaries']['id'], 1) - # should get result + # should get no result result = self.sg.summarize('Shot', filters=filters, summary_fields=summaries, grouping=grouping, include_template_projects=False) - self.assertEqual(result['summaries']['id'], 1) + self.assertEquals(result['summaries']['id'], 0) + + # setting defaults to False, should get no result + result = self.sg.summarize('Shot', + filters=filters, + summary_fields=summaries, + grouping=grouping) + self.assertEquals(result['summaries']['id'], 0) # reset project - self.sg.update('Project', self.project['id'], {'template':False}) + self.sg.update('Project', self.project['id'], {'is_template':False}) + + self.LOG.setLevel(logging.WARN) + def test_include_archived_projects(self): """Test find with 'include_archived_projects'""" + # Ticket #25082 + + filters = [['id', 'is', self.shot['id']]] + if self.sg.server_caps.version > (5, 3, 13): - # Ticket #25082 - result = self.sg.find_one('Shot', [['id','is',self.shot['id']]]) - self.assertEquals(self.shot['id'], result['id']) + # control + result = self.sg.find_one('Shot', + filters=filters) + self.assertEquals(result['id'], self.shot['id']) # archive project self.sg.update('Project', self.project['id'], {'archived':True}) - # setting defaults to True, so we should get result - result = self.sg.find_one('Shot', [['id','is',self.shot['id']]]) - self.assertEquals(self.shot['id'], result['id']) - # should get no result - result = self.sg.find_one('Shot', [['id','is',self.shot['id']]], include_archived_projects=False) - self.assertEquals(None, result) + result = self.sg.find_one('Shot', + filters=filters, + include_archived_projects=False) + self.assertEquals(result, None) + + # should get result + result = self.sg.find_one('Shot', + filters=filters, + include_archived_projects=True) + self.assertEquals(result['id'], self.shot['id']) + + # setting defaults to True, should get result + result = self.sg.find_one('Shot', + filters=filters) + self.assertEquals(result['id'], self.shot['id']) # reset project self.sg.update('Project', self.project['id'], {'archived':False}) + def test_include_template_projects(self): """Test find with 'include_template_projects'""" - if self.sg.server_caps.version > (6, 0, 0): - # Ticket #28441 + # Ticket #28441 + + self.LOG.setLevel(logging.DEBUG) + + filters = [['id', 'is', self.shot['id']]] + + # control + result = self.sg.find_one('Shot', + filters=filters) + self.assertEquals(result['id'], self.shot['id']) + + # backwards-compatibility + if self.sg.server_caps.version < (6, 0, 0): - result = self.sg.find_one('Shot', [['id','is',self.shot['id']]]) - self.assertEquals(self.shot['id'], result['id']) + self.assertRaises(ShotgunError, self.sg.find_one, 'Shot', + filters=filters, + include_template_projects=True) + + # test new features + if self.sg.server_caps.version >= (6, 0, 0): # set as template project self.sg.update('Project', self.project['id'], {'is_template':True}) - # setting defaults to False, so we should not get result - result = self.sg.find_one('Shot', [['id','is',self.shot['id']]]) - self.assertEquals(None, result) - # should get result - result = self.sg.find_one('Shot', [['id','is',self.shot['id']]], include_template_projects=True) - self.assertEquals(self.shot['id'], result['id']) + result = self.sg.find_one('Shot', + filters=filters, + include_template_projects=True) + self.assertEquals(result['id'], self.shot['id']) + + # should get no result + result = self.sg.find_one('Shot', + filters=filters, + include_template_projects=False) + self.assertEquals(result, None) + + # setting defaults to False, should get no result + result = self.sg.find_one('Shot', + filters=filters) + self.assertEquals(result, None) # reset project self.sg.update('Project', self.project['id'], {'is_template':False}) + self.LOG.setLevel(logging.WARN) + + + def test_find_template_project(self): + """Test find the 'Template Project'""" + + # Ticket #28441 + + if self.sg.server_caps.version >= (6, 0, 0): + + # find by name + result = self.sg.find_one('Project', [['name', 'is', self.template_project['name']]]) + self.assertEquals(result['id'], self.template_project['id']) + + # find by ID + result = self.sg.find_one('Project', [['id', 'is', self.template_project['id']]]) + self.assertEquals(result['id'], self.template_project['id']) + + # find attached entity + result = self.sg.find_one( + 'Ticket', + [ + ['id', 'is', self.template_ticket['id']], + ['project.Project.name', 'is', 'Template Project'], + ['project.Project.layout_project', 'is', None] + ] + ) + self.assertEquals(result['id'], self.template_ticket['id'])