From 2a47af3f8e0aa2cf136940c3adf946d037fdcfb6 Mon Sep 17 00:00:00 2001 From: Tyler Doyle Date: Tue, 13 Sep 2016 14:00:24 -0700 Subject: [PATCH 01/16] Scrub some strings in the test files (#82) * Scrub some strings in the test file so they don't include internal references * Updated some of the assets to be consistent in casing --- test/assets/CONNECTION.xml | 2 +- test/assets/TABLEAU_10_TDS.tds | 2 +- test/bvt.py | 44 +++++++++++++++++----------------- 3 files changed, 24 insertions(+), 24 deletions(-) diff --git a/test/assets/CONNECTION.xml b/test/assets/CONNECTION.xml index 392d112..6cce11f 100644 --- a/test/assets/CONNECTION.xml +++ b/test/assets/CONNECTION.xml @@ -1 +1 @@ - + diff --git a/test/assets/TABLEAU_10_TDS.tds b/test/assets/TABLEAU_10_TDS.tds index 7a81784..01fe7c3 100644 --- a/test/assets/TABLEAU_10_TDS.tds +++ b/test/assets/TABLEAU_10_TDS.tds @@ -1 +1 @@ - + diff --git a/test/bvt.py b/test/bvt.py index 4ea3d11..60e42c3 100644 --- a/test/bvt.py +++ b/test/bvt.py @@ -42,7 +42,7 @@ def test_can_extract_federated_connections(self): connections = parser.get_connections() self.assertIsInstance(connections, list) self.assertIsInstance(connections[0], Connection) - self.assertEqual(connections[0].dbname, 'testv1') + self.assertEqual(connections[0].dbname, 'TestV1') class ConnectionModelTests(unittest.TestCase): @@ -54,18 +54,18 @@ def test_can_read_attributes_from_connection(self): conn = Connection(self.connection) self.assertEqual(conn.dbname, 'TestV1') self.assertEqual(conn.username, '') - self.assertEqual(conn.server, 'mssql2012.test.tsi.lan') + self.assertEqual(conn.server, 'mssql2012') self.assertEqual(conn.dbclass, 'sqlserver') self.assertEqual(conn.authentication, 'sspi') def test_can_write_attributes_to_connection(self): conn = Connection(self.connection) conn.dbname = 'BubblesInMyDrink' - conn.server = 'mssql2014.test.tsi.lan' + conn.server = 'mssql2014' conn.username = 'bob' self.assertEqual(conn.dbname, 'BubblesInMyDrink') self.assertEqual(conn.username, 'bob') - self.assertEqual(conn.server, 'mssql2014.test.tsi.lan') + self.assertEqual(conn.server, 'mssql2014') def test_bad_dbclass_rasies_attribute_error(self): conn = Connection(self.connection) @@ -117,15 +117,15 @@ def test_can_extract_connection(self): def test_can_save_tds(self): original_tds = Datasource.from_file(self.tds_file.name) - original_tds.connections[0].dbname = 'newdb.test.tsi.lan' + original_tds.connections[0].dbname = 'newdb' original_tds.save() new_tds = Datasource.from_file(self.tds_file.name) - self.assertEqual(new_tds.connections[0].dbname, 'newdb.test.tsi.lan') + self.assertEqual(new_tds.connections[0].dbname, 'newdb') def test_save_has_xml_declaration(self): original_tds = Datasource.from_file(self.tds_file.name) - original_tds.connections[0].dbname = 'newdb.test.tsi.lan' + original_tds.connections[0].dbname = 'newdb' original_tds.save() @@ -158,11 +158,11 @@ def test_can_extract_connection(self): def test_can_save_tds(self): original_tds = Datasource.from_file(self.tds_file.name) - original_tds.connections[0].dbname = 'newdb.test.tsi.lan' + original_tds.connections[0].dbname = 'newdb' original_tds.save() new_tds = Datasource.from_file(self.tds_file.name) - self.assertEqual(new_tds.connections[0].dbname, 'newdb.test.tsi.lan') + self.assertEqual(new_tds.connections[0].dbname, 'newdb') class DatasourceModelV10TDSXTests(unittest.TestCase): @@ -183,22 +183,22 @@ def test_can_open_tdsx(self): def test_can_open_tdsx_and_save_changes(self): original_tdsx = Datasource.from_file(self.tdsx_file.name) - original_tdsx.connections[0].server = 'newdb.test.tsi.lan' + original_tdsx.connections[0].server = 'newdb' original_tdsx.save() new_tdsx = Datasource.from_file(self.tdsx_file.name) self.assertEqual(new_tdsx.connections[ - 0].server, 'newdb.test.tsi.lan') + 0].server, 'newdb') def test_can_open_tdsx_and_save_as_changes(self): new_tdsx_filename = 'newtdsx.tdsx' original_wb = Datasource.from_file(self.tdsx_file.name) - original_wb.connections[0].server = 'newdb.test.tsi.lan' + original_wb.connections[0].server = 'newdb' original_wb.save_as(new_tdsx_filename) new_wb = Datasource.from_file(new_tdsx_filename) self.assertEqual(new_wb.connections[ - 0].server, 'newdb.test.tsi.lan') + 0].server, 'newdb') os.unlink(new_tdsx_filename) @@ -230,12 +230,12 @@ def test_has_filename(self): def test_can_update_datasource_connection_and_save(self): original_wb = Workbook(self.workbook_file.name) - original_wb.datasources[0].connections[0].dbname = 'newdb.test.tsi.lan' + original_wb.datasources[0].connections[0].dbname = 'newdb' original_wb.save() new_wb = Workbook(self.workbook_file.name) self.assertEqual(new_wb.datasources[0].connections[ - 0].dbname, 'newdb.test.tsi.lan') + 0].dbname, 'newdb') class WorkbookModelV10Tests(unittest.TestCase): @@ -260,17 +260,17 @@ def test_can_extract_datasourceV10(self): def test_can_update_datasource_connection_and_saveV10(self): original_wb = Workbook(self.workbook_file.name) - original_wb.datasources[0].connections[0].dbname = 'newdb.test.tsi.lan' + original_wb.datasources[0].connections[0].dbname = 'newdb' original_wb.save() new_wb = Workbook(self.workbook_file.name) self.assertEqual(new_wb.datasources[0].connections[ - 0].dbname, 'newdb.test.tsi.lan') + 0].dbname, 'newdb') def test_save_has_xml_declaration(self): original_wb = Workbook(self.workbook_file.name) - original_wb.datasources[0].connections[0].dbname = 'newdb.test.tsi.lan' + original_wb.datasources[0].connections[0].dbname = 'newdb' original_wb.save() @@ -298,22 +298,22 @@ def test_can_open_twbx(self): def test_can_open_twbx_and_save_changes(self): original_wb = Workbook(self.workbook_file.name) - original_wb.datasources[0].connections[0].server = 'newdb.test.tsi.lan' + original_wb.datasources[0].connections[0].server = 'newdb' original_wb.save() new_wb = Workbook(self.workbook_file.name) self.assertEqual(new_wb.datasources[0].connections[ - 0].server, 'newdb.test.tsi.lan') + 0].server, 'newdb') def test_can_open_twbx_and_save_as_changes(self): new_twbx_filename = 'newtwbx.twbx' original_wb = Workbook(self.workbook_file.name) - original_wb.datasources[0].connections[0].server = 'newdb.test.tsi.lan' + original_wb.datasources[0].connections[0].server = 'newdb' original_wb.save_as(new_twbx_filename) new_wb = Workbook(new_twbx_filename) self.assertEqual(new_wb.datasources[0].connections[ - 0].server, 'newdb.test.tsi.lan') + 0].server, 'newdb') os.unlink(new_twbx_filename) From a4e41f9bc7649eeb9045053c7cfd581c25aaf2b8 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Miguel=20S=C3=A1nchez?= Date: Mon, 19 Sep 2016 17:59:42 +0200 Subject: [PATCH 02/16] Fix encoding bugs in Python2 (non-ASCII characters) (#80) --- tableaudocumentapi/datasource.py | 2 +- tableaudocumentapi/field.py | 7 ++++++- 2 files changed, 7 insertions(+), 2 deletions(-) diff --git a/tableaudocumentapi/datasource.py b/tableaudocumentapi/datasource.py index dd3d5c5..69318ed 100644 --- a/tableaudocumentapi/datasource.py +++ b/tableaudocumentapi/datasource.py @@ -31,7 +31,7 @@ def _get_metadata_xml_for_field(root_xml, field_name): if "'" in field_name: field_name = sax.escape(field_name, {"'": "'"}) - xpath = ".//metadata-record[@class='column'][local-name='{}']".format(field_name) + xpath = u".//metadata-record[@class='column'][local-name='{}']".format(field_name) return root_xml.find(xpath) diff --git a/tableaudocumentapi/field.py b/tableaudocumentapi/field.py index 63cc72c..65ce78d 100644 --- a/tableaudocumentapi/field.py +++ b/tableaudocumentapi/field.py @@ -199,4 +199,9 @@ def _read_description(xmldata): if description is None: return None - return u'{}'.format(ET.tostring(description, encoding='utf-8')) # This is necessary for py3 support + description_string = ET.tostring(description, encoding='utf-8') + # Format expects a unicode string so in Python 2 we have to do the explicit conversion + if isinstance(description_string, bytes): + description_string = description_string.decode('utf-8') + + return description_string From 95f104ba8002a55865fcdf9464188140df0f9ebf Mon Sep 17 00:00:00 2001 From: Russell Hay Date: Mon, 19 Sep 2016 10:14:05 -0700 Subject: [PATCH 03/16] Add a test to verify #80 (#83) --- test/assets/unicode.tds | 93 +++++++++++++++++++++++++++++++++++++++++ test/test_field.py | 10 +++++ 2 files changed, 103 insertions(+) create mode 100644 test/assets/unicode.tds diff --git a/test/assets/unicode.tds b/test/assets/unicode.tds new file mode 100644 index 0000000..6c6e764 --- /dev/null +++ b/test/assets/unicode.tds @@ -0,0 +1,93 @@ + + + + + + + a + 130 + [a] + [xy] + a + 1 + string + Count + 255 + true + + "SQL_WVARCHAR" + "SQL_C_WCHAR" + "true" + + + + Today's Date + 130 + [Today's Date] + [xy] + a + 1 + string + Count + 255 + true + + "SQL_WVARCHAR" + "SQL_C_WCHAR" + "true" + + + + x + 3 + [x] + [xy] + x + 2 + integer + Sum + 10 + true + + "SQL_INTEGER" + "SQL_C_SLONG" + + + + y + 3 + [y] + [xy] + y + 3 + integer + Sum + 10 + true + + "SQL_INTEGER" + "SQL_C_SLONG" + + + + + + + + + + + + año + Something will go here too, in a muted gray + + + + + + + + + + + diff --git a/test/test_field.py b/test/test_field.py index 7cbe885..b222e81 100644 --- a/test/test_field.py +++ b/test/test_field.py @@ -12,6 +12,10 @@ TEST_ASSET_DIR, 'datasource_test.tds' ) +TEST_UNICODE_FILE = os.path.join( + TEST_ASSET_DIR, + 'unicode.tds' +) class FieldsUnitTest(unittest.TestCase): @@ -27,3 +31,9 @@ def find(self, *args, **kwargs): def test_find_metadata_record_returns_none(self): self.assertIsNone(_find_metadata_record(self.MockXmlWithNoFind(), 'foo')) + + +class FieldsHandleUnicode(unittest.TestCase): + def test_description_unicode(self): + ds = Datasource.from_file(TEST_UNICODE_FILE) + self.assertIsNotNone(ds.fields['A'].description) From 0a0232e1ab5353877ee79bf7ae2b28642bb9fa7e Mon Sep 17 00:00:00 2001 From: Russell Hay Date: Fri, 30 Sep 2016 09:53:19 -0700 Subject: [PATCH 04/16] Create CONTRIBUTORS.md (#84) * Create CONTRIBUTORS.md * Adding Charley Peng who contributed doc fixes * Update CONTRIBUTORS.md --- CONTRIBUTORS.md | 14 ++++++++++++++ 1 file changed, 14 insertions(+) create mode 100644 CONTRIBUTORS.md diff --git a/CONTRIBUTORS.md b/CONTRIBUTORS.md new file mode 100644 index 0000000..d79e152 --- /dev/null +++ b/CONTRIBUTORS.md @@ -0,0 +1,14 @@ +This project wouldn't be possible without our amazing contributors. + +The following people have contributed to this project to make it possible, and we thank them for their contributions! + +## Contributors + +* [Charley Peng](https://github.com/chid) +* [Miguel Sánchez](https://github.com/MiguelSR) + +## Core Team + +* [Tyler Doyle](https://github.com/t8y8) +* [Russell Hay](https://github.com/RussTheAerialist) + From aa84176661a2e52c53b4ba711a424c18d0582331 Mon Sep 17 00:00:00 2001 From: Tyler Doyle Date: Fri, 30 Sep 2016 15:21:05 -0500 Subject: [PATCH 05/16] Fix #87 by working around non-federated sqlproxy connections (#89) When Cross Database Joins were introduced in 10 all connections became "federated" by default, even if they weren't actually joined to anything else. In the file format that means they get represented as named-connection/connection elements. Expressed under one top-level connection element with a class of 'federated'. Except for 'sqlproxy' connections (Published Data Sources) -- they stay in the old connection style as a top level connection element. We need to, when in a 10.0 or greater workbook, get all federated connections (named-connection) plus go back and find any sqlproxy connections as well. --- tableaudocumentapi/datasource.py | 4 +++- test/assets/multiple_connections.twb | 35 ++++++++++++++++++++++++++++ test/bvt.py | 23 ++++++++++++++++++ 3 files changed, 61 insertions(+), 1 deletion(-) create mode 100644 test/assets/multiple_connections.twb diff --git a/tableaudocumentapi/datasource.py b/tableaudocumentapi/datasource.py index 69318ed..7f60829 100644 --- a/tableaudocumentapi/datasource.py +++ b/tableaudocumentapi/datasource.py @@ -101,7 +101,9 @@ def __init__(self, datasource_xml, version): self._dsversion = version def _extract_federated_connections(self): - return list(map(Connection, self._dsxml.findall('.//named-connections/named-connection/*'))) + connections = list(map(Connection, self._dsxml.findall('.//named-connections/named-connection/*'))) + connections.extend(map(Connection, self._dsxml.findall("./connection[@class='sqlproxy']"))) + return connections def _extract_legacy_connection(self): return list(map(Connection, self._dsxml.findall('connection'))) diff --git a/test/assets/multiple_connections.twb b/test/assets/multiple_connections.twb new file mode 100644 index 0000000..0bdc774 --- /dev/null +++ b/test/assets/multiple_connections.twb @@ -0,0 +1,35 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/test/bvt.py b/test/bvt.py index 60e42c3..57eead6 100644 --- a/test/bvt.py +++ b/test/bvt.py @@ -27,6 +27,9 @@ EMPTY_WORKBOOK = os.path.join(TEST_DIR, 'assets', 'empty_workbook.twb') +MULTI_CONNECTION_10 = os.path.join( + TEST_DIR, 'assets', 'multiple_connections.twb') + class ConnectionParserTests(unittest.TestCase): @@ -94,6 +97,26 @@ def test_can_create_datasource_from_connections(self): self.assertEqual(ds.connections[1].server, '1') +class ConnectionParserInComplicatedWorkbooks(unittest.TestCase): + + def setUp(self): + with open(MULTI_CONNECTION_10, 'rb') as in_file, open('test.twb', 'wb') as out_file: + out_file.write(in_file.read()) + self.twb_file = out_file + + def tearDown(self): + self.twb_file.close() + os.unlink(self.twb_file.name) + + def test_can_mixed_connections_workbook(self): + wb = Workbook(self.twb_file.name) + self.assertTrue(len(wb.datasources), 2) + self.assertTrue(len(wb.datasources[1].connections), 2) + self.assertEqual(wb.datasources[0].connections[0].dbclass, 'sqlproxy') + self.assertEqual(wb.datasources[1].connections[0].dbclass, 'mysql') + self.assertEqual(wb.datasources[1].connections[1].dbclass, 'sqlserver') + + class DatasourceModelTests(unittest.TestCase): def setUp(self): From 7ced1d1280bdbd6ecf720bb8b5778460d335ea1c Mon Sep 17 00:00:00 2001 From: Tyler Doyle Date: Sun, 2 Oct 2016 15:35:18 -0500 Subject: [PATCH 06/16] Removing unused imports per linter report (#90) --- tableaudocumentapi/datasource.py | 1 - tableaudocumentapi/workbook.py | 3 --- 2 files changed, 4 deletions(-) diff --git a/tableaudocumentapi/datasource.py b/tableaudocumentapi/datasource.py index 7f60829..a106000 100644 --- a/tableaudocumentapi/datasource.py +++ b/tableaudocumentapi/datasource.py @@ -5,7 +5,6 @@ ############################################################################### import collections import itertools -import random import xml.etree.ElementTree as ET import xml.sax.saxutils as sax from uuid import uuid4 diff --git a/tableaudocumentapi/workbook.py b/tableaudocumentapi/workbook.py index 1359356..4b4ce59 100644 --- a/tableaudocumentapi/workbook.py +++ b/tableaudocumentapi/workbook.py @@ -3,11 +3,8 @@ # Workbook - A class for writing Tableau workbook files # ############################################################################### -import os -import zipfile import weakref -import xml.etree.ElementTree as ET from tableaudocumentapi import Datasource, xfile from tableaudocumentapi.xfile import xml_open From 8478800e64f90f86969987addc0a1e6a12fb08ab Mon Sep 17 00:00:00 2001 From: Russell Hay Date: Tue, 4 Oct 2016 11:16:46 -0700 Subject: [PATCH 07/16] #86 Repository location stripping (#88) * Add the ability to clear repository location on the datasource for retargetting * fixing a pep8 issue that was missed * Adding None check for repository-location in case it doesn't exist --- tableaudocumentapi/datasource.py | 5 + tableaudocumentapi/xfile.py | 6 +- test/assets/datasource_test.tds | 187 ++++++++++++++++--------------- test/bvt.py | 18 +++ test/test_datasource.py | 13 +++ 5 files changed, 133 insertions(+), 96 deletions(-) diff --git a/tableaudocumentapi/datasource.py b/tableaudocumentapi/datasource.py index a106000..63157e9 100644 --- a/tableaudocumentapi/datasource.py +++ b/tableaudocumentapi/datasource.py @@ -214,6 +214,11 @@ def version(self): def connections(self): return self._connections + def clear_repository_location(self): + tag = self._datasourceXML.find('./repository-location') + if tag is not None: + self._datasourceXML.remove(tag) + ########### # fields ########### diff --git a/tableaudocumentapi/xfile.py b/tableaudocumentapi/xfile.py index 66e5aac..3067781 100644 --- a/tableaudocumentapi/xfile.py +++ b/tableaudocumentapi/xfile.py @@ -105,10 +105,10 @@ def save_into_archive(xml_tree, filename, new_filename=None): def _save_file(container_file, xml_tree, new_filename=None): - if container_file is None: - container_file = new_filename + if new_filename is None: + new_filename = container_file if zipfile.is_zipfile(container_file): save_into_archive(xml_tree, container_file, new_filename) else: - xml_tree.write(container_file, encoding="utf-8", xml_declaration=True) + xml_tree.write(new_filename, encoding="utf-8", xml_declaration=True) diff --git a/test/assets/datasource_test.tds b/test/assets/datasource_test.tds index bfab77b..5f280eb 100644 --- a/test/assets/datasource_test.tds +++ b/test/assets/datasource_test.tds @@ -1,93 +1,94 @@ - - - - - - - a - 130 - [a] - [xy] - a - 1 - string - Count - 255 - true - - "SQL_WVARCHAR" - "SQL_C_WCHAR" - "true" - - - - Today's Date - 130 - [Today's Date] - [xy] - a - 1 - string - Count - 255 - true - - "SQL_WVARCHAR" - "SQL_C_WCHAR" - "true" - - - - x - 3 - [x] - [xy] - x - 2 - integer - Sum - 10 - true - - "SQL_INTEGER" - "SQL_C_SLONG" - - - - y - 3 - [y] - [xy] - y - 3 - integer - Sum - 10 - true - - "SQL_INTEGER" - "SQL_C_SLONG" - - - - - - - - - - - - A thing - Something will go here too, in a muted gray - - - - - - - - - - - + + + + + + + + a + 130 + [a] + [xy] + a + 1 + string + Count + 255 + true + + "SQL_WVARCHAR" + "SQL_C_WCHAR" + "true" + + + + Today's Date + 130 + [Today's Date] + [xy] + a + 1 + string + Count + 255 + true + + "SQL_WVARCHAR" + "SQL_C_WCHAR" + "true" + + + + x + 3 + [x] + [xy] + x + 2 + integer + Sum + 10 + true + + "SQL_INTEGER" + "SQL_C_SLONG" + + + + y + 3 + [y] + [xy] + y + 3 + integer + Sum + 10 + true + + "SQL_INTEGER" + "SQL_C_SLONG" + + + + + + + + + + + + A thing + Something will go here too, in a muted gray + + + + + + + + + + + diff --git a/test/bvt.py b/test/bvt.py index 57eead6..5f96483 100644 --- a/test/bvt.py +++ b/test/bvt.py @@ -187,6 +187,24 @@ def test_can_save_tds(self): new_tds = Datasource.from_file(self.tds_file.name) self.assertEqual(new_tds.connections[0].dbname, 'newdb') + def test_can_save_as_tds(self): + new_filename = os.path.join( + os.path.dirname(self.tds_file.name), + "new_{}".format(os.path.basename(self.tds_file.name)) + ) + + try: + original_tds = Datasource.from_file(self.tds_file.name) + original_tds.connections[0].dbname = 'newdb' + + original_tds.save_as(new_filename) + + new_tds = Datasource.from_file(new_filename) + self.assertEqual(new_tds.connections[0].dbname, 'newdb') + finally: + if os.path.exists(new_filename): + os.unlink(new_filename) + class DatasourceModelV10TDSXTests(unittest.TestCase): diff --git a/test/test_datasource.py b/test/test_datasource.py index 66b3f79..baf5bc3 100644 --- a/test/test_datasource.py +++ b/test/test_datasource.py @@ -63,6 +63,19 @@ def test_datasource_field_description(self): self.assertIsNotNone(actual) self.assertTrue(u'muted gray' in actual) + def test_datasource_clear_repository_location(self): + filename = os.path.join(TEST_ASSET_DIR, 'clear-repository-test.tds') + + self.assertIsNotNone(self.ds._datasourceXML.find('.//repository-location')) + self.ds.clear_repository_location() + try: + self.ds.save_as(filename) + with open(filename, 'r') as newfile: + self.assertFalse('repository-location' in newfile.read()) + finally: + if os.path.exists(filename): + os.unlink(filename) + class DataSourceFieldsTWB(unittest.TestCase): From f054732a94d37b2f164f8cfe8299a6cdceab385a Mon Sep 17 00:00:00 2001 From: Russell Hay Date: Fri, 7 Oct 2016 10:01:35 -0700 Subject: [PATCH 08/16] bump version and create changelog (#91) --- CHANGELOG.md | 5 +++++ setup.py | 2 +- 2 files changed, 6 insertions(+), 1 deletion(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 7ba40cd..8f78464 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,3 +1,8 @@ +## 0.4 (07 October 2016) + +* Add ability to remove repository location (#86) +* Fixed bug in connection parsing when federated connections are present (#87) + ## 0.3 (31 August 2016) * Added basic connection class retargeting (#65) diff --git a/setup.py b/setup.py index 998914e..a907a26 100644 --- a/setup.py +++ b/setup.py @@ -5,7 +5,7 @@ setup( name='tableaudocumentapi', - version='0.3', + version='0.4', author='Tableau', author_email='github@tableau.com', url='https://github.com/tableau/document-api-python', From 32a0ac341c4de3c4eb71c5e92f315db95b6b1f7f Mon Sep 17 00:00:00 2001 From: Russell Hay Date: Fri, 7 Oct 2016 10:10:39 -0700 Subject: [PATCH 09/16] Add #80 to changelog (#92) * Updating link in readme We renamed Examples to samples, but forgot to update the readme. * Adding #80 to changelog --- CHANGELOG.md | 1 + 1 file changed, 1 insertion(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 8f78464..b53abc0 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -2,6 +2,7 @@ * Add ability to remove repository location (#86) * Fixed bug in connection parsing when federated connections are present (#87) +* Fixed bug in UNICODE support (#80) ## 0.3 (31 August 2016) From c29d36495730b22f582e715c75b515b9a24876c9 Mon Sep 17 00:00:00 2001 From: Russell Hay Date: Fri, 7 Oct 2016 11:16:44 -0700 Subject: [PATCH 10/16] begin development of v0.5 (#94) --- setup.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/setup.py b/setup.py index a907a26..797b6de 100644 --- a/setup.py +++ b/setup.py @@ -5,7 +5,7 @@ setup( name='tableaudocumentapi', - version='0.4', + version='0.5.dev0', author='Tableau', author_email='github@tableau.com', url='https://github.com/tableau/document-api-python', From 94d1f8255899888003ebdd3e9ce11d2e3d202204 Mon Sep 17 00:00:00 2001 From: Tyler Doyle Date: Mon, 17 Oct 2016 12:50:23 -0500 Subject: [PATCH 11/16] Issue #96: Add port support (#97) Issue #96: Add port support We now support reading and setting the port attribute on connections. If you set the port attribute to None it will remove it entirely. Small change to tests so that they don't have side effects between test cases. --- tableaudocumentapi/connection.py | 23 ++++++++++++++++++++++- test/assets/CONNECTION.xml | 2 +- test/bvt.py | 18 ++++++++++++++---- 3 files changed, 37 insertions(+), 6 deletions(-) diff --git a/tableaudocumentapi/connection.py b/tableaudocumentapi/connection.py index 8e9eb58..f71daa8 100644 --- a/tableaudocumentapi/connection.py +++ b/tableaudocumentapi/connection.py @@ -30,18 +30,20 @@ def __init__(self, connxml): self._username = connxml.get('username') self._authentication = connxml.get('authentication') self._class = connxml.get('class') + self._port = connxml.get('port', None) def __repr__(self): return "''".format(self._server, self._dbname, hex(id(self))) @classmethod - def from_attributes(cls, server, dbname, username, dbclass, authentication=''): + def from_attributes(cls, server, dbname, username, dbclass, port=None, authentication=''): root = ET.Element('connection', authentication=authentication) xml = cls(root) xml.server = server xml.dbname = dbname xml.username = username xml.dbclass = dbclass + xml.port = port return xml @@ -133,3 +135,22 @@ def dbclass(self, value): self._class = value self._connectionXML.set('class', value) + + ########### + # port + ########### + @property + def port(self): + return self._port + + @port.setter + def port(self, value): + self._port = value + # If port is None we remove the element and don't write it to XML + if value is None: + try: + del self._connectionXML.attrib['port'] + except KeyError: + pass + else: + self._connectionXML.set('port', value) diff --git a/test/assets/CONNECTION.xml b/test/assets/CONNECTION.xml index 6cce11f..beb606f 100644 --- a/test/assets/CONNECTION.xml +++ b/test/assets/CONNECTION.xml @@ -1 +1 @@ - + diff --git a/test/bvt.py b/test/bvt.py index 5f96483..e09ec55 100644 --- a/test/bvt.py +++ b/test/bvt.py @@ -18,8 +18,7 @@ TABLEAU_10_TWB = os.path.join(TEST_DIR, 'assets', 'TABLEAU_10_TWB.twb') -TABLEAU_CONNECTION_XML = ET.parse(os.path.join( - TEST_DIR, 'assets', 'CONNECTION.xml')).getroot() +TABLEAU_CONNECTION_XML = os.path.join(TEST_DIR, 'assets', 'CONNECTION.xml') TABLEAU_10_TWBX = os.path.join(TEST_DIR, 'assets', 'TABLEAU_10_TWBX.twbx') @@ -51,7 +50,7 @@ def test_can_extract_federated_connections(self): class ConnectionModelTests(unittest.TestCase): def setUp(self): - self.connection = TABLEAU_CONNECTION_XML + self.connection = ET.parse(TABLEAU_CONNECTION_XML).getroot() def test_can_read_attributes_from_connection(self): conn = Connection(self.connection) @@ -60,15 +59,24 @@ def test_can_read_attributes_from_connection(self): self.assertEqual(conn.server, 'mssql2012') self.assertEqual(conn.dbclass, 'sqlserver') self.assertEqual(conn.authentication, 'sspi') + self.assertEqual(conn.port, '1433') def test_can_write_attributes_to_connection(self): conn = Connection(self.connection) conn.dbname = 'BubblesInMyDrink' conn.server = 'mssql2014' conn.username = 'bob' + conn.port = '1337' self.assertEqual(conn.dbname, 'BubblesInMyDrink') self.assertEqual(conn.username, 'bob') self.assertEqual(conn.server, 'mssql2014') + self.assertEqual(conn.port, '1337') + + def test_can_delete_port_from_connection(self): + conn = Connection(self.connection) + conn.port = None + self.assertEqual(conn.port, None) + self.assertIsNone(conn._connectionXML.get('port')) def test_bad_dbclass_rasies_attribute_error(self): conn = Connection(self.connection) @@ -90,11 +98,13 @@ def test_can_create_datasource_from_connections(self): conn1 = Connection.from_attributes( server='a', dbname='b', username='c', dbclass='mysql', authentication='d') conn2 = Connection.from_attributes( - server='1', dbname='2', username='3', dbclass='mysql', authentication='7') + server='1', dbname='2', username='3', dbclass='mysql', port='1337', authentication='7') ds = Datasource.from_connections('test', connections=[conn1, conn2]) self.assertEqual(ds.connections[0].server, 'a') + self.assertEqual(ds.connections[0].port, None) self.assertEqual(ds.connections[1].server, '1') + self.assertEqual(ds.connections[1].port, '1337') class ConnectionParserInComplicatedWorkbooks(unittest.TestCase): From cb0de80cd989b7aa608e76a83b419d7f4c57c4d0 Mon Sep 17 00:00:00 2001 From: Russell Hay Date: Tue, 18 Oct 2016 14:24:14 -0700 Subject: [PATCH 12/16] Add caption support for datasources (#99) * Add caption support for datasources * fix a pep8 issue E303 * Add ability to remove the caption using del --- tableaudocumentapi/datasource.py | 15 +++++++++++ test/assets/datasource_test.tds | 3 ++- test/test_datasource.py | 43 +++++++++++++++++++++++++++++++- 3 files changed, 59 insertions(+), 2 deletions(-) diff --git a/tableaudocumentapi/datasource.py b/tableaudocumentapi/datasource.py index 63157e9..a34cba5 100644 --- a/tableaudocumentapi/datasource.py +++ b/tableaudocumentapi/datasource.py @@ -137,6 +137,7 @@ def __init__(self, dsxml, filename=None): self._name = self._datasourceXML.get('name') or self._datasourceXML.get( 'formatted-name') # TDS files don't have a name attribute self._version = self._datasourceXML.get('version') + self._caption = self._datasourceXML.get('caption', '') self._connection_parser = ConnectionParser( self._datasourceXML, version=self._version) self._connections = self._connection_parser.get_connections() @@ -207,6 +208,20 @@ def name(self): def version(self): return self._version + @property + def caption(self): + return self._caption + + @caption.setter + def caption(self, value): + self._datasourceXML.set('caption', value) + self._caption = value + + @caption.deleter + def caption(self): + del self._datasourceXML.attrib['caption'] + self._caption = '' + ########### # connections ########### diff --git a/test/assets/datasource_test.tds b/test/assets/datasource_test.tds index 5f280eb..407127d 100644 --- a/test/assets/datasource_test.tds +++ b/test/assets/datasource_test.tds @@ -1,5 +1,6 @@ - + diff --git a/test/test_datasource.py b/test/test_datasource.py index baf5bc3..838bc55 100644 --- a/test/test_datasource.py +++ b/test/test_datasource.py @@ -1,5 +1,9 @@ -import unittest +import os import os.path +import shutil +import tempfile +import unittest + from tableaudocumentapi import Datasource, Workbook @@ -22,6 +26,19 @@ class DataSourceFieldsTDS(unittest.TestCase): def setUp(self): self.ds = Datasource.from_file(TEST_TDS_FILE) + self.to_delete = set() + + def cleanUp(self): + for path in self.to_delete: + if os.path.isdir(path): + shutil.rmtree(path, ignore_errors=True) + elif os.path.isfile(path): + os.unlink(path) + + def get_temp_file(self, filename): + tempdir = tempfile.mkdtemp('tda-datasource') + self.to_delete.add(tempdir) + return os.path.join(tempdir, filename) def test_datasource_returns_correct_fields(self): self.assertIsNotNone(self.ds.fields) @@ -63,6 +80,30 @@ def test_datasource_field_description(self): self.assertIsNotNone(actual) self.assertTrue(u'muted gray' in actual) + def test_datasource_caption(self): + actual = self.ds.caption + self.assertIsNotNone(actual) + self.assertEqual(actual, 'foo') + + def test_datasource_can_set_caption(self): + filename = self.get_temp_file('test_datasource_can_set_caption') + self.ds.caption = 'bar' + self.ds.save_as(filename) + + actual = Datasource.from_file(filename) + self.assertIsNotNone(actual) + self.assertIsNotNone(actual.caption) + self.assertEqual(actual.caption, 'bar') + + def test_datasource_can_remove_caption(self): + filename = self.get_temp_file('test_datasource_can_remove_caption') + del self.ds.caption + self.ds.save_as(filename) + + actual = Datasource.from_file(filename) + self.assertIsNotNone(actual) + self.assertEqual(actual.caption, '') + def test_datasource_clear_repository_location(self): filename = os.path.join(TEST_ASSET_DIR, 'clear-repository-test.tds') From 601b05c204d94cfa3d61ff0ec2d6671c4f0fe01b Mon Sep 17 00:00:00 2001 From: T8y8 Date: Sun, 23 Oct 2016 17:13:05 -0700 Subject: [PATCH 13/16] Crack at a mkdocs build skeleton --- docs/api-ref.md | 40 ++++++++++++++++ docs/contributing.md | 32 +++++++++++++ docs/css/extra.css | 14 ++++++ docs/devguide.md | 41 ++++++++++++++++ docs/index.md | 110 +++++++++++++++++++++++++++++++++++++++++++ docs/user-guide.md | 6 +++ mkdocs.yml | 23 +++++++++ 7 files changed, 266 insertions(+) create mode 100644 docs/api-ref.md create mode 100644 docs/contributing.md create mode 100644 docs/css/extra.css create mode 100644 docs/devguide.md create mode 100644 docs/index.md create mode 100644 docs/user-guide.md create mode 100644 mkdocs.yml diff --git a/docs/api-ref.md b/docs/api-ref.md new file mode 100644 index 0000000..7210507 --- /dev/null +++ b/docs/api-ref.md @@ -0,0 +1,40 @@ +# Workbooks +```python +class Workbook(filename): +``` + +The Workbook class represents a tableau workbook. It may be either a TWB or TWBX, and the library will handle packaging and unpackaging automatically. + +**Params:** +`filename` takes a string representing the path to the workbook file. + +**Raises:** +`TableauVersionNotSupportedException` if the workbook is not a supported version +`TableauInvalidFileException` if the file is not a valid tableau workbook file + +**Methods:** +`Workbook.save(self):` +Saves any changes to the workbook to the existing file + +`Workbook.save_as(self, new_filename):` +Saves any changes to the workbook to a new file specified by the `new_file` parameter + +**Properities:** +`self.worksheets:` Returns a list of worksheets found in the workbook +`self.datasources:` Returns a list of Datasource objects found in the workbook +`self.filename:` Returns the filename of the workbook + +#Datasources +```python +class Datasource(dsxml, filename=None) +``` + +# Connections +```python +class Connection(connxml) +``` + +# Fields +```python +class Workbook(column_xml=None, metadata_xml=None) +``` diff --git a/docs/contributing.md b/docs/contributing.md new file mode 100644 index 0000000..ae41c2e --- /dev/null +++ b/docs/contributing.md @@ -0,0 +1,32 @@ +# Contributing + +We welcome contributions to this project! + +There are many ways to contribute! We welcome all contributions including: + +* File an Issue +* Request a Feature +* Implement a Requested Feature +* Fix an Issue/Bug +* Add/Fix documentation + +Contributions must follow the guidelines outlined on the [Tableau Organization](http://tableau.github.io/) page, though filing an issue or requesting +a feature do not require the CLA. + +## Issues and Feature Requests + +To submit an issue/bug report, or to request a feature, please submit a [github issue](https://github.com/tableau/document-api-python/issues) to the repo. + +If you are submiting a bug report, please provide as much information as you can, including clear and concise repro steps, attaching any necessary +files to assist in the repro. **Be sure to scrub the files of any potentially sensitive information. Issues are public.** + +For a feature request, please try to describe the scenario you are trying to accomplish that requires the feature. This will help us understand +the limitations that you are running into, and provide us with a use case to know if we've satisfied your request. Psuedo code that describes how you would like things to work is also very helpful. + +## Fixes, Implementations, and Documentation + +For all other things, please submit a PR that includes the fix, documentation, or new code that you are trying to contribute. More information on +creating a PR can be found in the [github documentation](https://help.github.com/articles/creating-a-pull-request/) + +If the feature is complex or has multiple solutions that could be equally appropriate approaches, it would be helpful to file an issue to discuss the +design trade-offs of each solution before implementing, to allow us to collectively arrive at the best solution, which most likely exists in the middle somewhere. diff --git a/docs/css/extra.css b/docs/css/extra.css new file mode 100644 index 0000000..804bb59 --- /dev/null +++ b/docs/css/extra.css @@ -0,0 +1,14 @@ +pre code { + white-space: pre; + word-wrap: normal; + display: block; + padding: 12px; + font-size: 14px; +} + +code { + white-space: pre-wrap; + word-wrap: break-word; + padding: 2px 5px; + font-size: 16px; +} diff --git a/docs/devguide.md b/docs/devguide.md new file mode 100644 index 0000000..aef40ca --- /dev/null +++ b/docs/devguide.md @@ -0,0 +1,41 @@ +Developer Guide +--------------- + +### Making your first patch + +0. Make sure you've signed the CLA +1. Clone the repo + + ```shell + git clone http://github.com/tableau/document-api-python + ``` + +2. Run the tests to make sure everything is peachy + + ```shell + python setup.py test + ``` + +3. Set up the feature, fix, or documentation branch. +It is recommended to use the format [issue#]-[type]-[description] (e.g. 13-fix-connection-bug) + + ```shell + git checkout -b 13-feature-new-stuff + ``` + +4. Code and Commit! + Here's a quick checklist for ensuring a good diff: + - The diff touches the minimal amount of files possible while still fufilling the purpose of the diff + - The diff uses Unix line endings + - The diff adheres to our PEP8 style guides. If you've cloned the repo you can run `pycodestyle .` +5. Add Tests +6. Update Documentation + Our documentation is written in markdown and built with [Mkdocs](http://www.mkdocs.org). More information on how to update and build the docs can be found [here](#updating-documentation) +7. Run the tests again and make sure they pass! +8. Submit to your fork +9. Submit a PR +10. Wait for a review, and address any feedback. + +### Updating Documentation + +### Running Tests diff --git a/docs/index.md b/docs/index.md new file mode 100644 index 0000000..45ccfbb --- /dev/null +++ b/docs/index.md @@ -0,0 +1,110 @@ +Tableau Document API +---------------------- + +This repo contains Python source and example files for the Tableau Document API. We're just getting started and have plans to expand what you find here. Help us by submitting feedback, issues, and pull requests! + + +The Document API provides a supported way to programmatically make updates to Tableau workbook and data source files. If you've been making changes to these file types by directly updating the XML--that is, by XML hacking--this SDK is for you :) + + +Features include: + +- Support for 9.X, and 10.X workbook and data source files + - Including TDSX and TWBX files +- Getting connection information from data sources and workbooks + - Server Name + - Username + - Database Name + - Authentication Type + - Connection Type +- Updating connection information in workbooks and data sources + - Server Name + - Username + - Database Name +- Getting Field information from data sources and workbooks + - Get all fields in a data source + - Get all fields in use by certain sheets in a workbook + +We don't yet support creating files from scratch, adding extracts into workbooks or data sources, or updating field information + + +### Getting Started +To use this SDK, you must have Python installed. You can use either 2.7.X or 3.3 and later. + +#### Installing the latest stable version (preferred) + +```shell +pip install tableaudocumentapi +``` + +#### Installing From Source + +Download the `.zip` file that contains the SDK. Unzip the file and then run the following command: + +```shell +pip install -e +``` + +#### Installing the Development Version from Git + +*Only do this if you know you want the development version, no guarantee that we won't break APIs during development* + +```shell +pip install git+https://github.com/tableau/document-api-python.git@development +``` + +If you go this route, but want to switch back to the non-development version, you need to run the following command before installing the stable version: + +```shell +pip uninstall tableaudocumentapi +``` + +###Basics +The following example shows the basic syntax for using the Document API to update a workbook: + +```python +from tableaudocumentapi import Workbook + +sourceWB = Workbook('WorkbookToUpdate.twb') + +sourceWB.datasources[0].connections[0].server = "MY-NEW-SERVER" +sourceWB.datasources[0].connections[0].dbname = "NEW-DATABASE" +sourceWB.datasources[0].connections[0].username = "benl" + +sourceWB.save() +``` + +With Data Integration in Tableau 10, a data source can have multiple connections. To access the connections simply index them like you would datasources. + +```python +from tableaudocumentapi import Workbook + +sourceWB = Workbook('WorkbookToUpdate.twb') + +sourceWB.datasources[0].connections[0].server = "MY-NEW-SERVER" +sourceWB.datasources[0].connections[0].dbname = "NEW-DATABASE" +sourceWB.datasources[0].connections[0].username = "benl" + +sourceWB.datasources[0].connections[1].server = "MY-NEW-SERVER" +sourceWB.datasources[0].connections[1].dbname = "NEW-DATABASE" +sourceWB.datasources[0].connections[1].username = "benl" + + +sourceWB.save() +``` + + +**Notes** + +- Import the `Workbook` object from the `tableaudocumentapi` module. +- To open a workbook, instantiate a `Workbook` object and pass the file name as the first argument. +- The `Workbook` object exposes a list of `datasources` in the workbook +- Each data source object has a `connection` object that supports a `server`, `dbname`, and `username` property. +- Save changes to the workbook by calling the `save` or `save_as` method. + +!!! tip "Introduced In v 2.4" + This call was added in version Tableau 10.3, XSDv2.5 + +###[Examples](samples) + +The downloadable package contains several example scripts that show more detailed usage of the Document API. diff --git a/docs/user-guide.md b/docs/user-guide.md new file mode 100644 index 0000000..c7bb53a --- /dev/null +++ b/docs/user-guide.md @@ -0,0 +1,6 @@ +# Concepts + +Imagine a world without kittens. What a world that would be. +This is why I am going to discuss our document object model. You see +it all starts with Workbooks, Datasources, and Connections. There are +also Fields. diff --git a/mkdocs.yml b/mkdocs.yml new file mode 100644 index 0000000..7548c83 --- /dev/null +++ b/mkdocs.yml @@ -0,0 +1,23 @@ +# Site information +site_name: Tableau Document API (Python) +repo_url: https://github.com/tableau/document-api-python/ +copyright: © Tableau Software + +# Pages and Layout +pages: + - Intro: index.md + - Documentation: + - 'User Guide': user-guide.md + - 'API Ref': api-ref.md + - Contributing: + - contributing.md + - devguide.md + +theme: readthedocs +extra_css: + - css/extra.css + +# Markdown Stuff +markdown_extensions: + - admonition # Customize for 'Version Added' ? + - pymdownx.extra # To use superfences so code and lists work nicely From fdb0ebd9ea3606fb7df53ba4a735c65b06f48d6b Mon Sep 17 00:00:00 2001 From: T8y8 Date: Sun, 23 Oct 2016 19:41:44 -0700 Subject: [PATCH 14/16] Adding Docs pipeline --- .gitignore | 5 +- .travis.yml | 63 ++- contributing.md | 5 +- deploy-docs.sh | 28 + docs/Gemfile | 3 + docs/Gemfile.lock | 129 +++++ docs/_config.yml | 17 + docs/_includes/docs_menu.html | 16 + docs/_includes/footer.html | 8 + docs/_includes/head.html | 15 + docs/_includes/header.html | 29 + docs/_includes/icon-github.svg | 1 + docs/_layouts/default.html | 34 ++ docs/_layouts/docs.html | 31 ++ docs/_layouts/home.html | 19 + docs/assets/logo.png | Bin 0 -> 2800 bytes docs/css/api_ref.css | 709 ++++++++++++++++++++++++ docs/css/github-highlight.css | 224 ++++++++ docs/css/main.css | 276 +++++++++ docs/{ => docs}/api-ref.md | 16 +- docs/{ => docs}/contributing.md | 12 +- docs/{devguide.md => docs/dev-guide.md} | 6 +- docs/docs/index.md | 88 +++ docs/index.md | 142 ++--- docs/user-guide.md | 6 - 25 files changed, 1738 insertions(+), 144 deletions(-) create mode 100644 deploy-docs.sh create mode 100644 docs/Gemfile create mode 100644 docs/Gemfile.lock create mode 100644 docs/_config.yml create mode 100644 docs/_includes/docs_menu.html create mode 100644 docs/_includes/footer.html create mode 100644 docs/_includes/head.html create mode 100644 docs/_includes/header.html create mode 100644 docs/_includes/icon-github.svg create mode 100644 docs/_layouts/default.html create mode 100644 docs/_layouts/docs.html create mode 100644 docs/_layouts/home.html create mode 100644 docs/assets/logo.png create mode 100644 docs/css/api_ref.css create mode 100644 docs/css/github-highlight.css create mode 100644 docs/css/main.css rename docs/{ => docs}/api-ref.md (90%) rename docs/{ => docs}/contributing.md (86%) rename docs/{devguide.md => docs/dev-guide.md} (96%) create mode 100644 docs/docs/index.md delete mode 100644 docs/user-guide.md diff --git a/.gitignore b/.gitignore index ee250af..1084afa 100644 --- a/.gitignore +++ b/.gitignore @@ -52,8 +52,9 @@ coverage.xml # Django stuff: *.log -# Sphinx documentation -docs/_build/ +# Documentation +docs/_site/ +docs/.jekyll-metadata # PyBuilder target/ diff --git a/.travis.yml b/.travis.yml index e2a9073..c4341b7 100644 --- a/.travis.yml +++ b/.travis.yml @@ -1,22 +1,49 @@ language: python -python: - - "2.7" - - "3.3" - - "3.4" - - "3.5" - - "pypy" -# command to install dependencies + +env: + global: + secure: !!binary | + elBwQ2pFZ0wyR3FCQ0xwT2F5LzdCL2JCTHBNSk02ZGJzcDk3UHBGUkNFMmZBWVovTlpFSHB6YVJP + Ykg2NGRMcGlPUDZpQyt3aER4ekpxRTA4aFlXRXc1N0MvcGNjaWZhWldJMGxoM2NsT3VMWDV1Sm90 + QVpmZi9vWjREM2N0NzhhNXZxQ1lTTVU2ZXVmSTZDbXM5d3g2L3Rha0c3MGxBQzJvcVM0R0lFUXV4 + anNTZFlBcnlsd2xNOHJybkxNUHVlL3NBc2dPcUZnTW1HNzNkUWt2SDBKWmw5VUxuem54dUp2bHRN + R3Q0b1RVa2gxN1dGMWlyUFlDNldqWWRpQXNQeWF1K0NCL3gyY25EdGJFOEVpczl1RHZWZWxFZFc3 + K0dkWVF4b1NjME4zWEMvRi9IR0w5L2xKVDF4RVR6c3hDSk4vUERWMW9QVUJwUjJ0RE4yM3RZdWNt + SXhnbzExTWJRUjdoTU5YeVNSNGlseUJKWURxODNpbGNjZWhMdFBTNGhSL3JTcUduU0svV2hPeW1M + d1BRcitZK1luWG8zRTVlUHhzYmJIcmxyejZjbjF6bnlUeXBJL0Rmc2cwWXZtTkV6ekZlalVFbnJI + aTR5eDBuN1dZUElwMGd4N3d4ZUpFUmdWNm5XQW9Pb0w0UVNIU1lLWG54Rjk0NkZ5Y3J0QmlVc3hS + NFlNRHlMY1lKRGUxTWZTblVWUFhtSEYzUmJlVmxIQzFKWHZ0RittUFFGSXdlSjZHaWRMTE1rSlhC + VkVjNFBMbnk1VkNOMHRPWUhpdlU4MDYwSGg4K0N0Rk9zWnh1VUM3OWRhNDRvQ2huSnBiM0VtUHUw + OWVxRjRuVGk0R3NGdUNZVFFubi9aMHBCdVZCRUpVNWtoTTJzZGNyWEJiWlFKeTRQVWppYkNSc1U9 + install: - - "pip install -e ." - - "pip install pep8" -# command to run tests +- pip install -e . +- pip install pycodestyle +- pip install mkdocs +- pip install pymdown-extensions +- chmod +x ./deploy-docs.sh + +matrix: + include: + - os: linux + python: 2.7 + - os: linux + python: 3.3 + - os: linux + python: 3.4 + - os: linux + env: BUILD_DOCS="true" + python: 3.5 + - os: linux + python: pypy + script: - # Tests - - python setup.py test - # pep8 - - pep8 . - # Examples - - (cd "samples/replicate-workbook" && python replicate_workbook.py) - - (cd "samples/list-tds-info" && python list_tds_info.py) - - (cd "samples/show-fields" && python show_fields.py) +- python setup.py test +- pycodestyle . +- (cd "samples/replicate-workbook" && python replicate_workbook.py) +- (cd "samples/list-tds-info" && python list_tds_info.py) +- (cd "samples/show-fields" && python show_fields.py) +- mkdocs build -s --clean +after_success: + - bash deploy-docs.sh diff --git a/contributing.md b/contributing.md index 15fc5f8..ef54847 100644 --- a/contributing.md +++ b/contributing.md @@ -1,4 +1,7 @@ -# Contributing +--- +title: Contributing +layout: docs +--- We welcome contributions to this project! diff --git a/deploy-docs.sh b/deploy-docs.sh new file mode 100644 index 0000000..b7e5a14 --- /dev/null +++ b/deploy-docs.sh @@ -0,0 +1,28 @@ +#!/bin/bash + +set -o errexit -o nounset + +if [ "$TRAVIS_BRANCH" != "master" -o "$BUILD_DOCS" != "true" ] +then + echo "Not deploying docs" + exit 0 +fi + +rev=$(git rev-parse --short HEAD) + +cd site + +git init +git config user.name "T8y8" +git config user.email "kingt8y8@gmail.com" + + +git remote add upstream "https://$GH_TOKEN@github.com/t8y8/document-api-python.git" +git fetch upstream +git reset upstream/gh-pages + +touch . + +git add -A . +git commit -m "auto-build rebuild pages at ${rev}" +git push -q upstream HEAD:gh-pages diff --git a/docs/Gemfile b/docs/Gemfile new file mode 100644 index 0000000..775d954 --- /dev/null +++ b/docs/Gemfile @@ -0,0 +1,3 @@ +source 'https://rubygems.org' +gem 'github-pages', group: :jekyll_plugins + diff --git a/docs/Gemfile.lock b/docs/Gemfile.lock new file mode 100644 index 0000000..e555d12 --- /dev/null +++ b/docs/Gemfile.lock @@ -0,0 +1,129 @@ +GEM + remote: https://rubygems.org/ + specs: + activesupport (4.2.6) + i18n (~> 0.7) + json (~> 1.7, >= 1.7.7) + minitest (~> 5.1) + thread_safe (~> 0.3, >= 0.3.4) + tzinfo (~> 1.1) + addressable (2.4.0) + coffee-script (2.4.1) + coffee-script-source + execjs + coffee-script-source (1.10.0) + colorator (0.1) + ethon (0.9.0) + ffi (>= 1.3.0) + execjs (2.7.0) + faraday (0.9.2) + multipart-post (>= 1.2, < 3) + ffi (1.9.10) + ffi (1.9.10-x86-mingw32) + gemoji (2.1.0) + github-pages (80) + github-pages-health-check (= 1.1.0) + jekyll (= 3.1.6) + jekyll-coffeescript (= 1.0.1) + jekyll-feed (= 0.5.1) + jekyll-gist (= 1.4.0) + jekyll-github-metadata (= 1.11.1) + jekyll-mentions (= 1.1.2) + jekyll-paginate (= 1.1.0) + jekyll-redirect-from (= 0.10.0) + jekyll-sass-converter (= 1.3.0) + jekyll-seo-tag (= 1.4.0) + jekyll-sitemap (= 0.10.0) + jemoji (= 0.6.2) + kramdown (= 1.10.0) + liquid (= 3.0.6) + listen (= 3.0.6) + mercenary (~> 0.3) + rouge (= 1.10.1) + terminal-table (~> 1.4) + github-pages-health-check (1.1.0) + addressable (~> 2.3) + net-dns (~> 0.8) + octokit (~> 4.0) + public_suffix (~> 1.4) + typhoeus (~> 0.7) + html-pipeline (2.4.1) + activesupport (>= 2, < 5) + nokogiri (>= 1.4) + i18n (0.7.0) + jekyll (3.1.6) + colorator (~> 0.1) + jekyll-sass-converter (~> 1.0) + jekyll-watch (~> 1.1) + kramdown (~> 1.3) + liquid (~> 3.0) + mercenary (~> 0.3.3) + rouge (~> 1.7) + safe_yaml (~> 1.0) + jekyll-coffeescript (1.0.1) + coffee-script (~> 2.2) + jekyll-feed (0.5.1) + jekyll-gist (1.4.0) + octokit (~> 4.2) + jekyll-github-metadata (1.11.1) + octokit (~> 4.0) + jekyll-mentions (1.1.2) + html-pipeline (~> 2.3) + jekyll (~> 3.0) + jekyll-paginate (1.1.0) + jekyll-redirect-from (0.10.0) + jekyll (>= 2.0) + jekyll-sass-converter (1.3.0) + sass (~> 3.2) + jekyll-seo-tag (1.4.0) + jekyll (~> 3.0) + jekyll-sitemap (0.10.0) + jekyll-watch (1.4.0) + listen (~> 3.0, < 3.1) + jemoji (0.6.2) + gemoji (~> 2.0) + html-pipeline (~> 2.2) + jekyll (>= 3.0) + json (1.8.3) + kramdown (1.10.0) + liquid (3.0.6) + listen (3.0.6) + rb-fsevent (>= 0.9.3) + rb-inotify (>= 0.9.7) + mercenary (0.3.6) + mini_portile2 (2.0.0) + minitest (5.9.0) + multipart-post (2.0.0) + net-dns (0.8.0) + nokogiri (1.6.7.2) + mini_portile2 (~> 2.0.0.rc2) + nokogiri (1.6.7.2-x86-mingw32) + mini_portile2 (~> 2.0.0.rc2) + octokit (4.3.0) + sawyer (~> 0.7.0, >= 0.5.3) + public_suffix (1.5.3) + rb-fsevent (0.9.7) + rb-inotify (0.9.7) + ffi (>= 0.5.0) + rouge (1.10.1) + safe_yaml (1.0.4) + sass (3.4.22) + sawyer (0.7.0) + addressable (>= 2.3.5, < 2.5) + faraday (~> 0.8, < 0.10) + terminal-table (1.5.2) + thread_safe (0.3.5) + typhoeus (0.8.0) + ethon (>= 0.8.0) + tzinfo (1.2.2) + thread_safe (~> 0.1) + +PLATFORMS + ruby + x86-mingw32 + +DEPENDENCIES + github-pages + +BUNDLED WITH + 1.12.5 diff --git a/docs/_config.yml b/docs/_config.yml new file mode 100644 index 0000000..1a4c913 --- /dev/null +++ b/docs/_config.yml @@ -0,0 +1,17 @@ +# Site settings +title: Tableau Document API +email: jdominguez@tableau.com +description: Programmatically update your Tableau workbooks and data sources. +baseurl: "/document-api-python" +permalinks: pretty +defaults: + - + scope: + path: "" # Apply to all files + values: + layout: "default" + +# Build settings +markdown: kramdown +highlighter: rouge + diff --git a/docs/_includes/docs_menu.html b/docs/_includes/docs_menu.html new file mode 100644 index 0000000..b7a76d5 --- /dev/null +++ b/docs/_includes/docs_menu.html @@ -0,0 +1,16 @@ +
+ +
diff --git a/docs/_includes/footer.html b/docs/_includes/footer.html new file mode 100644 index 0000000..5cae7ea --- /dev/null +++ b/docs/_includes/footer.html @@ -0,0 +1,8 @@ + +
+
+ +

This site is open source. Suggestions and pull requests are welcome on our GitHub page.

+

(c) Copyright 2016 Tableau

+
+
diff --git a/docs/_includes/head.html b/docs/_includes/head.html new file mode 100644 index 0000000..288d960 --- /dev/null +++ b/docs/_includes/head.html @@ -0,0 +1,15 @@ + + + + + {% if page.title %}{{ page.title | escape }}{% else %}{{ site.title | escape }}{% endif %} + + + + + + + + + + diff --git a/docs/_includes/header.html b/docs/_includes/header.html new file mode 100644 index 0000000..cad2487 --- /dev/null +++ b/docs/_includes/header.html @@ -0,0 +1,29 @@ + diff --git a/docs/_includes/icon-github.svg b/docs/_includes/icon-github.svg new file mode 100644 index 0000000..4422c4f --- /dev/null +++ b/docs/_includes/icon-github.svg @@ -0,0 +1 @@ + diff --git a/docs/_layouts/default.html b/docs/_layouts/default.html new file mode 100644 index 0000000..38ee020 --- /dev/null +++ b/docs/_layouts/default.html @@ -0,0 +1,34 @@ + + + + + {% include head.html %} + + + +
+ {% include header.html %} +
    + {% for post in site.posts %} +
    +

    {{ post.title }}

    +
    +

    Posted on {{ post.date | date: "%-d %B %Y" }}

    +
    +

    + {{ post.abstract }} +

    + {% if post.photoname %} + {% endif %} +
    +
    + {{ post.content }} +
    +
    + {% endfor %} +
+ {% include footer.html %} +
+ + + diff --git a/docs/_layouts/docs.html b/docs/_layouts/docs.html new file mode 100644 index 0000000..1269e39 --- /dev/null +++ b/docs/_layouts/docs.html @@ -0,0 +1,31 @@ +--- +layout: docs +--- + + + + + + {% include head.html %} + + + +
+ {% include header.html %} + {% include docs_menu.html %} + +
+

{{ page.title }}

+ +
+ {{ content }} + {% include footer.html %} +
+
+ + + diff --git a/docs/_layouts/home.html b/docs/_layouts/home.html new file mode 100644 index 0000000..c2cf32f --- /dev/null +++ b/docs/_layouts/home.html @@ -0,0 +1,19 @@ +--- +layout: home +--- + + + + + {% include head.html %} + + + +
+ {% include header.html %} + {{ content }} + {% include footer.html %} +
+ + + diff --git a/docs/assets/logo.png b/docs/assets/logo.png new file mode 100644 index 0000000000000000000000000000000000000000..60761152152291896e7b27f94d981fc82e71a2dd GIT binary patch literal 2800 zcmb7GdpuNWA3rk7jj(zc zA_9#@6Cwf+j{N!tppenAD|!*Tt? zar#(2+n89aH%lSV-lRb3+Ff~Q<4W3MbT z@VqTFFgsTxdBECEUTw_=QFxAOl%7~&W%r{p6`lSZYEWfw_N(j+_voCtm@(sQg^ZaR zsIPw<-y$EgstQv?0+7GAo#~0Ybf<5&_#AxMFXwb?)9ujtx3<&f(hsV490=3nB^pAJ zMTU9sKs7)h8oMvf0Qh2Q4X6li#W2BNGXt$=HfdHSpi}0iurAQrwHK3}3Gfe>Z7BX< zq3$f|V$_05hx;%OITN-*535$zVg^snD5$=sNBEW(^)1Yyj%k;vy{wVRpW9#M8}ico z`xoM42SX)!goCAfAkAyd301SblpHR7C55iE*E9iL;}XNM-OoAN8fZ={l9N%*^3B|5 z+V!G$`?J+2lS2Zi$d%7H)4u(#VPVgB&d*}qPUgG#ThLM3Ie<@F!WLyAXzZ~#FYjD+ zIcvcy&*fuR^=SCM2%}Q-v zc6@V!Cjpy)p^eN{4k)7%h1+@LE#?q-XV*rWJ!rK@qaIQK(v4YOFAB7x9QfDe6NM9T z-$s!EQY1{y|L>u^zOJag>xEZBY%Z{hcR3Wh0HoqajIrx=-0Sw7LfNocy;ckD zlN=YnmJy+F`L(I;)F4-+U7OKAJsbV3r2%VZ}eDU0r44YGzvYI}6J8SJ|guC1AP%`f-qACE3fV`|%cn zrm1{UtI64mKbG$DOOHzKkquq?rE8foh%;9Zm5nB+dL)iK+JaZ!U~6g!!hIxBnI&b# z$l_tno&l91PI7OBh3n}-#xAoh>oCc6XWvxt`wJTuf^d(s<}9GdK6Q<@;aQV|ORNy{ zSCSp6KuQ-%D(pXbx-1S)|y2kxwzf%4+Yp1u28EW3orO>jQfZQH*0SGO#V6wj~LH z<)E0yt_=N*&R#lMQ)g8PJQE`8ISv+%*X0_3T9W3rIcWicZqKG%Yk(AR--d?gIh72) z2r+UimAg<>@mA|l^qZc=n4@o@>6*ej!eE54DU+`cUpewrq~Ax-pgA#OG0qwL6UKbl z=eL@fTZ+?e-#8TA!rJd7GLVpNwr!iz;6k09xMrmF4 zt`Smbk^5c}plr=9^O)~g_!J7(`n47)Ylp(22O5Cjh!n;$4dB}~!?dtq_0Gs8N3eQ1 zP17Ubq4G{@a|0!xqt}$?3}}UEp`@G7$sOPMYInvg^IFEIqhj{zM)b;clX8_^($FpV zt<%~0;HRkUMv~TXfd4S@#1#${xvfNjuoX5p@EfZjDDz}~u|v7Nrpl4=zes(cc3LCn9^5yHcG4y0kRne# ztY1!+o3y&qA zttx95Z>yADY4Q91Ulv7;c6mO(mRHs9)^Np*XLpaxze8FIHcYEz0#evG5M-$;1_?3( za|j;>1-mGX;#mPCHLIkrNa>*{o6z_CiqqjK0lkSx&FzkS7Yk+B-62H&(gnE395AbQp4>LZYT`AK6 z;mN3uerOU4Zk;)^D5Po7bTw{9&P-|lK+$M(zbm&!Xc)N6RXFL6Tkw02sw!d+QCar@ z5bLMn&_+=8JJX;%;!;wr(0~AiZc26oNF#=17!(BX|MH;|H80s{HcH;Y$85@-7/*! normalize.css v1.1.3 | MIT License | git.io/normalize */ +/* 2 */ +/* ========================================================================== Tables ========================================================================== */ +/** Remove most spacing between table cells. */ +table { border-collapse: collapse; border-spacing: 0; } + +/* Visual Studio-like style based on original C# coloring by Jason Diamond */ +.hljs { display: inline-block; padding: 0.5em; background: white; color: black; } + +.hljs-comment, .hljs-annotation, .hljs-template_comment, .diff .hljs-header, .hljs-chunk, .apache .hljs-cbracket { color: #008000; } + +.hljs-keyword, .hljs-id, .hljs-built_in, .css .smalltalk .hljs-class, .hljs-winutils, .bash .hljs-variable, .tex .hljs-command, .hljs-request, .hljs-status, .nginx .hljs-title { color: #00f; } + +.xml .hljs-tag { color: #00f; } +.xml .hljs-tag .hljs-value { color: #00f; } + +.hljs-string, .hljs-title, .hljs-parent, .hljs-tag .hljs-value, .hljs-rules .hljs-value { color: #a31515; } + +.ruby .hljs-symbol { color: #a31515; } +.ruby .hljs-symbol .hljs-string { color: #a31515; } + +.hljs-template_tag, .django .hljs-variable, .hljs-addition, .hljs-flow, .hljs-stream, .apache .hljs-tag, .hljs-date, .tex .hljs-formula, .coffeescript .hljs-attribute { color: #a31515; } + +.ruby .hljs-string, .hljs-decorator, .hljs-filter .hljs-argument, .hljs-localvars, .hljs-array, .hljs-attr_selector, .hljs-pseudo, .hljs-pi, .hljs-doctype, .hljs-deletion, .hljs-envvar, .hljs-shebang, .hljs-preprocessor, .hljs-pragma, .userType, .apache .hljs-sqbracket, .nginx .hljs-built_in, .tex .hljs-special, .hljs-prompt { color: #2b91af; } + +.hljs-phpdoc, .hljs-javadoc, .hljs-xmlDocTag { color: #808080; } + +.vhdl .hljs-typename { font-weight: bold; } +.vhdl .hljs-string { color: #666666; } +.vhdl .hljs-literal { color: #a31515; } +.vhdl .hljs-attribute { color: #00b0e8; } + +.xml .hljs-attribute { color: #f00; } + +.col > :first-child, .col-1 > :first-child, .col-2 > :first-child, .col-3 > :first-child, .col-4 > :first-child, .col-5 > :first-child, .col-6 > :first-child, .col-7 > :first-child, .col-8 > :first-child, .col-9 > :first-child, .col-10 > :first-child, .col-11 > :first-child, .tsd-panel > :first-child, ul.tsd-descriptions > li > :first-child, .col > :first-child > :first-child, .col-1 > :first-child > :first-child, .col-2 > :first-child > :first-child, .col-3 > :first-child > :first-child, .col-4 > :first-child > :first-child, .col-5 > :first-child > :first-child, .col-6 > :first-child > :first-child, .col-7 > :first-child > :first-child, .col-8 > :first-child > :first-child, .col-9 > :first-child > :first-child, .col-10 > :first-child > :first-child, .col-11 > :first-child > :first-child, .tsd-panel > :first-child > :first-child, ul.tsd-descriptions > li > :first-child > :first-child, .col > :first-child > :first-child > :first-child, .col-1 > :first-child > :first-child > :first-child, .col-2 > :first-child > :first-child > :first-child, .col-3 > :first-child > :first-child > :first-child, .col-4 > :first-child > :first-child > :first-child, .col-5 > :first-child > :first-child > :first-child, .col-6 > :first-child > :first-child > :first-child, .col-7 > :first-child > :first-child > :first-child, .col-8 > :first-child > :first-child > :first-child, .col-9 > :first-child > :first-child > :first-child, .col-10 > :first-child > :first-child > :first-child, .col-11 > :first-child > :first-child > :first-child, .tsd-panel > :first-child > :first-child > :first-child, ul.tsd-descriptions > li > :first-child > :first-child > :first-child { margin-top: 0; } +.col > :last-child, .col-1 > :last-child, .col-2 > :last-child, .col-3 > :last-child, .col-4 > :last-child, .col-5 > :last-child, .col-6 > :last-child, .col-7 > :last-child, .col-8 > :last-child, .col-9 > :last-child, .col-10 > :last-child, .col-11 > :last-child, .tsd-panel > :last-child, ul.tsd-descriptions > li > :last-child, .col > :last-child > :last-child, .col-1 > :last-child > :last-child, .col-2 > :last-child > :last-child, .col-3 > :last-child > :last-child, .col-4 > :last-child > :last-child, .col-5 > :last-child > :last-child, .col-6 > :last-child > :last-child, .col-7 > :last-child > :last-child, .col-8 > :last-child > :last-child, .col-9 > :last-child > :last-child, .col-10 > :last-child > :last-child, .col-11 > :last-child > :last-child, .tsd-panel > :last-child > :last-child, ul.tsd-descriptions > li > :last-child > :last-child, .col > :last-child > :last-child > :last-child, .col-1 > :last-child > :last-child > :last-child, .col-2 > :last-child > :last-child > :last-child, .col-3 > :last-child > :last-child > :last-child, .col-4 > :last-child > :last-child > :last-child, .col-5 > :last-child > :last-child > :last-child, .col-6 > :last-child > :last-child > :last-child, .col-7 > :last-child > :last-child > :last-child, .col-8 > :last-child > :last-child > :last-child, .col-9 > :last-child > :last-child > :last-child, .col-10 > :last-child > :last-child > :last-child, .col-11 > :last-child > :last-child > :last-child, .tsd-panel > :last-child > :last-child > :last-child, ul.tsd-descriptions > li > :last-child > :last-child > :last-child { margin-bottom: 0; } + +@media (max-width: 640px) { .container { padding: 0 20px; } } + +.container-main { padding-bottom: 200px; } + +.row { position: relative; margin: 0 -10px; } +.row:after { visibility: hidden; display: block; content: ""; clear: both; height: 0; } + +.col, .col-1, .col-2, .col-3, .col-4, .col-5, .col-6, .col-7, .col-8, .col-9, .col-10, .col-11 { box-sizing: border-box; float: left; padding: 0 10px; } + +.col-1 { width: 8.33333%; } + +.offset-1 { margin-left: 8.33333%; } + +.col-2 { width: 16.66667%; } + +.offset-2 { margin-left: 16.66667%; } + +.col-3 { width: 25%; } + +.offset-3 { margin-left: 25%; } + +.col-4 { width: 33.33333%; } + +.offset-4 { margin-left: 33.33333%; } + +.col-5 { width: 41.66667%; } + +.offset-5 { margin-left: 41.66667%; } + +.col-6 { width: 50%; } + +.offset-6 { margin-left: 50%; } + +.col-7 { width: 58.33333%; } + +.offset-7 { margin-left: 58.33333%; } + +.col-8 { width: 66.66667%; } + +.offset-8 { margin-left: 66.66667%; } + +.col-9 { width: 75%; } + +.offset-9 { margin-left: 75%; } + +.col-10 { width: 83.33333%; } + +.offset-10 { margin-left: 83.33333%; } + +.col-11 { width: 91.66667%; } + +.offset-11 { margin-left: 91.66667%; } + +.tsd-kind-icon { display: block; position: relative; padding-left: 20px; text-indent: -20px; } +.tsd-kind-icon:before { content: ''; display: inline-block; vertical-align: middle; width: 17px; height: 17px; margin: 0 3px 2px 0; background-image: url(); } +@media (-webkit-min-device-pixel-ratio: 1.5), (min-device-pixel-ratio: 1.5), (min-resolution: 144dpi) { .tsd-kind-icon:before { background-image: url(); background-size: 238px 204px; } } + +.tsd-signature.tsd-kind-icon:before { background-position: 0 -153px; } + +.tsd-kind-object-literal > .tsd-kind-icon:before { background-position: 0px -17px; } +.tsd-kind-object-literal.tsd-is-protected > .tsd-kind-icon:before { background-position: -17px -17px; } +.tsd-kind-object-literal.tsd-is-private > .tsd-kind-icon:before { background-position: -34px -17px; } + +.tsd-kind-class > .tsd-kind-icon:before { background-position: 0px -34px; } +.tsd-kind-class.tsd-is-protected > .tsd-kind-icon:before { background-position: -17px -34px; } +.tsd-kind-class.tsd-is-private > .tsd-kind-icon:before { background-position: -34px -34px; } + +.tsd-kind-class.tsd-has-type-parameter > .tsd-kind-icon:before { background-position: 0px -51px; } +.tsd-kind-class.tsd-has-type-parameter.tsd-is-protected > .tsd-kind-icon:before { background-position: -17px -51px; } +.tsd-kind-class.tsd-has-type-parameter.tsd-is-private > .tsd-kind-icon:before { background-position: -34px -51px; } + +.tsd-kind-interface > .tsd-kind-icon:before { background-position: 0px -68px; } +.tsd-kind-interface.tsd-is-protected > .tsd-kind-icon:before { background-position: -17px -68px; } +.tsd-kind-interface.tsd-is-private > .tsd-kind-icon:before { background-position: -34px -68px; } + +.tsd-kind-interface.tsd-has-type-parameter > .tsd-kind-icon:before { background-position: 0px -85px; } +.tsd-kind-interface.tsd-has-type-parameter.tsd-is-protected > .tsd-kind-icon:before { background-position: -17px -85px; } +.tsd-kind-interface.tsd-has-type-parameter.tsd-is-private > .tsd-kind-icon:before { background-position: -34px -85px; } + +.tsd-kind-module > .tsd-kind-icon:before { background-position: 0px -102px; } +.tsd-kind-module.tsd-is-protected > .tsd-kind-icon:before { background-position: -17px -102px; } +.tsd-kind-module.tsd-is-private > .tsd-kind-icon:before { background-position: -34px -102px; } + +.tsd-kind-external-module > .tsd-kind-icon:before { background-position: 0px -102px; } +.tsd-kind-external-module.tsd-is-protected > .tsd-kind-icon:before { background-position: -17px -102px; } +.tsd-kind-external-module.tsd-is-private > .tsd-kind-icon:before { background-position: -34px -102px; } + +.tsd-kind-enum > .tsd-kind-icon:before { background-position: 0px -119px; } +.tsd-kind-enum.tsd-is-protected > .tsd-kind-icon:before { background-position: -17px -119px; } +.tsd-kind-enum.tsd-is-private > .tsd-kind-icon:before { background-position: -34px -119px; } + +.tsd-kind-enum-member > .tsd-kind-icon:before { background-position: 0px -136px; } +.tsd-kind-enum-member.tsd-is-protected > .tsd-kind-icon:before { background-position: -17px -136px; } +.tsd-kind-enum-member.tsd-is-private > .tsd-kind-icon:before { background-position: -34px -136px; } + +.tsd-kind-signature > .tsd-kind-icon:before { background-position: 0px -153px; } +.tsd-kind-signature.tsd-is-protected > .tsd-kind-icon:before { background-position: -17px -153px; } +.tsd-kind-signature.tsd-is-private > .tsd-kind-icon:before { background-position: -34px -153px; } + +.tsd-kind-type-alias > .tsd-kind-icon:before { background-position: 0px -170px; } +.tsd-kind-type-alias.tsd-is-protected > .tsd-kind-icon:before { background-position: -17px -170px; } +.tsd-kind-type-alias.tsd-is-private > .tsd-kind-icon:before { background-position: -34px -170px; } + +.tsd-kind-variable > .tsd-kind-icon:before { background-position: -136px -0px; } +.tsd-kind-variable.tsd-is-protected > .tsd-kind-icon:before { background-position: -153px -0px; } +.tsd-kind-variable.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -0px; } +.tsd-kind-variable.tsd-parent-kind-class > .tsd-kind-icon:before { background-position: -51px -0px; } +.tsd-kind-variable.tsd-parent-kind-class.tsd-is-inherited > .tsd-kind-icon:before { background-position: -68px -0px; } +.tsd-kind-variable.tsd-parent-kind-class.tsd-is-protected > .tsd-kind-icon:before { background-position: -85px -0px; } +.tsd-kind-variable.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited > .tsd-kind-icon:before { background-position: -102px -0px; } +.tsd-kind-variable.tsd-parent-kind-class.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -0px; } +.tsd-kind-variable.tsd-parent-kind-enum > .tsd-kind-icon:before { background-position: -170px -0px; } +.tsd-kind-variable.tsd-parent-kind-enum.tsd-is-protected > .tsd-kind-icon:before { background-position: -187px -0px; } +.tsd-kind-variable.tsd-parent-kind-enum.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -0px; } +.tsd-kind-variable.tsd-parent-kind-interface > .tsd-kind-icon:before { background-position: -204px -0px; } +.tsd-kind-variable.tsd-parent-kind-interface.tsd-is-inherited > .tsd-kind-icon:before { background-position: -221px -0px; } + +.tsd-kind-property > .tsd-kind-icon:before { background-position: -136px -0px; } +.tsd-kind-property.tsd-is-protected > .tsd-kind-icon:before { background-position: -153px -0px; } +.tsd-kind-property.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -0px; } +.tsd-kind-property.tsd-parent-kind-class > .tsd-kind-icon:before { background-position: -51px -0px; } +.tsd-kind-property.tsd-parent-kind-class.tsd-is-inherited > .tsd-kind-icon:before { background-position: -68px -0px; } +.tsd-kind-property.tsd-parent-kind-class.tsd-is-protected > .tsd-kind-icon:before { background-position: -85px -0px; } +.tsd-kind-property.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited > .tsd-kind-icon:before { background-position: -102px -0px; } +.tsd-kind-property.tsd-parent-kind-class.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -0px; } +.tsd-kind-property.tsd-parent-kind-enum > .tsd-kind-icon:before { background-position: -170px -0px; } +.tsd-kind-property.tsd-parent-kind-enum.tsd-is-protected > .tsd-kind-icon:before { background-position: -187px -0px; } +.tsd-kind-property.tsd-parent-kind-enum.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -0px; } +.tsd-kind-property.tsd-parent-kind-interface > .tsd-kind-icon:before { background-position: -204px -0px; } +.tsd-kind-property.tsd-parent-kind-interface.tsd-is-inherited > .tsd-kind-icon:before { background-position: -221px -0px; } + +.tsd-kind-get-signature > .tsd-kind-icon:before { background-position: -136px -17px; } +.tsd-kind-get-signature.tsd-is-protected > .tsd-kind-icon:before { background-position: -153px -17px; } +.tsd-kind-get-signature.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -17px; } +.tsd-kind-get-signature.tsd-parent-kind-class > .tsd-kind-icon:before { background-position: -51px -17px; } +.tsd-kind-get-signature.tsd-parent-kind-class.tsd-is-inherited > .tsd-kind-icon:before { background-position: -68px -17px; } +.tsd-kind-get-signature.tsd-parent-kind-class.tsd-is-protected > .tsd-kind-icon:before { background-position: -85px -17px; } +.tsd-kind-get-signature.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited > .tsd-kind-icon:before { background-position: -102px -17px; } +.tsd-kind-get-signature.tsd-parent-kind-class.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -17px; } +.tsd-kind-get-signature.tsd-parent-kind-enum > .tsd-kind-icon:before { background-position: -170px -17px; } +.tsd-kind-get-signature.tsd-parent-kind-enum.tsd-is-protected > .tsd-kind-icon:before { background-position: -187px -17px; } +.tsd-kind-get-signature.tsd-parent-kind-enum.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -17px; } +.tsd-kind-get-signature.tsd-parent-kind-interface > .tsd-kind-icon:before { background-position: -204px -17px; } +.tsd-kind-get-signature.tsd-parent-kind-interface.tsd-is-inherited > .tsd-kind-icon:before { background-position: -221px -17px; } + +.tsd-kind-set-signature > .tsd-kind-icon:before { background-position: -136px -34px; } +.tsd-kind-set-signature.tsd-is-protected > .tsd-kind-icon:before { background-position: -153px -34px; } +.tsd-kind-set-signature.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -34px; } +.tsd-kind-set-signature.tsd-parent-kind-class > .tsd-kind-icon:before { background-position: -51px -34px; } +.tsd-kind-set-signature.tsd-parent-kind-class.tsd-is-inherited > .tsd-kind-icon:before { background-position: -68px -34px; } +.tsd-kind-set-signature.tsd-parent-kind-class.tsd-is-protected > .tsd-kind-icon:before { background-position: -85px -34px; } +.tsd-kind-set-signature.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited > .tsd-kind-icon:before { background-position: -102px -34px; } +.tsd-kind-set-signature.tsd-parent-kind-class.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -34px; } +.tsd-kind-set-signature.tsd-parent-kind-enum > .tsd-kind-icon:before { background-position: -170px -34px; } +.tsd-kind-set-signature.tsd-parent-kind-enum.tsd-is-protected > .tsd-kind-icon:before { background-position: -187px -34px; } +.tsd-kind-set-signature.tsd-parent-kind-enum.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -34px; } +.tsd-kind-set-signature.tsd-parent-kind-interface > .tsd-kind-icon:before { background-position: -204px -34px; } +.tsd-kind-set-signature.tsd-parent-kind-interface.tsd-is-inherited > .tsd-kind-icon:before { background-position: -221px -34px; } + +.tsd-kind-accessor > .tsd-kind-icon:before { background-position: -136px -51px; } +.tsd-kind-accessor.tsd-is-protected > .tsd-kind-icon:before { background-position: -153px -51px; } +.tsd-kind-accessor.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -51px; } +.tsd-kind-accessor.tsd-parent-kind-class > .tsd-kind-icon:before { background-position: -51px -51px; } +.tsd-kind-accessor.tsd-parent-kind-class.tsd-is-inherited > .tsd-kind-icon:before { background-position: -68px -51px; } +.tsd-kind-accessor.tsd-parent-kind-class.tsd-is-protected > .tsd-kind-icon:before { background-position: -85px -51px; } +.tsd-kind-accessor.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited > .tsd-kind-icon:before { background-position: -102px -51px; } +.tsd-kind-accessor.tsd-parent-kind-class.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -51px; } +.tsd-kind-accessor.tsd-parent-kind-enum > .tsd-kind-icon:before { background-position: -170px -51px; } +.tsd-kind-accessor.tsd-parent-kind-enum.tsd-is-protected > .tsd-kind-icon:before { background-position: -187px -51px; } +.tsd-kind-accessor.tsd-parent-kind-enum.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -51px; } +.tsd-kind-accessor.tsd-parent-kind-interface > .tsd-kind-icon:before { background-position: -204px -51px; } +.tsd-kind-accessor.tsd-parent-kind-interface.tsd-is-inherited > .tsd-kind-icon:before { background-position: -221px -51px; } + +.tsd-kind-function > .tsd-kind-icon:before { background-position: -136px -68px; } +.tsd-kind-function.tsd-is-protected > .tsd-kind-icon:before { background-position: -153px -68px; } +.tsd-kind-function.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -68px; } +.tsd-kind-function.tsd-parent-kind-class > .tsd-kind-icon:before { background-position: -51px -68px; } +.tsd-kind-function.tsd-parent-kind-class.tsd-is-inherited > .tsd-kind-icon:before { background-position: -68px -68px; } +.tsd-kind-function.tsd-parent-kind-class.tsd-is-protected > .tsd-kind-icon:before { background-position: -85px -68px; } +.tsd-kind-function.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited > .tsd-kind-icon:before { background-position: -102px -68px; } +.tsd-kind-function.tsd-parent-kind-class.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -68px; } +.tsd-kind-function.tsd-parent-kind-enum > .tsd-kind-icon:before { background-position: -170px -68px; } +.tsd-kind-function.tsd-parent-kind-enum.tsd-is-protected > .tsd-kind-icon:before { background-position: -187px -68px; } +.tsd-kind-function.tsd-parent-kind-enum.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -68px; } +.tsd-kind-function.tsd-parent-kind-interface > .tsd-kind-icon:before { background-position: -204px -68px; } +.tsd-kind-function.tsd-parent-kind-interface.tsd-is-inherited > .tsd-kind-icon:before { background-position: -221px -68px; } + +.tsd-kind-method > .tsd-kind-icon:before { background-position: -136px -68px; } +.tsd-kind-method.tsd-is-protected > .tsd-kind-icon:before { background-position: -153px -68px; } +.tsd-kind-method.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -68px; } +.tsd-kind-method.tsd-parent-kind-class > .tsd-kind-icon:before { background-position: -51px -68px; } +.tsd-kind-method.tsd-parent-kind-class.tsd-is-inherited > .tsd-kind-icon:before { background-position: -68px -68px; } +.tsd-kind-method.tsd-parent-kind-class.tsd-is-protected > .tsd-kind-icon:before { background-position: -85px -68px; } +.tsd-kind-method.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited > .tsd-kind-icon:before { background-position: -102px -68px; } +.tsd-kind-method.tsd-parent-kind-class.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -68px; } +.tsd-kind-method.tsd-parent-kind-enum > .tsd-kind-icon:before { background-position: -170px -68px; } +.tsd-kind-method.tsd-parent-kind-enum.tsd-is-protected > .tsd-kind-icon:before { background-position: -187px -68px; } +.tsd-kind-method.tsd-parent-kind-enum.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -68px; } +.tsd-kind-method.tsd-parent-kind-interface > .tsd-kind-icon:before { background-position: -204px -68px; } +.tsd-kind-method.tsd-parent-kind-interface.tsd-is-inherited > .tsd-kind-icon:before { background-position: -221px -68px; } + +.tsd-kind-call-signature > .tsd-kind-icon:before { background-position: -136px -68px; } +.tsd-kind-call-signature.tsd-is-protected > .tsd-kind-icon:before { background-position: -153px -68px; } +.tsd-kind-call-signature.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -68px; } +.tsd-kind-call-signature.tsd-parent-kind-class > .tsd-kind-icon:before { background-position: -51px -68px; } +.tsd-kind-call-signature.tsd-parent-kind-class.tsd-is-inherited > .tsd-kind-icon:before { background-position: -68px -68px; } +.tsd-kind-call-signature.tsd-parent-kind-class.tsd-is-protected > .tsd-kind-icon:before { background-position: -85px -68px; } +.tsd-kind-call-signature.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited > .tsd-kind-icon:before { background-position: -102px -68px; } +.tsd-kind-call-signature.tsd-parent-kind-class.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -68px; } +.tsd-kind-call-signature.tsd-parent-kind-enum > .tsd-kind-icon:before { background-position: -170px -68px; } +.tsd-kind-call-signature.tsd-parent-kind-enum.tsd-is-protected > .tsd-kind-icon:before { background-position: -187px -68px; } +.tsd-kind-call-signature.tsd-parent-kind-enum.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -68px; } +.tsd-kind-call-signature.tsd-parent-kind-interface > .tsd-kind-icon:before { background-position: -204px -68px; } +.tsd-kind-call-signature.tsd-parent-kind-interface.tsd-is-inherited > .tsd-kind-icon:before { background-position: -221px -68px; } + +.tsd-kind-function.tsd-has-type-parameter > .tsd-kind-icon:before { background-position: -136px -85px; } +.tsd-kind-function.tsd-has-type-parameter.tsd-is-protected > .tsd-kind-icon:before { background-position: -153px -85px; } +.tsd-kind-function.tsd-has-type-parameter.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -85px; } +.tsd-kind-function.tsd-has-type-parameter.tsd-parent-kind-class > .tsd-kind-icon:before { background-position: -51px -85px; } +.tsd-kind-function.tsd-has-type-parameter.tsd-parent-kind-class.tsd-is-inherited > .tsd-kind-icon:before { background-position: -68px -85px; } +.tsd-kind-function.tsd-has-type-parameter.tsd-parent-kind-class.tsd-is-protected > .tsd-kind-icon:before { background-position: -85px -85px; } +.tsd-kind-function.tsd-has-type-parameter.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited > .tsd-kind-icon:before { background-position: -102px -85px; } +.tsd-kind-function.tsd-has-type-parameter.tsd-parent-kind-class.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -85px; } +.tsd-kind-function.tsd-has-type-parameter.tsd-parent-kind-enum > .tsd-kind-icon:before { background-position: -170px -85px; } +.tsd-kind-function.tsd-has-type-parameter.tsd-parent-kind-enum.tsd-is-protected > .tsd-kind-icon:before { background-position: -187px -85px; } +.tsd-kind-function.tsd-has-type-parameter.tsd-parent-kind-enum.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -85px; } +.tsd-kind-function.tsd-has-type-parameter.tsd-parent-kind-interface > .tsd-kind-icon:before { background-position: -204px -85px; } +.tsd-kind-function.tsd-has-type-parameter.tsd-parent-kind-interface.tsd-is-inherited > .tsd-kind-icon:before { background-position: -221px -85px; } + +.tsd-kind-method.tsd-has-type-parameter > .tsd-kind-icon:before { background-position: -136px -85px; } +.tsd-kind-method.tsd-has-type-parameter.tsd-is-protected > .tsd-kind-icon:before { background-position: -153px -85px; } +.tsd-kind-method.tsd-has-type-parameter.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -85px; } +.tsd-kind-method.tsd-has-type-parameter.tsd-parent-kind-class > .tsd-kind-icon:before { background-position: -51px -85px; } +.tsd-kind-method.tsd-has-type-parameter.tsd-parent-kind-class.tsd-is-inherited > .tsd-kind-icon:before { background-position: -68px -85px; } +.tsd-kind-method.tsd-has-type-parameter.tsd-parent-kind-class.tsd-is-protected > .tsd-kind-icon:before { background-position: -85px -85px; } +.tsd-kind-method.tsd-has-type-parameter.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited > .tsd-kind-icon:before { background-position: -102px -85px; } +.tsd-kind-method.tsd-has-type-parameter.tsd-parent-kind-class.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -85px; } +.tsd-kind-method.tsd-has-type-parameter.tsd-parent-kind-enum > .tsd-kind-icon:before { background-position: -170px -85px; } +.tsd-kind-method.tsd-has-type-parameter.tsd-parent-kind-enum.tsd-is-protected > .tsd-kind-icon:before { background-position: -187px -85px; } +.tsd-kind-method.tsd-has-type-parameter.tsd-parent-kind-enum.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -85px; } +.tsd-kind-method.tsd-has-type-parameter.tsd-parent-kind-interface > .tsd-kind-icon:before { background-position: -204px -85px; } +.tsd-kind-method.tsd-has-type-parameter.tsd-parent-kind-interface.tsd-is-inherited > .tsd-kind-icon:before { background-position: -221px -85px; } + +.tsd-kind-constructor > .tsd-kind-icon:before { background-position: -136px -102px; } +.tsd-kind-constructor.tsd-is-protected > .tsd-kind-icon:before { background-position: -153px -102px; } +.tsd-kind-constructor.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -102px; } +.tsd-kind-constructor.tsd-parent-kind-class > .tsd-kind-icon:before { background-position: -51px -102px; } +.tsd-kind-constructor.tsd-parent-kind-class.tsd-is-inherited > .tsd-kind-icon:before { background-position: -68px -102px; } +.tsd-kind-constructor.tsd-parent-kind-class.tsd-is-protected > .tsd-kind-icon:before { background-position: -85px -102px; } +.tsd-kind-constructor.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited > .tsd-kind-icon:before { background-position: -102px -102px; } +.tsd-kind-constructor.tsd-parent-kind-class.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -102px; } +.tsd-kind-constructor.tsd-parent-kind-enum > .tsd-kind-icon:before { background-position: -170px -102px; } +.tsd-kind-constructor.tsd-parent-kind-enum.tsd-is-protected > .tsd-kind-icon:before { background-position: -187px -102px; } +.tsd-kind-constructor.tsd-parent-kind-enum.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -102px; } +.tsd-kind-constructor.tsd-parent-kind-interface > .tsd-kind-icon:before { background-position: -204px -102px; } +.tsd-kind-constructor.tsd-parent-kind-interface.tsd-is-inherited > .tsd-kind-icon:before { background-position: -221px -102px; } + +.tsd-kind-constructor-signature > .tsd-kind-icon:before { background-position: -136px -102px; } +.tsd-kind-constructor-signature.tsd-is-protected > .tsd-kind-icon:before { background-position: -153px -102px; } +.tsd-kind-constructor-signature.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -102px; } +.tsd-kind-constructor-signature.tsd-parent-kind-class > .tsd-kind-icon:before { background-position: -51px -102px; } +.tsd-kind-constructor-signature.tsd-parent-kind-class.tsd-is-inherited > .tsd-kind-icon:before { background-position: -68px -102px; } +.tsd-kind-constructor-signature.tsd-parent-kind-class.tsd-is-protected > .tsd-kind-icon:before { background-position: -85px -102px; } +.tsd-kind-constructor-signature.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited > .tsd-kind-icon:before { background-position: -102px -102px; } +.tsd-kind-constructor-signature.tsd-parent-kind-class.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -102px; } +.tsd-kind-constructor-signature.tsd-parent-kind-enum > .tsd-kind-icon:before { background-position: -170px -102px; } +.tsd-kind-constructor-signature.tsd-parent-kind-enum.tsd-is-protected > .tsd-kind-icon:before { background-position: -187px -102px; } +.tsd-kind-constructor-signature.tsd-parent-kind-enum.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -102px; } +.tsd-kind-constructor-signature.tsd-parent-kind-interface > .tsd-kind-icon:before { background-position: -204px -102px; } +.tsd-kind-constructor-signature.tsd-parent-kind-interface.tsd-is-inherited > .tsd-kind-icon:before { background-position: -221px -102px; } + +.tsd-kind-index-signature > .tsd-kind-icon:before { background-position: -136px -119px; } +.tsd-kind-index-signature.tsd-is-protected > .tsd-kind-icon:before { background-position: -153px -119px; } +.tsd-kind-index-signature.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -119px; } +.tsd-kind-index-signature.tsd-parent-kind-class > .tsd-kind-icon:before { background-position: -51px -119px; } +.tsd-kind-index-signature.tsd-parent-kind-class.tsd-is-inherited > .tsd-kind-icon:before { background-position: -68px -119px; } +.tsd-kind-index-signature.tsd-parent-kind-class.tsd-is-protected > .tsd-kind-icon:before { background-position: -85px -119px; } +.tsd-kind-index-signature.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited > .tsd-kind-icon:before { background-position: -102px -119px; } +.tsd-kind-index-signature.tsd-parent-kind-class.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -119px; } +.tsd-kind-index-signature.tsd-parent-kind-enum > .tsd-kind-icon:before { background-position: -170px -119px; } +.tsd-kind-index-signature.tsd-parent-kind-enum.tsd-is-protected > .tsd-kind-icon:before { background-position: -187px -119px; } +.tsd-kind-index-signature.tsd-parent-kind-enum.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -119px; } +.tsd-kind-index-signature.tsd-parent-kind-interface > .tsd-kind-icon:before { background-position: -204px -119px; } +.tsd-kind-index-signature.tsd-parent-kind-interface.tsd-is-inherited > .tsd-kind-icon:before { background-position: -221px -119px; } + +.tsd-kind-event > .tsd-kind-icon:before { background-position: -136px -136px; } +.tsd-kind-event.tsd-is-protected > .tsd-kind-icon:before { background-position: -153px -136px; } +.tsd-kind-event.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -136px; } +.tsd-kind-event.tsd-parent-kind-class > .tsd-kind-icon:before { background-position: -51px -136px; } +.tsd-kind-event.tsd-parent-kind-class.tsd-is-inherited > .tsd-kind-icon:before { background-position: -68px -136px; } +.tsd-kind-event.tsd-parent-kind-class.tsd-is-protected > .tsd-kind-icon:before { background-position: -85px -136px; } +.tsd-kind-event.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited > .tsd-kind-icon:before { background-position: -102px -136px; } +.tsd-kind-event.tsd-parent-kind-class.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -136px; } +.tsd-kind-event.tsd-parent-kind-enum > .tsd-kind-icon:before { background-position: -170px -136px; } +.tsd-kind-event.tsd-parent-kind-enum.tsd-is-protected > .tsd-kind-icon:before { background-position: -187px -136px; } +.tsd-kind-event.tsd-parent-kind-enum.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -136px; } +.tsd-kind-event.tsd-parent-kind-interface > .tsd-kind-icon:before { background-position: -204px -136px; } +.tsd-kind-event.tsd-parent-kind-interface.tsd-is-inherited > .tsd-kind-icon:before { background-position: -221px -136px; } + +.tsd-is-static > .tsd-kind-icon:before { background-position: -136px -153px; } +.tsd-is-static.tsd-is-protected > .tsd-kind-icon:before { background-position: -153px -153px; } +.tsd-is-static.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -153px; } +.tsd-is-static.tsd-parent-kind-class > .tsd-kind-icon:before { background-position: -51px -153px; } +.tsd-is-static.tsd-parent-kind-class.tsd-is-inherited > .tsd-kind-icon:before { background-position: -68px -153px; } +.tsd-is-static.tsd-parent-kind-class.tsd-is-protected > .tsd-kind-icon:before { background-position: -85px -153px; } +.tsd-is-static.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited > .tsd-kind-icon:before { background-position: -102px -153px; } +.tsd-is-static.tsd-parent-kind-class.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -153px; } +.tsd-is-static.tsd-parent-kind-enum > .tsd-kind-icon:before { background-position: -170px -153px; } +.tsd-is-static.tsd-parent-kind-enum.tsd-is-protected > .tsd-kind-icon:before { background-position: -187px -153px; } +.tsd-is-static.tsd-parent-kind-enum.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -153px; } +.tsd-is-static.tsd-parent-kind-interface > .tsd-kind-icon:before { background-position: -204px -153px; } +.tsd-is-static.tsd-parent-kind-interface.tsd-is-inherited > .tsd-kind-icon:before { background-position: -221px -153px; } + +.tsd-is-static.tsd-kind-function > .tsd-kind-icon:before { background-position: -136px -170px; } +.tsd-is-static.tsd-kind-function.tsd-is-protected > .tsd-kind-icon:before { background-position: -153px -170px; } +.tsd-is-static.tsd-kind-function.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -170px; } +.tsd-is-static.tsd-kind-function.tsd-parent-kind-class > .tsd-kind-icon:before { background-position: -51px -170px; } +.tsd-is-static.tsd-kind-function.tsd-parent-kind-class.tsd-is-inherited > .tsd-kind-icon:before { background-position: -68px -170px; } +.tsd-is-static.tsd-kind-function.tsd-parent-kind-class.tsd-is-protected > .tsd-kind-icon:before { background-position: -85px -170px; } +.tsd-is-static.tsd-kind-function.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited > .tsd-kind-icon:before { background-position: -102px -170px; } +.tsd-is-static.tsd-kind-function.tsd-parent-kind-class.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -170px; } +.tsd-is-static.tsd-kind-function.tsd-parent-kind-enum > .tsd-kind-icon:before { background-position: -170px -170px; } +.tsd-is-static.tsd-kind-function.tsd-parent-kind-enum.tsd-is-protected > .tsd-kind-icon:before { background-position: -187px -170px; } +.tsd-is-static.tsd-kind-function.tsd-parent-kind-enum.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -170px; } +.tsd-is-static.tsd-kind-function.tsd-parent-kind-interface > .tsd-kind-icon:before { background-position: -204px -170px; } +.tsd-is-static.tsd-kind-function.tsd-parent-kind-interface.tsd-is-inherited > .tsd-kind-icon:before { background-position: -221px -170px; } + +.tsd-is-static.tsd-kind-method > .tsd-kind-icon:before { background-position: -136px -170px; } +.tsd-is-static.tsd-kind-method.tsd-is-protected > .tsd-kind-icon:before { background-position: -153px -170px; } +.tsd-is-static.tsd-kind-method.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -170px; } +.tsd-is-static.tsd-kind-method.tsd-parent-kind-class > .tsd-kind-icon:before { background-position: -51px -170px; } +.tsd-is-static.tsd-kind-method.tsd-parent-kind-class.tsd-is-inherited > .tsd-kind-icon:before { background-position: -68px -170px; } +.tsd-is-static.tsd-kind-method.tsd-parent-kind-class.tsd-is-protected > .tsd-kind-icon:before { background-position: -85px -170px; } +.tsd-is-static.tsd-kind-method.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited > .tsd-kind-icon:before { background-position: -102px -170px; } +.tsd-is-static.tsd-kind-method.tsd-parent-kind-class.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -170px; } +.tsd-is-static.tsd-kind-method.tsd-parent-kind-enum > .tsd-kind-icon:before { background-position: -170px -170px; } +.tsd-is-static.tsd-kind-method.tsd-parent-kind-enum.tsd-is-protected > .tsd-kind-icon:before { background-position: -187px -170px; } +.tsd-is-static.tsd-kind-method.tsd-parent-kind-enum.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -170px; } +.tsd-is-static.tsd-kind-method.tsd-parent-kind-interface > .tsd-kind-icon:before { background-position: -204px -170px; } +.tsd-is-static.tsd-kind-method.tsd-parent-kind-interface.tsd-is-inherited > .tsd-kind-icon:before { background-position: -221px -170px; } + +.tsd-is-static.tsd-kind-call-signature > .tsd-kind-icon:before { background-position: -136px -170px; } +.tsd-is-static.tsd-kind-call-signature.tsd-is-protected > .tsd-kind-icon:before { background-position: -153px -170px; } +.tsd-is-static.tsd-kind-call-signature.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -170px; } +.tsd-is-static.tsd-kind-call-signature.tsd-parent-kind-class > .tsd-kind-icon:before { background-position: -51px -170px; } +.tsd-is-static.tsd-kind-call-signature.tsd-parent-kind-class.tsd-is-inherited > .tsd-kind-icon:before { background-position: -68px -170px; } +.tsd-is-static.tsd-kind-call-signature.tsd-parent-kind-class.tsd-is-protected > .tsd-kind-icon:before { background-position: -85px -170px; } +.tsd-is-static.tsd-kind-call-signature.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited > .tsd-kind-icon:before { background-position: -102px -170px; } +.tsd-is-static.tsd-kind-call-signature.tsd-parent-kind-class.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -170px; } +.tsd-is-static.tsd-kind-call-signature.tsd-parent-kind-enum > .tsd-kind-icon:before { background-position: -170px -170px; } +.tsd-is-static.tsd-kind-call-signature.tsd-parent-kind-enum.tsd-is-protected > .tsd-kind-icon:before { background-position: -187px -170px; } +.tsd-is-static.tsd-kind-call-signature.tsd-parent-kind-enum.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -170px; } +.tsd-is-static.tsd-kind-call-signature.tsd-parent-kind-interface > .tsd-kind-icon:before { background-position: -204px -170px; } +.tsd-is-static.tsd-kind-call-signature.tsd-parent-kind-interface.tsd-is-inherited > .tsd-kind-icon:before { background-position: -221px -170px; } + +.tsd-is-static.tsd-kind-event > .tsd-kind-icon:before { background-position: -136px -187px; } +.tsd-is-static.tsd-kind-event.tsd-is-protected > .tsd-kind-icon:before { background-position: -153px -187px; } +.tsd-is-static.tsd-kind-event.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -187px; } +.tsd-is-static.tsd-kind-event.tsd-parent-kind-class > .tsd-kind-icon:before { background-position: -51px -187px; } +.tsd-is-static.tsd-kind-event.tsd-parent-kind-class.tsd-is-inherited > .tsd-kind-icon:before { background-position: -68px -187px; } +.tsd-is-static.tsd-kind-event.tsd-parent-kind-class.tsd-is-protected > .tsd-kind-icon:before { background-position: -85px -187px; } +.tsd-is-static.tsd-kind-event.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited > .tsd-kind-icon:before { background-position: -102px -187px; } +.tsd-is-static.tsd-kind-event.tsd-parent-kind-class.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -187px; } +.tsd-is-static.tsd-kind-event.tsd-parent-kind-enum > .tsd-kind-icon:before { background-position: -170px -187px; } +.tsd-is-static.tsd-kind-event.tsd-parent-kind-enum.tsd-is-protected > .tsd-kind-icon:before { background-position: -187px -187px; } +.tsd-is-static.tsd-kind-event.tsd-parent-kind-enum.tsd-is-private > .tsd-kind-icon:before { background-position: -119px -187px; } +.tsd-is-static.tsd-kind-event.tsd-parent-kind-interface > .tsd-kind-icon:before { background-position: -204px -187px; } +.tsd-is-static.tsd-kind-event.tsd-parent-kind-interface.tsd-is-inherited > .tsd-kind-icon:before { background-position: -221px -187px; } + +.no-transition { -webkit-transition: none !important; transition: none !important; } + +@-webkit-keyframes fade-in { + from { opacity: 0; } + to { opacity: 1; } } + +@keyframes fade-in { from { opacity: 0; } + to { opacity: 1; } } +@-webkit-keyframes fade-out { + from { opacity: 1; visibility: visible; } + to { opacity: 0; } } +@keyframes fade-out { from { opacity: 1; visibility: visible; } + to { opacity: 0; } } +@-webkit-keyframes fade-in-delayed { + 0% { opacity: 0; } + 33% { opacity: 0; } + 100% { opacity: 1; } } +@keyframes fade-in-delayed { 0% { opacity: 0; } + 33% { opacity: 0; } + 100% { opacity: 1; } } +@-webkit-keyframes fade-out-delayed { + 0% { opacity: 1; visibility: visible; } + 66% { opacity: 0; } + 100% { opacity: 0; } } +@keyframes fade-out-delayed { 0% { opacity: 1; visibility: visible; } + 66% { opacity: 0; } + 100% { opacity: 0; } } +@-webkit-keyframes shift-to-left { + from { -webkit-transform: translate(0, 0); transform: translate(0, 0); } + to { -webkit-transform: translate(-25%, 0); transform: translate(-25%, 0); } } +@keyframes shift-to-left { from { -webkit-transform: translate(0, 0); transform: translate(0, 0); } + to { -webkit-transform: translate(-25%, 0); transform: translate(-25%, 0); } } +@-webkit-keyframes unshift-to-left { + from { -webkit-transform: translate(-25%, 0); transform: translate(-25%, 0); } + to { -webkit-transform: translate(0, 0); transform: translate(0, 0); } } +@keyframes unshift-to-left { from { -webkit-transform: translate(-25%, 0); transform: translate(-25%, 0); } + to { -webkit-transform: translate(0, 0); transform: translate(0, 0); } } +@-webkit-keyframes pop-in-from-right { + from { -webkit-transform: translate(100%, 0); transform: translate(100%, 0); } + to { -webkit-transform: translate(0, 0); transform: translate(0, 0); } } +@keyframes pop-in-from-right { from { -webkit-transform: translate(100%, 0); transform: translate(100%, 0); } + to { -webkit-transform: translate(0, 0); transform: translate(0, 0); } } +@-webkit-keyframes pop-out-to-right { + from { -webkit-transform: translate(0, 0); transform: translate(0, 0); visibility: visible; } + to { -webkit-transform: translate(100%, 0); transform: translate(100%, 0); } } +@keyframes pop-out-to-right { from { -webkit-transform: translate(0, 0); transform: translate(0, 0); visibility: visible; } + to { -webkit-transform: translate(100%, 0); transform: translate(100%, 0); } } + +.tsd-typography { line-height: 1.333em; } +.tsd-typography ul { list-style: square; padding: 0 0 0 20px; margin: 0; } +.tsd-typography h4, .tsd-typography .tsd-index-panel h3, .tsd-index-panel .tsd-typography h3, .tsd-typography h5, .tsd-typography h6 { font-size: 1em; margin: 0; } +.tsd-typography h5, .tsd-typography h6 { font-weight: normal; } +.tsd-typography p, .tsd-typography ul, .tsd-typography ol { margin: 1em 0; } + +@media (min-width: 901px) and (max-width: 1024px) { html.default .col-content { width: 72%; } + html.default .col-menu { width: 28%; } + html.default .tsd-navigation { padding-left: 10px; } } +@media (max-width: 900px) { html.default .col-content { float: none; width: 100%; } + html.default .col-menu { position: fixed !important; overflow: auto; -webkit-overflow-scrolling: touch; overflow-scrolling: touch; z-index: 1024; top: 0 !important; bottom: 0 !important; left: auto !important; right: 0 !important; width: 100%; padding: 20px 20px 0 0; max-width: 450px; visibility: hidden; background-color: #fff; -webkit-transform: translate(100%, 0); -ms-transform: translate(100%, 0); transform: translate(100%, 0); } + html.default .col-menu > *:last-child { padding-bottom: 20px; } + html.default .overlay { content: ""; display: block; position: fixed; z-index: 1023; top: 0; left: 0; right: 0; bottom: 0; background-color: rgba(0, 0, 0, 0.75); visibility: hidden; } + html.default.to-has-menu .overlay { -webkit-animation: fade-in 0.4s; animation: fade-in 0.4s; } + html.default.to-has-menu header, html.default.to-has-menu footer, html.default.to-has-menu .col-content { -webkit-animation: shift-to-left 0.4s; animation: shift-to-left 0.4s; } + html.default.to-has-menu .col-menu { -webkit-animation: pop-in-from-right 0.4s; animation: pop-in-from-right 0.4s; } + html.default.from-has-menu .overlay { -webkit-animation: fade-out 0.4s; animation: fade-out 0.4s; } + html.default.from-has-menu header, html.default.from-has-menu footer, html.default.from-has-menu .col-content { -webkit-animation: unshift-to-left 0.4s; animation: unshift-to-left 0.4s; } + html.default.from-has-menu .col-menu { -webkit-animation: pop-out-to-right 0.4s; animation: pop-out-to-right 0.4s; } + html.default.has-menu body { overflow: hidden; } + html.default.has-menu .overlay { visibility: visible; } + html.default.has-menu header, html.default.has-menu footer, html.default.has-menu .col-content { -webkit-transform: translate(-25%, 0); -ms-transform: translate(-25%, 0); transform: translate(-25%, 0); } + html.default.has-menu .col-menu { visibility: visible; -webkit-transform: translate(0, 0); -ms-transform: translate(0, 0); transform: translate(0, 0); } } + +.tsd-page-title { padding: 70px 0 20px 0; margin: 0 0 40px 0; background: #fff; box-shadow: 0 0 5px rgba(0, 0, 0, 0.35); } +.tsd-page-title h1 { margin: 0; } + +.tsd-breadcrumb { margin: 0; padding: 0; color: #808080; } +.tsd-breadcrumb a { color: #808080; text-decoration: none; } +.tsd-breadcrumb a:hover { text-decoration: underline; } +.tsd-breadcrumb li { display: inline; } +.tsd-breadcrumb li:after { content: " / "; } + +html.minimal .container-main { padding-bottom: 0; } +html.minimal .content-wrap { padding-left: 340px; } +html.minimal .tsd-navigation { position: fixed !important; float: left; overflow: auto; -webkit-overflow-scrolling: touch; overflow-scrolling: touch; box-sizing: border-box; z-index: 1; top: 60px; bottom: 0; width: 300px; padding: 20px; margin: 0; } +html.minimal .tsd-member .tsd-member { margin-left: 0; } +html.minimal .tsd-page-toolbar { position: fixed; z-index: 2; } +html.minimal #tsd-filter .tsd-filter-group { right: 0; -webkit-transform: none; -ms-transform: none; transform: none; } +html.minimal footer { background-color: transparent; } +html.minimal footer .container { padding: 0; } +html.minimal .tsd-generator { padding: 0; } +@media (max-width: 900px) { html.minimal .tsd-navigation { display: none; } + html.minimal .content-wrap { padding-left: 0; } } + +dl.tsd-comment-tags { overflow: hidden; } +dl.tsd-comment-tags dt { clear: both; float: left; padding: 1px 5px; margin: 0 10px 0 0; border-radius: 4px; border: 1px solid #808080; color: #808080; font-size: 0.8em; font-weight: normal; } +dl.tsd-comment-tags dd { margin: 0 0 10px 0; } +dl.tsd-comment-tags p { margin: 0; } + +.tsd-panel.tsd-comment .lead { font-size: 1.1em; line-height: 1.333em; margin-bottom: 2em; } +.tsd-panel.tsd-comment .lead:last-child { margin-bottom: 0; } + +.toggle-protected .tsd-is-private { display: none; } + +.toggle-public .tsd-is-private, .toggle-public .tsd-is-protected, .toggle-public .tsd-is-private-protected { display: none; } + +.toggle-inherited .tsd-is-inherited { display: none; } + +.toggle-only-exported .tsd-is-not-exported { display: none; } + +.toggle-externals .tsd-is-external { display: none; } + +#tsd-filter { position: relative; display: inline-block; height: 40px; vertical-align: bottom; } +.no-filter #tsd-filter { display: none; } +#tsd-filter .tsd-filter-group { display: inline-block; height: 40px; vertical-align: bottom; white-space: nowrap; } +#tsd-filter input { display: none; } +@media (max-width: 900px) { #tsd-filter .tsd-filter-group { display: block; position: absolute; top: 40px; right: 20px; height: auto; background-color: #fff; visibility: hidden; -webkit-transform: translate(50%, 0); -ms-transform: translate(50%, 0); transform: translate(50%, 0); box-shadow: 0 0 4px rgba(0, 0, 0, 0.25); } + .has-options #tsd-filter .tsd-filter-group { visibility: visible; } + .to-has-options #tsd-filter .tsd-filter-group { -webkit-animation: fade-in 0.2s; animation: fade-in 0.2s; } + .from-has-options #tsd-filter .tsd-filter-group { -webkit-animation: fade-out 0.2s; animation: fade-out 0.2s; } + #tsd-filter label, #tsd-filter .tsd-select { display: block; padding-right: 20px; } } + +footer { background-color: #fff; } +footer.with-border-bottom { border-bottom: 1px solid #eee; margin-left: 20px } +footer .tsd-legend-group { font-size: 0; } +footer .tsd-legend { display: inline-block; width: 25%; padding: 0; font-size: 16px; list-style: none; line-height: 1.333em; vertical-align: top; } +@media (max-width: 900px) { footer .tsd-legend { width: 50%; } } + +.tsd-hierarchy { list-style: square; padding: 0 0 0 20px; margin: 0; } +.tsd-hierarchy .target { font-weight: bold; } + +.tsd-index-panel .tsd-index-content { margin-bottom: -30px !important; } +.tsd-index-panel .tsd-index-section { margin-bottom: 30px !important; } +.tsd-index-panel h3 { margin: 0 -20px 10px -20px; padding: 0 20px 10px 20px; border-bottom: 1px solid #eee; } +.tsd-index-panel ul.tsd-index-list { -webkit-column-count: 3; -moz-column-count: 3; -ms-column-count: 3; column-count: 3; -webkit-column-gap: 20px; -moz-column-gap: 20px; -ms-column-gap: 20px; column-gap: 20px; padding: 0; list-style: none; line-height: 1.333em; } +@media (max-width: 900px) { .tsd-index-panel ul.tsd-index-list { -webkit-column-count: 1; -moz-column-count: 1; -ms-column-count: 1; column-count: 1; } } +@media (min-width: 901px) and (max-width: 1024px) { .tsd-index-panel ul.tsd-index-list { -webkit-column-count: 2; -moz-column-count: 2; -ms-column-count: 2; column-count: 2; } } +.tsd-index-panel ul.tsd-index-list li { -webkit-column-break-inside: avoid; -moz-column-break-inside: avoid; -ms-column-break-inside: avoid; -o-column-break-inside: avoid; column-break-inside: avoid; -webkit-page-break-inside: avoid; -moz-page-break-inside: avoid; -ms-page-break-inside: avoid; -o-page-break-inside: avoid; page-break-inside: avoid; } +.tsd-index-panel a, .tsd-index-panel .tsd-parent-kind-module a { color: #9600ff; } +.tsd-index-panel .tsd-parent-kind-interface a { color: #7da01f; } +.tsd-index-panel .tsd-parent-kind-enum a { color: #cc9900; } +.tsd-index-panel .tsd-parent-kind-class a { color: #4da6ff; } +.tsd-index-panel .tsd-kind-module a { color: #9600ff; } +.tsd-index-panel .tsd-kind-interface a { color: #7da01f; } +.tsd-index-panel .tsd-kind-enum a { color: #cc9900; } +.tsd-index-panel .tsd-kind-class a { color: #4da6ff; } +.tsd-index-panel .tsd-is-private a { color: #808080; } + +.tsd-flag { display: inline-block; padding: 1px 5px; border-radius: 4px; color: #fff; background-color: #808080; text-indent: 0; font-size: 14px; font-weight: normal; } + +.tsd-anchor { position: absolute; top: -100px; } + +.tsd-member { position: relative; } +.tsd-member .tsd-anchor + h3 { margin-top: 0; margin-bottom: 0; border-bottom: none; } + +.tsd-navigation { padding: 0 0 0 40px; } +.tsd-navigation a { display: block; padding-top: 2px; padding-bottom: 2px; border-left: 2px solid transparent; color: #222; text-decoration: none; -webkit-transition: border-left-color 0.1s; transition: border-left-color 0.1s; } +.tsd-navigation a:hover { text-decoration: underline; } +.tsd-navigation ul { margin: 0; padding: 0; list-style: none; } +.tsd-navigation li { padding: 0; } + +.tsd-navigation.primary { padding-bottom: 40px; } +.tsd-navigation.primary a { display: block; padding-top: 6px; padding-bottom: 6px; } +.tsd-navigation.primary ul li a { padding-left: 5px; } +.tsd-navigation.primary ul li li a { padding-left: 25px; } +.tsd-navigation.primary ul li li li a { padding-left: 45px; } +.tsd-navigation.primary ul li li li li a { padding-left: 65px; } +.tsd-navigation.primary ul li li li li li a { padding-left: 85px; } +.tsd-navigation.primary ul li li li li li li a { padding-left: 105px; } +.tsd-navigation.primary > ul { border-bottom: 1px solid #eee; } +.tsd-navigation.primary li { border-top: 1px solid #eee; } +.tsd-navigation.primary li.current > a { font-weight: bold; } +.tsd-navigation.primary li.label span { display: block; padding: 20px 0 6px 5px; color: #808080; } +.tsd-navigation.primary li.globals + li > span, .tsd-navigation.primary li.globals + li > a { padding-top: 20px; } + +.tsd-navigation.secondary ul { -webkit-transition: opacity 0.2s; transition: opacity 0.2s; } +.tsd-navigation.secondary ul li a { padding-left: 25px; } +.tsd-navigation.secondary ul li li a { padding-left: 45px; } +.tsd-navigation.secondary ul li li li a { padding-left: 65px; } +.tsd-navigation.secondary ul li li li li a { padding-left: 85px; } +.tsd-navigation.secondary ul li li li li li a { padding-left: 105px; } +.tsd-navigation.secondary ul li li li li li li a { padding-left: 125px; } +.tsd-navigation.secondary ul.current a { border-left-color: #eee; } +.tsd-navigation.secondary li.focus > a, .tsd-navigation.secondary ul.current li.focus > a { border-left-color: #000; } +.tsd-navigation.secondary li.current { margin-top: 20px; margin-bottom: 20px; border-left-color: #eee; } +.tsd-navigation.secondary li.current > a { font-weight: bold; } + +@media (min-width: 901px) { .menu-sticky-wrap { position: static; } + .no-csspositionsticky .menu-sticky-wrap.sticky { position: fixed; } + .no-csspositionsticky .menu-sticky-wrap.sticky-current { position: fixed; } + .no-csspositionsticky .menu-sticky-wrap.sticky-current ul.before-current, .no-csspositionsticky .menu-sticky-wrap.sticky-current ul.after-current { opacity: 0; } + .no-csspositionsticky .menu-sticky-wrap.sticky-bottom { position: absolute; top: auto !important; left: auto !important; bottom: 0; right: 0; } + .csspositionsticky .menu-sticky-wrap.sticky { position: -webkit-sticky; position: sticky; } + .csspositionsticky .menu-sticky-wrap.sticky-current { position: -webkit-sticky; position: sticky; } } + +.tsd-panel { margin: 20px 0; padding: 20px; background-color: #fff; box-shadow: 0 0 4px rgba(0, 0, 0, 0.25); } +.tsd-panel:empty { display: none; } +.tsd-panel > h1, .tsd-panel > h2, .tsd-panel > h3 { margin: 1.5em -20px 10px -20px; padding: 0 20px 10px 20px; border-bottom: 1px solid #eee; } +.tsd-panel > h1.tsd-before-signature, .tsd-panel > h2.tsd-before-signature, .tsd-panel > h3.tsd-before-signature { margin-bottom: 0; border-bottom: 0; } +.tsd-panel table { display: block; width: 100%; overflow: auto; margin-top: 10px; word-break: normal; word-break: keep-all; } +.tsd-panel table th { font-weight: bold; } +.tsd-panel table th, .tsd-panel table td { padding: 6px 13px; border: 1px solid #ddd; } +.tsd-panel table tr { background-color: #fff; border-top: 1px solid #ccc; } +.tsd-panel table tr:nth-child(2n) { background-color: #f8f8f8; } + +.tsd-panel-group { margin: 30px 0; } +.tsd-panel-group > h1, .tsd-panel-group > h2, .tsd-panel-group > h3 { padding-left: 20px; padding-right: 20px; } + +#tsd-search { -webkit-transition: background-color 0.2s; transition: background-color 0.2s; } +#tsd-search .title { position: relative; z-index: 2; } +#tsd-search .field { position: absolute; left: 0; top: 0; right: 40px; height: 40px; } +#tsd-search .field input { box-sizing: border-box; position: relative; top: -50px; z-index: 1; width: 100%; padding: 0 10px; opacity: 0; outline: 0; border: 0; background: transparent; color: #222; } +#tsd-search .field label { position: absolute; overflow: hidden; right: -40px; } +#tsd-search .field input, #tsd-search .title { -webkit-transition: opacity 0.2s; transition: opacity 0.2s; } +#tsd-search .results { position: absolute; visibility: hidden; top: 40px; width: 100%; margin: 0; padding: 0; list-style: none; box-shadow: 0 0 4px rgba(0, 0, 0, 0.25); } +#tsd-search .results li { padding: 0 10px; background-color: #fdfdfd; } +#tsd-search .results li:nth-child(even) { background-color: #fff; } +#tsd-search .results li.state { display: none; } +#tsd-search .results li.current, #tsd-search .results li:hover { background-color: #eee; } +#tsd-search .results a { display: block; } +#tsd-search .results a:before { top: 10px; } +#tsd-search .results span.parent { color: #808080; font-weight: normal; } +#tsd-search.has-focus { background-color: #eee; } +#tsd-search.has-focus .field input { top: 0; opacity: 1; } +#tsd-search.has-focus .title { z-index: 0; opacity: 0; } +#tsd-search.has-focus .results { visibility: visible; } +#tsd-search.loading .results li.state.loading { display: block; } +#tsd-search.failure .results li.state.failure { display: block; } + +.tsd-signature { margin: 0 0 1em 0; padding: 10px; border: 1px solid #eee; font-family: Menlo, Monaco, Consolas, "Courier New", monospace; font-size: 14px; } +.tsd-signature.tsd-kind-icon { padding-left: 30px; } +.tsd-signature.tsd-kind-icon:before { top: 10px; left: 10px; } +.tsd-panel > .tsd-signature { margin-left: -20px; margin-right: -20px; border-width: 1px 0; } +.tsd-panel > .tsd-signature.tsd-kind-icon { padding-left: 40px; } +.tsd-panel > .tsd-signature.tsd-kind-icon:before { left: 20px; } + +.tsd-signature-symbol { color: #808080; font-weight: normal; } + +.tsd-signature-type { font-style: italic; font-weight: normal; } + +.tsd-signatures { padding: 0; margin: 0 0 1em 0; border: 1px solid #eee; } +.tsd-signatures .tsd-signature { margin: 0; border-width: 1px 0 0 0; -webkit-transition: background-color 0.1s; transition: background-color 0.1s; } +.tsd-signatures .tsd-signature:first-child { border-top-width: 0; } +.tsd-signatures .tsd-signature.current { background-color: #eee; } +.tsd-signatures.active > .tsd-signature { cursor: pointer; } +.tsd-panel > .tsd-signatures { margin-left: -20px; margin-right: -20px; border-width: 1px 0; } +.tsd-panel > .tsd-signatures .tsd-signature.tsd-kind-icon { padding-left: 40px; } +.tsd-panel > .tsd-signatures .tsd-signature.tsd-kind-icon:before { left: 20px; } +.tsd-panel > a.anchor + .tsd-signatures { border-top-width: 0; margin-top: -20px; } + +ul.tsd-descriptions { position: relative; overflow: hidden; -webkit-transition: height 0.3s; transition: height 0.3s; padding: 0; list-style: none; } +ul.tsd-descriptions.active > .tsd-description { display: none; } +ul.tsd-descriptions.active > .tsd-description.current { display: block; } +ul.tsd-descriptions.active > .tsd-description.fade-in { -webkit-animation: fade-in-delayed 0.3s; animation: fade-in-delayed 0.3s; } +ul.tsd-descriptions.active > .tsd-description.fade-out { -webkit-animation: fade-out-delayed 0.3s; animation: fade-out-delayed 0.3s; position: absolute; display: block; top: 0; left: 0; right: 0; opacity: 0; visibility: hidden; } +ul.tsd-descriptions h4, ul.tsd-descriptions .tsd-index-panel h3, .tsd-index-panel ul.tsd-descriptions h3 { font-size: 16px; margin: 1em 0 0.5em 0; } + +ul.tsd-parameters, ul.tsd-type-parameters { list-style: square; margin: 0; padding-left: 20px; } +ul.tsd-parameters > li.tsd-parameter-siganture, ul.tsd-type-parameters > li.tsd-parameter-siganture { list-style: none; margin-left: -20px; } +ul.tsd-parameters h5, ul.tsd-type-parameters h5 { font-size: 16px; margin: 1em 0 0.5em 0; } +ul.tsd-parameters .tsd-comment, ul.tsd-type-parameters .tsd-comment { margin-top: -0.5em; } + +.tsd-sources { font-size: 14px; color: #808080; margin: 0 0 1em 0; } +.tsd-sources a { color: #808080; text-decoration: underline; } +.tsd-sources ul, .tsd-sources p { margin: 0 !important; } +.tsd-sources ul { list-style: none; padding: 0; } + +.tsd-page-toolbar { position: absolute; z-index: 1; top: 0; left: 0; width: 100%; height: 40px; color: #333; background: #fff; border-bottom: 1px solid #eee; } +.tsd-page-toolbar a { color: #333; text-decoration: none; } +.tsd-page-toolbar a.title { font-weight: bold; } +.tsd-page-toolbar a.title:hover { text-decoration: underline; } +.tsd-page-toolbar .table-wrap { display: table; width: 100%; height: 40px; } +.tsd-page-toolbar .table-cell { display: table-cell; position: relative; white-space: nowrap; line-height: 40px; } +.tsd-page-toolbar .table-cell:first-child { width: 100%; } + +.tsd-widget:before, .tsd-select .tsd-select-label:before, .tsd-select .tsd-select-list li:before { content: ""; display: inline-block; width: 40px; height: 40px; margin: 0 -8px 0 0; background-image: url(); background-repeat: no-repeat; text-indent: -1024px; vertical-align: bottom; } +@media (-webkit-min-device-pixel-ratio: 1.5), (min-device-pixel-ratio: 1.5), (min-resolution: 144dpi) { .tsd-widget:before, .tsd-select .tsd-select-label:before, .tsd-select .tsd-select-list li:before { background-image: url(); background-size: 320px 40px; } } + +.tsd-widget { display: inline-block; overflow: hidden; opacity: 0.6; height: 40px; -webkit-transition: opacity 0.1s, background-color 0.2s; transition: opacity 0.1s, background-color 0.2s; vertical-align: bottom; cursor: pointer; } +.tsd-widget:hover { opacity: 0.8; } +.tsd-widget.active { opacity: 1; background-color: #eee; } +.tsd-widget.no-caption { width: 40px; } +.tsd-widget.no-caption:before { margin: 0; } +.tsd-widget.search:before { background-position: 0 0; } +.tsd-widget.menu:before { background-position: -40px 0; } +.tsd-widget.options:before { background-position: -80px 0; } +.tsd-widget.options, .tsd-widget.menu { display: none; } +@media (max-width: 900px) { .tsd-widget.options, .tsd-widget.menu { display: inline-block; } } +input[type=checkbox] + .tsd-widget:before { background-position: -120px 0; } +input[type=checkbox]:checked + .tsd-widget:before { background-position: -160px 0; } + +.tsd-select { position: relative; display: inline-block; height: 40px; -webkit-transition: opacity 0.1s, background-color 0.2s; transition: opacity 0.1s, background-color 0.2s; vertical-align: bottom; cursor: pointer; } +.tsd-select .tsd-select-label { opacity: 0.6; -webkit-transition: opacity 0.2s; transition: opacity 0.2s; } +.tsd-select .tsd-select-label:before { background-position: -240px 0; } +.tsd-select.active .tsd-select-label { opacity: 0.8; } +.tsd-select.active .tsd-select-list { visibility: visible; opacity: 1; -webkit-transition-delay: 0s; transition-delay: 0s; } +.tsd-select .tsd-select-list { position: absolute; visibility: hidden; top: 40px; left: 0; margin: 0; padding: 0; opacity: 0; list-style: none; box-shadow: 0 0 4px rgba(0, 0, 0, 0.25); -webkit-transition: visibility 0s 0.2s, opacity 0.2s; transition: visibility 0s 0.2s, opacity 0.2s; } +.tsd-select .tsd-select-list li { padding: 0 20px 0 0; background-color: #fdfdfd; } +.tsd-select .tsd-select-list li:before { background-position: 40px 0; } +.tsd-select .tsd-select-list li:nth-child(even) { background-color: #fff; } +.tsd-select .tsd-select-list li:hover { background-color: #eee; } +.tsd-select .tsd-select-list li.selected:before { background-position: -200px 0; } +@media (max-width: 900px) { .tsd-select .tsd-select-list { top: 0; left: auto; right: 100%; margin-right: -5px; } + .tsd-select .tsd-select-label:before { background-position: -280px 0; } } diff --git a/docs/css/github-highlight.css b/docs/css/github-highlight.css new file mode 100644 index 0000000..52b1887 --- /dev/null +++ b/docs/css/github-highlight.css @@ -0,0 +1,224 @@ +pre { + border: none !important; + background-color: #fff !important; +} +code { + color: inherit !important; + background-color: #fff; +} +pre, code { + white-space: pre !important; +} +.highlight table pre { margin: 0; } +.highlight .cm { + color: #999988; + font-style: italic; +} +.highlight .cp { + color: #999999; + font-weight: bold; +} +.highlight .c1 { + color: #999988; + font-style: italic; +} +.highlight .cs { + color: #999999; + font-weight: bold; + font-style: italic; +} +.highlight .c, .highlight .cd { + color: #999988; + font-style: italic; +} +.highlight .err { + color: #a61717; + background-color: #e3d2d2; +} +.highlight .gd { + color: #000000; + background-color: #ffdddd; +} +.highlight .ge { + color: #000000; + font-style: italic; +} +.highlight .gr { + color: #aa0000; +} +.highlight .gh { + color: #999999; +} +.highlight .gi { + color: #000000; + background-color: #ddffdd; +} +.highlight .go { + color: #888888; +} +.highlight .gp { + color: #555555; +} +.highlight .gs { + font-weight: bold; +} +.highlight .gu { + color: #aaaaaa; +} +.highlight .gt { + color: #aa0000; +} +.highlight .kc { + color: #000000; + font-weight: bold; +} +.highlight .kd { + color: #000000; + font-weight: bold; +} +.highlight .kn { + color: #000000; + font-weight: bold; +} +.highlight .kp { + color: #000000; + font-weight: bold; +} +.highlight .kr { + color: #000000; + font-weight: bold; +} +.highlight .kt { + color: #445588; + font-weight: bold; +} +.highlight .k, .highlight .kv { + color: #000000; + font-weight: bold; +} +.highlight .mf { + color: #009999; +} +.highlight .mh { + color: #009999; +} +.highlight .il { + color: #009999; +} +.highlight .mi { + color: #009999; +} +.highlight .mo { + color: #009999; +} +.highlight .m, .highlight .mb, .highlight .mx { + color: #009999; +} +.highlight .sb { + color: #d14; +} +.highlight .sc { + color: #d14; +} +.highlight .sd { + color: #d14; +} +.highlight .s2 { + color: #d14; +} +.highlight .se { + color: #d14; +} +.highlight .sh { + color: #d14; +} +.highlight .si { + color: #d14; +} +.highlight .sx { + color: #d14; +} +.highlight .sr { + color: #009926; +} +.highlight .s1 { + color: #d14; +} +.highlight .ss { + color: #990073; +} +.highlight .s { + color: #d14; +} +.highlight .na { + color: #008080; +} +.highlight .bp { + color: #999999; +} +.highlight .nb { + color: #0086B3; +} +.highlight .nc { + color: #445588; + font-weight: bold; +} +.highlight .no { + color: #008080; +} +.highlight .nd { + color: #3c5d5d; + font-weight: bold; +} +.highlight .ni { + color: #800080; +} +.highlight .ne { + color: #990000; + font-weight: bold; +} +.highlight .nf { + color: #990000; + font-weight: bold; +} +.highlight .nl { + color: #990000; + font-weight: bold; +} +.highlight .nn { + color: #555555; +} +.highlight .nt { + color: #000080; +} +.highlight .vc { + color: #008080; +} +.highlight .vg { + color: #008080; +} +.highlight .vi { + color: #008080; +} +.highlight .nv { + color: #008080; +} +.highlight .ow { + color: #000000; + font-weight: bold; +} +.highlight .o { + color: #000000; + font-weight: bold; +} +.highlight .w { + color: #bbbbbb; +} +.highlight { + background-color: #fff; + padding-bottom: 0px; +} +.highlighter-rouge { + border: 1px solid #ccc; + margin-bottom: 1em; +} diff --git a/docs/css/main.css b/docs/css/main.css new file mode 100644 index 0000000..05fd446 --- /dev/null +++ b/docs/css/main.css @@ -0,0 +1,276 @@ +/* General CSS */ +@import url(https://fonts.googleapis.com/css?family=Open+Sans); + +body { + font-family: "Open Sans", Helvetica, Arial, sans-serif; +} + +img { + max-width: 90%; +} + +/* Custom CSS for home page */ +.hero-spacer { + margin-top: 50px; +} + +.hero-feature { + margin-bottom: 30px; +} + +.blog-content-wrap { + margin-top: 20px; + padding-top: 40px; +} + +/* Custom CSS for the docs */ + +.edit-links { + color: #cccccc; +} + +/* Prevent in-page links from scrolling under top nav */ +h2::before, h3::before { + display: block; + content: " "; + margin-top: -80px; + height: 80px; + visibility: hidden; +} + +/* Custom CSS for header and footer */ +.site-header +{ + width: 100%; + padding: 24px 0px; +} + +img.logo +{ + height: 28px; + margin-left: -18px; +} + +.icon > svg +{ + display: inline-block; + width: 28px; + height: 28px; +} + +.icon > svg path +{ + fill: #333333; +} + + +.thumb-home { + height: 340px; +} + +.img-home { + height: 150px !important; + margin: 10px; +} + +.tableauIcon { + margin-left: 18px; +} + +.jumbotron { + background-color: #fafafa; + border: 1px solid #e8e8e8; +} + +#community-jumbo { + background-color: #F0F8FF; +} + +footer { + margin: 30px 0; + text-align: center; +} + +.footer-hr { + width: 100%; + position: relative; +} + +table { + border: 1px solid #c2c2c2; + border-collapse: collapse; + margin: 1em 0px; +} + +th +{ + font-weight: bold; + border: 1px solid #c2c2c2; + text-align: left; + padding: .3em; + vertical-align: top; + background-color: #fafafa; +} + +td +{ + border: 1px solid #c2c2c2; + text-align: left; + padding: .3em; + vertical-align: top; +} + +/* So the scroll bar width doesn't cause the page to jump */ +html { + overflow-y: scroll; +} + +.label { + margin-right: 3px; +} + +/* to get right navbar icons to respect collapse */ +@media (max-width: 992px) { + .navbar-header { + float: none; + } + .navbar-left,.navbar-right { + float: none !important; + } + .navbar-toggle { + display: block; + } + .navbar-collapse { + border-top: 1px solid transparent; + box-shadow: inset 0 1px 0 rgba(255,255,255,0.1); + } + .navbar-fixed-top { + top: 0; + border-width: 0 0 1px; + } + .navbar-collapse.collapse { + display: none!important; + } + .navbar-nav { + float: none!important; + margin-top: 7.5px; + } + .navbar-nav>li { + float: none; + } + .navbar-nav>li>a { + padding-top: 10px; + padding-bottom: 10px; + } + .collapse.in{ + display:block !important; + } +} + + +/* Custom css for news section */ +.blog-content { + margin-bottom: 70px; +} + +.blogul { + padding: 0px; +} + +.blogul h1 { + margin-top: 40px; +} + +.blog-content > h4 { + margin-top: 20px; +} + +.blog-content > ol { + margin-bottom: 20px; +} + + +/* Community connectors */ +.thumbnail { + background-color: #fff; + border: 1px solid #ccc; + margin: 12px; +} + +.thumbnail h2 { + border-left: 2px solid #337ab7; + text-decoration: none; + font-size: 20px; + padding: 6px; + margin-left: 10px; +} + +.thumbnail h2 a { + text-decoration: none; + color: #333333; +} + +.well { + background-color: #ffffff; + margin-bottom: 40px; +} + +.tsd-navigation { + padding: 10px !important; +} + +/* Media queries for responsive design */ +#grid[data-columns]::before { + content: '3 .column.size-1of3'; +} + +.column { float: left; } +.size-1of3 { width: 33.333%; } + + +@media screen and (max-width: 767px) { + #grid[data-columns]::before { + content: '1 .column.size-1of1'; + } + + /* Docs Menu*/ + .docs-menu, .tsd-navigation { + top: 20px; + position: relative; + } + +} +@media screen and (min-width: 769px) { + #grid[data-columns]::before { + content: '3 .column.size-1of3'; + } + + /* Docs Menu*/ + .docs-menu, .tsd-navigation { + position: fixed; + overflow: auto; + top: 90px; + max-height: 90%; + max-width: 250px; + } + + .content { + position: relative; + margin: 40px 0px 0px 275px; + max-width: 1000px; + } + + /* API Reference */ + + .ref-content { + margin: 40px 0px 0px 275px; + max-width: 1000px; + } +} + +.column { float: left; } +.size-1of1 { width: 100%; } +.size-1of2 { width: 50%; } +.size-1of3 { width: 33.333%; } + + diff --git a/docs/api-ref.md b/docs/docs/api-ref.md similarity index 90% rename from docs/api-ref.md rename to docs/docs/api-ref.md index 7210507..c5dc0e5 100644 --- a/docs/api-ref.md +++ b/docs/docs/api-ref.md @@ -1,4 +1,12 @@ -# Workbooks +--- +title: API reference +layout: docs +--- + +* TOC +{:toc} + +## Workbooks ```python class Workbook(filename): ``` @@ -24,17 +32,17 @@ Saves any changes to the workbook to a new file specified by the `new_file` para `self.datasources:` Returns a list of Datasource objects found in the workbook `self.filename:` Returns the filename of the workbook -#Datasources +## Datasources ```python class Datasource(dsxml, filename=None) ``` -# Connections +## Connections ```python class Connection(connxml) ``` -# Fields +## Fields ```python class Workbook(column_xml=None, metadata_xml=None) ``` diff --git a/docs/contributing.md b/docs/docs/contributing.md similarity index 86% rename from docs/contributing.md rename to docs/docs/contributing.md index ae41c2e..ef54847 100644 --- a/docs/contributing.md +++ b/docs/docs/contributing.md @@ -1,8 +1,11 @@ -# Contributing +--- +title: Contributing +layout: docs +--- We welcome contributions to this project! -There are many ways to contribute! We welcome all contributions including: +Contribution can include, but are not limited to, any of the following: * File an Issue * Request a Feature @@ -21,7 +24,7 @@ If you are submiting a bug report, please provide as much information as you can files to assist in the repro. **Be sure to scrub the files of any potentially sensitive information. Issues are public.** For a feature request, please try to describe the scenario you are trying to accomplish that requires the feature. This will help us understand -the limitations that you are running into, and provide us with a use case to know if we've satisfied your request. Psuedo code that describes how you would like things to work is also very helpful. +the limitations that you are running into, and provide us with a use case to know if we've satisfied your request. ## Fixes, Implementations, and Documentation @@ -29,4 +32,5 @@ For all other things, please submit a PR that includes the fix, documentation, o creating a PR can be found in the [github documentation](https://help.github.com/articles/creating-a-pull-request/) If the feature is complex or has multiple solutions that could be equally appropriate approaches, it would be helpful to file an issue to discuss the -design trade-offs of each solution before implementing, to allow us to collectively arrive at the best solution, which most likely exists in the middle somewhere. +design trade-offs of each solution before implementing, to allow us to collectively arrive at the best solution, which most likely exists in the middle +somewhere. diff --git a/docs/devguide.md b/docs/docs/dev-guide.md similarity index 96% rename from docs/devguide.md rename to docs/docs/dev-guide.md index aef40ca..4aa78ca 100644 --- a/docs/devguide.md +++ b/docs/docs/dev-guide.md @@ -1,5 +1,7 @@ -Developer Guide ---------------- +--- +title: Developer Guide +layout: docs +--- ### Making your first patch diff --git a/docs/docs/index.md b/docs/docs/index.md new file mode 100644 index 0000000..89dcba4 --- /dev/null +++ b/docs/docs/index.md @@ -0,0 +1,88 @@ +--- +title: Get Started +layout: docs +--- + +To use this SDK, you must have Python installed. You can use either 2.7.X or 3.3 and later. + +* TOC +{:toc} + +### Installing the latest stable version (preferred) + +```shell +pip install tableaudocumentapi +``` + +### Installing From Source + +Download the `.zip` file that contains the SDK. Unzip the file and then run the following command: + +```shell +pip install -e +``` + +### Installing the Development Version from Git + +*Only do this if you know you want the development version, no guarantee that we won't break APIs during development* + +```shell +pip install git+https://github.com/tableau/document-api-python.git@development +``` + +If you go this route, but want to switch back to the non-development version, you need to run the following command before installing the stable version: + +```shell +pip uninstall tableaudocumentapi +``` + +## Basics +The following example shows the basic syntax for using the Document API to update a workbook: + +```python +from tableaudocumentapi import Workbook + +sourceWB = Workbook('WorkbookToUpdate.twb') + +sourceWB.datasources[0].connections[0].server = "MY-NEW-SERVER" +sourceWB.datasources[0].connections[0].dbname = "NEW-DATABASE" +sourceWB.datasources[0].connections[0].username = "benl" + +sourceWB.save() +``` + +With Data Integration in Tableau 10, a data source can have multiple connections. To access the connections simply index them like you would datasources. + +```python +from tableaudocumentapi import Workbook + +sourceWB = Workbook('WorkbookToUpdate.twb') + +sourceWB.datasources[0].connections[0].server = "MY-NEW-SERVER" +sourceWB.datasources[0].connections[0].dbname = "NEW-DATABASE" +sourceWB.datasources[0].connections[0].username = "benl" + +sourceWB.datasources[0].connections[1].server = "MY-NEW-SERVER" +sourceWB.datasources[0].connections[1].dbname = "NEW-DATABASE" +sourceWB.datasources[0].connections[1].username = "benl" + + +sourceWB.save() +``` + + +**Notes** + +- Import the `Workbook` object from the `tableaudocumentapi` module. +- To open a workbook, instantiate a `Workbook` object and pass the file name as the first argument. +- The `Workbook` object exposes a list of `datasources` in the workbook +- Each data source object has a `connection` object that supports a `server`, `dbname`, and `username` property. +- Save changes to the workbook by calling the `save` or `save_as` method. + +
+

Note: This call was added in version Tableau 10.3, XSDv2.5

+
+ +### Samples + +The downloadable package contains several example scripts that show more detailed usage of the Document API. diff --git a/docs/index.md b/docs/index.md index 45ccfbb..7cb8e3d 100644 --- a/docs/index.md +++ b/docs/index.md @@ -1,110 +1,38 @@ -Tableau Document API ----------------------- - +--- +layout: home +--- +
+

Document API Python

+

Programmatically update your Tableau workbooks and data sources.

+

This site will get you up and running with the Python version of the Tableau Document API. The Document API, including the samples and documentation, are all open source.

+
+ Get Started   + Download +
+ +# Document API Overview This repo contains Python source and example files for the Tableau Document API. We're just getting started and have plans to expand what you find here. Help us by submitting feedback, issues, and pull requests! -The Document API provides a supported way to programmatically make updates to Tableau workbook and data source files. If you've been making changes to these file types by directly updating the XML--that is, by XML hacking--this SDK is for you :) - - -Features include: - -- Support for 9.X, and 10.X workbook and data source files - - Including TDSX and TWBX files -- Getting connection information from data sources and workbooks - - Server Name - - Username - - Database Name - - Authentication Type - - Connection Type -- Updating connection information in workbooks and data sources - - Server Name - - Username - - Database Name -- Getting Field information from data sources and workbooks - - Get all fields in a data source - - Get all fields in use by certain sheets in a workbook - -We don't yet support creating files from scratch, adding extracts into workbooks or data sources, or updating field information - - -### Getting Started -To use this SDK, you must have Python installed. You can use either 2.7.X or 3.3 and later. - -#### Installing the latest stable version (preferred) - -```shell -pip install tableaudocumentapi -``` - -#### Installing From Source - -Download the `.zip` file that contains the SDK. Unzip the file and then run the following command: - -```shell -pip install -e -``` - -#### Installing the Development Version from Git - -*Only do this if you know you want the development version, no guarantee that we won't break APIs during development* - -```shell -pip install git+https://github.com/tableau/document-api-python.git@development -``` - -If you go this route, but want to switch back to the non-development version, you need to run the following command before installing the stable version: - -```shell -pip uninstall tableaudocumentapi -``` - -###Basics -The following example shows the basic syntax for using the Document API to update a workbook: - -```python -from tableaudocumentapi import Workbook - -sourceWB = Workbook('WorkbookToUpdate.twb') - -sourceWB.datasources[0].connections[0].server = "MY-NEW-SERVER" -sourceWB.datasources[0].connections[0].dbname = "NEW-DATABASE" -sourceWB.datasources[0].connections[0].username = "benl" - -sourceWB.save() -``` - -With Data Integration in Tableau 10, a data source can have multiple connections. To access the connections simply index them like you would datasources. - -```python -from tableaudocumentapi import Workbook - -sourceWB = Workbook('WorkbookToUpdate.twb') - -sourceWB.datasources[0].connections[0].server = "MY-NEW-SERVER" -sourceWB.datasources[0].connections[0].dbname = "NEW-DATABASE" -sourceWB.datasources[0].connections[0].username = "benl" - -sourceWB.datasources[0].connections[1].server = "MY-NEW-SERVER" -sourceWB.datasources[0].connections[1].dbname = "NEW-DATABASE" -sourceWB.datasources[0].connections[1].username = "benl" - - -sourceWB.save() -``` - - -**Notes** - -- Import the `Workbook` object from the `tableaudocumentapi` module. -- To open a workbook, instantiate a `Workbook` object and pass the file name as the first argument. -- The `Workbook` object exposes a list of `datasources` in the workbook -- Each data source object has a `connection` object that supports a `server`, `dbname`, and `username` property. -- Save changes to the workbook by calling the `save` or `save_as` method. - -!!! tip "Introduced In v 2.4" - This call was added in version Tableau 10.3, XSDv2.5 - -###[Examples](samples) - -The downloadable package contains several example scripts that show more detailed usage of the Document API. +The Document API provides a supported way to programmatically make updates to Tableau workbook and data source files. If you've been making changes to these file types by directly updating the XML--that is, by XML hacking--this SDK is for you. + + +>Features include: +> +>- Support for 9.X, and 10.X workbook and data source files +> - Including TDSX and TWBX files +>- Getting connection information from data sources and workbooks +> - Server Name +> - Username +> - Database Name +> - Authentication Type +> - Connection Type +>- Updating connection information in workbooks and data sources +> - Server Name +> - Username +> - Database Name +>- Getting Field information from data sources and workbooks +> - Get all fields in a data source +> - Get all fields in use by certain sheets in a workbook + +We don't yet support creating files from scratch, adding extracts into workbooks or data sources, or updating field information. diff --git a/docs/user-guide.md b/docs/user-guide.md deleted file mode 100644 index c7bb53a..0000000 --- a/docs/user-guide.md +++ /dev/null @@ -1,6 +0,0 @@ -# Concepts - -Imagine a world without kittens. What a world that would be. -This is why I am going to discuss our document object model. You see -it all starts with Workbooks, Datasources, and Connections. There are -also Fields. From 2366f86b41a3a99c5c4177129c0df0c2425a2c85 Mon Sep 17 00:00:00 2001 From: T8y8 Date: Wed, 26 Oct 2016 15:37:34 -0700 Subject: [PATCH 15/16] This should build --- .travis.yml | 35 +++++++++++++---------------------- docs/docs/api-ref.md | 4 +++- mkdocs.yml | 23 ----------------------- 3 files changed, 16 insertions(+), 46 deletions(-) delete mode 100644 mkdocs.yml diff --git a/.travis.yml b/.travis.yml index c4341b7..7e48103 100644 --- a/.travis.yml +++ b/.travis.yml @@ -1,27 +1,18 @@ language: python - -env: - global: - secure: !!binary | - elBwQ2pFZ0wyR3FCQ0xwT2F5LzdCL2JCTHBNSk02ZGJzcDk3UHBGUkNFMmZBWVovTlpFSHB6YVJP - Ykg2NGRMcGlPUDZpQyt3aER4ekpxRTA4aFlXRXc1N0MvcGNjaWZhWldJMGxoM2NsT3VMWDV1Sm90 - QVpmZi9vWjREM2N0NzhhNXZxQ1lTTVU2ZXVmSTZDbXM5d3g2L3Rha0c3MGxBQzJvcVM0R0lFUXV4 - anNTZFlBcnlsd2xNOHJybkxNUHVlL3NBc2dPcUZnTW1HNzNkUWt2SDBKWmw5VUxuem54dUp2bHRN - R3Q0b1RVa2gxN1dGMWlyUFlDNldqWWRpQXNQeWF1K0NCL3gyY25EdGJFOEVpczl1RHZWZWxFZFc3 - K0dkWVF4b1NjME4zWEMvRi9IR0w5L2xKVDF4RVR6c3hDSk4vUERWMW9QVUJwUjJ0RE4yM3RZdWNt - SXhnbzExTWJRUjdoTU5YeVNSNGlseUJKWURxODNpbGNjZWhMdFBTNGhSL3JTcUduU0svV2hPeW1M - d1BRcitZK1luWG8zRTVlUHhzYmJIcmxyejZjbjF6bnlUeXBJL0Rmc2cwWXZtTkV6ekZlalVFbnJI - aTR5eDBuN1dZUElwMGd4N3d4ZUpFUmdWNm5XQW9Pb0w0UVNIU1lLWG54Rjk0NkZ5Y3J0QmlVc3hS - NFlNRHlMY1lKRGUxTWZTblVWUFhtSEYzUmJlVmxIQzFKWHZ0RittUFFGSXdlSjZHaWRMTE1rSlhC - VkVjNFBMbnk1VkNOMHRPWUhpdlU4MDYwSGg4K0N0Rk9zWnh1VUM3OWRhNDRvQ2huSnBiM0VtUHUw - OWVxRjRuVGk0R3NGdUNZVFFubi9aMHBCdVZCRUpVNWtoTTJzZGNyWEJiWlFKeTRQVWppYkNSc1U9 +sudo: required +cache: pip install: - pip install -e . - pip install pycodestyle -- pip install mkdocs -- pip install pymdown-extensions -- chmod +x ./deploy-docs.sh +- sudo apt-get update +- sudo apt-get install ruby +- gem install bundle +- gem install jeykll +- bundle install +# - pip install mkdocs +# - pip install pymdown-extensions +# - chmod +x ./deploy-docs.sh matrix: include: @@ -43,7 +34,7 @@ script: - (cd "samples/replicate-workbook" && python replicate_workbook.py) - (cd "samples/list-tds-info" && python list_tds_info.py) - (cd "samples/show-fields" && python show_fields.py) -- mkdocs build -s --clean +# - bundle exec jekyll build -after_success: - - bash deploy-docs.sh +# after_success: +# - bash deploy-docs.sh diff --git a/docs/docs/api-ref.md b/docs/docs/api-ref.md index c5dc0e5..b0f950c 100644 --- a/docs/docs/api-ref.md +++ b/docs/docs/api-ref.md @@ -1,11 +1,13 @@ --- title: API reference -layout: docs +layout: garbage --- * TOC {:toc} +[BROKEN LINK](lol.md) + ## Workbooks ```python class Workbook(filename): diff --git a/mkdocs.yml b/mkdocs.yml deleted file mode 100644 index 7548c83..0000000 --- a/mkdocs.yml +++ /dev/null @@ -1,23 +0,0 @@ -# Site information -site_name: Tableau Document API (Python) -repo_url: https://github.com/tableau/document-api-python/ -copyright: © Tableau Software - -# Pages and Layout -pages: - - Intro: index.md - - Documentation: - - 'User Guide': user-guide.md - - 'API Ref': api-ref.md - - Contributing: - - contributing.md - - devguide.md - -theme: readthedocs -extra_css: - - css/extra.css - -# Markdown Stuff -markdown_extensions: - - admonition # Customize for 'Version Added' ? - - pymdownx.extra # To use superfences so code and lists work nicely From ddedf0f922e8a7fa2834c0de8ddb13058a208526 Mon Sep 17 00:00:00 2001 From: Jared Dominguez Date: Thu, 27 Oct 2016 11:48:50 -0700 Subject: [PATCH 16/16] Formatting for dev guide and api ref --- docs/docs/api-ref.md | 40 +++++++++++++++------------ docs/docs/dev-guide.md | 63 +++++++++++++++++++++++++----------------- 2 files changed, 60 insertions(+), 43 deletions(-) diff --git a/docs/docs/api-ref.md b/docs/docs/api-ref.md index b0f950c..3819b70 100644 --- a/docs/docs/api-ref.md +++ b/docs/docs/api-ref.md @@ -1,38 +1,42 @@ --- title: API reference -layout: garbage +layout: docs --- * TOC {:toc} -[BROKEN LINK](lol.md) - ## Workbooks ```python class Workbook(filename): ``` -The Workbook class represents a tableau workbook. It may be either a TWB or TWBX, and the library will handle packaging and unpackaging automatically. +The Workbook class represents a tableau workbook. It may be either a TWB or TWBX, and the library will handle packaging and unpackaging automatically. + +**Params:** + +`filename` takes a string representing the path to the workbook file. + +**Raises:** + +`TableauVersionNotSupportedException` if the workbook is not a supported version. +`TableauInvalidFileException` if the file is not a valid tableau workbook file. + +**Methods:** + +`Workbook.save(self):` +Saves any changes to the workbook to the existing file. -**Params:** -`filename` takes a string representing the path to the workbook file. +`Workbook.save_as(self, new_filename):` +Saves any changes to the workbook to a new file specified by the `new_file` parameter. -**Raises:** -`TableauVersionNotSupportedException` if the workbook is not a supported version -`TableauInvalidFileException` if the file is not a valid tableau workbook file +**Properities:** -**Methods:** -`Workbook.save(self):` -Saves any changes to the workbook to the existing file +`self.worksheets:` Returns a list of worksheets found in the workbook. -`Workbook.save_as(self, new_filename):` -Saves any changes to the workbook to a new file specified by the `new_file` parameter +`self.datasources:` Returns a list of Datasource objects found in the workbook. -**Properities:** -`self.worksheets:` Returns a list of worksheets found in the workbook -`self.datasources:` Returns a list of Datasource objects found in the workbook -`self.filename:` Returns the filename of the workbook +`self.filename:` Returns the filename of the workbook. ## Datasources ```python diff --git a/docs/docs/dev-guide.md b/docs/docs/dev-guide.md index 4aa78ca..cd6bfee 100644 --- a/docs/docs/dev-guide.md +++ b/docs/docs/dev-guide.md @@ -5,39 +5,52 @@ layout: docs ### Making your first patch -0. Make sure you've signed the CLA +1. Make sure you've signed the CLA + 1. Clone the repo - ```shell - git clone http://github.com/tableau/document-api-python - ``` + ```shell + git clone http://github.com/tableau/document-api-python + ``` + +1. Run the tests to make sure everything is peachy + + ```shell + python setup.py test + ``` + +1. Set up the feature, fix, or documentation branch. + + It is recommended to use the format [issue#]-[type]-[description] (e.g. 13-fix-connection-bug) + + ```shell + git checkout -b 13-feature-new-stuff + ``` + +1. Code and Commit! + + Here's a quick checklist for ensuring a good diff: + + - The diff touches the minimal amount of files possible while still fufilling the purpose of the diff + - The diff uses Unix line endings + - The diff adheres to our PEP8 style guides. If you've cloned the repo you can run `pycodestyle .` + +1. Add Tests + +1. Update Documentation -2. Run the tests to make sure everything is peachy + Our documentation is written in markdown and built with [Mkdocs](http://www.mkdocs.org). More information on how to update and build the docs can be found [here](#updating-documentation) - ```shell - python setup.py test - ``` +1. Run the tests again and make sure they pass! -3. Set up the feature, fix, or documentation branch. -It is recommended to use the format [issue#]-[type]-[description] (e.g. 13-fix-connection-bug) +1. Submit to your fork - ```shell - git checkout -b 13-feature-new-stuff - ``` +1. Submit a PR -4. Code and Commit! - Here's a quick checklist for ensuring a good diff: - - The diff touches the minimal amount of files possible while still fufilling the purpose of the diff - - The diff uses Unix line endings - - The diff adheres to our PEP8 style guides. If you've cloned the repo you can run `pycodestyle .` -5. Add Tests -6. Update Documentation - Our documentation is written in markdown and built with [Mkdocs](http://www.mkdocs.org). More information on how to update and build the docs can be found [here](#updating-documentation) -7. Run the tests again and make sure they pass! -8. Submit to your fork -9. Submit a PR -10. Wait for a review, and address any feedback. +1. Wait for a review, and address any feedback. +