From cdbe29c70bcd16f41beeccaf259d3bb92999acea Mon Sep 17 00:00:00 2001 From: Pino de Candia Date: Mon, 8 Jan 2018 16:34:38 +0000 Subject: [PATCH] Copy Designate-Dashboard to get started on Tatu dashboard. --- .gitignore | 33 ++ .gitreview | 4 + .zuul.yaml | 4 + CONTRIBUTING.rst | 17 + HACKING.rst | 4 + LICENSE | 175 ++++++ README.rst | 79 +++ babel-django.cfg | 5 + designatedashboard/__init__.py | 19 + designatedashboard/api/__init__.py | 1 + designatedashboard/api/designate.py | 171 ++++++ designatedashboard/api/rest/__init__.py | 16 + designatedashboard/api/rest/passthrough.py | 119 ++++ designatedashboard/dashboards/__init__.py | 0 .../dashboards/project/__init__.py | 0 .../project/dns_domains/__init__.py | 0 .../dashboards/project/dns_domains/forms.py | 549 ++++++++++++++++++ .../dashboards/project/dns_domains/panel.py | 26 + .../dashboards/project/dns_domains/tables.py | 224 +++++++ .../templates/dns_domains/_create_domain.html | 38 ++ .../templates/dns_domains/_create_record.html | 37 ++ .../templates/dns_domains/_domain_detail.html | 32 + .../templates/dns_domains/_record_detail.html | 36 ++ .../templates/dns_domains/_update_domain.html | 36 ++ .../templates/dns_domains/_update_record.html | 9 + .../templates/dns_domains/create_domain.html | 11 + .../templates/dns_domains/create_record.html | 11 + .../templates/dns_domains/domain_detail.html | 11 + .../templates/dns_domains/index.html | 11 + .../dns_domains/prefix_field_style.html | 4 + .../dns_domains/prefix_html_widget.html | 2 + .../templates/dns_domains/record_detail.html | 11 + .../templates/dns_domains/records.html | 32 + .../templates/dns_domains/update_domain.html | 11 + .../templates/dns_domains/update_record.html | 11 + .../dashboards/project/dns_domains/urls.py | 52 ++ .../dashboards/project/dns_domains/utils.py | 20 + .../dashboards/project/dns_domains/views.py | 243 ++++++++ .../dashboards/project/ngdns/__init__.py | 1 + .../project/ngdns/reverse_dns/__init__.py | 0 .../project/ngdns/reverse_dns/panel.py | 25 + .../project/ngdns/reverse_dns/urls.py | 22 + .../project/ngdns/reverse_dns/views.py | 19 + .../project/ngdns/zones/__init__.py | 0 .../dashboards/project/ngdns/zones/panel.py | 25 + .../dashboards/project/ngdns/zones/urls.py | 22 + .../dashboards/project/ngdns/zones/views.py | 19 + .../enabled/_1710_project_dns_panel_group.py | 20 + .../enabled/_1720_project_dns_panel.py | 36 ++ .../enabled/_1721_dns_zones_panel.py | 40 ++ .../enabled/_1722_dns_reversedns_panel.py | 38 ++ designatedashboard/enabled/__init__.py | 0 designatedashboard/exceptions.py | 29 + .../locale/cs/LC_MESSAGES/django.mo | Bin 0 -> 6965 bytes .../locale/cs/LC_MESSAGES/django.po | 348 +++++++++++ .../locale/de/LC_MESSAGES/django.mo | Bin 0 -> 7264 bytes .../locale/de/LC_MESSAGES/django.po | 351 +++++++++++ .../locale/en_GB/LC_MESSAGES/django.mo | Bin 0 -> 6783 bytes .../locale/en_GB/LC_MESSAGES/django.po | 351 +++++++++++ .../locale/es/LC_MESSAGES/django.mo | Bin 0 -> 7000 bytes .../locale/es/LC_MESSAGES/django.po | 342 +++++++++++ .../locale/fr/LC_MESSAGES/django.mo | Bin 0 -> 7398 bytes .../locale/fr/LC_MESSAGES/django.po | 362 ++++++++++++ .../locale/id/LC_MESSAGES/django.mo | Bin 0 -> 7183 bytes .../locale/id/LC_MESSAGES/django.po | 350 +++++++++++ .../locale/ja/LC_MESSAGES/django.mo | Bin 0 -> 7759 bytes .../locale/ja/LC_MESSAGES/django.po | 347 +++++++++++ .../locale/ko_KR/LC_MESSAGES/django.mo | Bin 0 -> 7499 bytes .../locale/ko_KR/LC_MESSAGES/django.po | 350 +++++++++++ .../locale/pt_BR/LC_MESSAGES/django.mo | Bin 0 -> 7192 bytes .../locale/pt_BR/LC_MESSAGES/django.po | 352 +++++++++++ .../locale/ru/LC_MESSAGES/django.mo | Bin 0 -> 8541 bytes .../locale/ru/LC_MESSAGES/django.po | 361 ++++++++++++ .../locale/tr_TR/LC_MESSAGES/django.mo | Bin 0 -> 6928 bytes .../locale/tr_TR/LC_MESSAGES/django.po | 360 ++++++++++++ .../locale/zh_CN/LC_MESSAGES/django.mo | Bin 0 -> 6537 bytes .../locale/zh_CN/LC_MESSAGES/django.po | 353 +++++++++++ .../designatedashboard.module.js | 78 +++ .../designatedashboard.scss | 0 .../actions/actions.module.js | 66 +++ .../actions/set.service.js | 170 ++++++ .../actions/unset.service.js | 139 +++++ .../os-designate-floatingip/api.service.js | 121 ++++ .../details/details.module.js | 66 +++ .../details/drawer.html | 10 + .../details/overview.controller.js | 46 ++ .../details/overview.html | 11 + .../os-designate-floatingip.module.js | 156 +++++ .../actions/actions.module.js | 79 +++ .../actions/common-forms.service.js | 165 ++++++ .../actions/create.service.js | 132 +++++ .../actions/delete.service.js | 182 ++++++ .../actions/update.service.js | 159 +++++ .../os-designate-recordset/api.service.js | 136 +++++ .../details/details.module.js | 109 ++++ .../details/drawer.html | 10 + .../details/overview.controller.js | 46 ++ .../details/overview.html | 45 ++ .../details/zone-recordsets.controller.js | 39 ++ .../details/zone-recordsets.html | 8 + .../os-designate-recordset.module.js | 240 ++++++++ .../actions/actions.module.js | 76 +++ .../actions/actions.module.spec.js | 49 ++ .../actions/common-forms.service.js | 185 ++++++ .../os-designate-zone/actions/create.html | 1 + .../actions/create.service.js | 126 ++++ .../actions/delete.service.js | 169 ++++++ .../actions/update.service.js | 147 +++++ .../os-designate-zone/api.service.js | 152 +++++ .../details/details.module.js | 61 ++ .../os-designate-zone/details/drawer.html | 10 + .../details/overview.controller.js | 55 ++ .../os-designate-zone/details/overview.html | 58 ++ .../os-designate-zone.module.js | 204 +++++++ .../resources/resources.module.js | 36 ++ .../resources/util.service.js | 107 ++++ .../designatedashboard/reverse_dns.html | 4 + .../static/designatedashboard/zones.html | 4 + designatedashboard/tests/.secret_key_store | 1 + designatedashboard/tests/__init__.py | 0 designatedashboard/tests/base.py | 100 ++++ designatedashboard/tests/settings.py | 83 +++ .../tests/test_designatedashboard.py | 406 +++++++++++++ .../tests/test_ptr_record_form.py | 85 +++ .../tests/test_spf_record_form.py | 70 +++ .../tests/test_sshfp_record_form.py | 92 +++ doc/source/conf.py | 82 +++ doc/source/contributor/index.rst | 5 + doc/source/index.rst | 21 + doc/source/install/index.rst | 12 + doc/source/readme.rst | 1 + doc/source/user/index.rst | 7 + karma.conf.js | 141 +++++ manage.py | 23 + openstack-common.conf | 6 + package.json | 26 + releasenotes/notes/.placeholder | 0 releasenotes/source/conf.py | 281 +++++++++ releasenotes/source/index.rst | 20 + .../locale/cs/LC_MESSAGES/releasenotes.po | 45 ++ .../locale/de/LC_MESSAGES/releasenotes.po | 46 ++ .../locale/en_GB/LC_MESSAGES/releasenotes.po | 45 ++ .../locale/fr/LC_MESSAGES/releasenotes.po | 21 + .../locale/id/LC_MESSAGES/releasenotes.po | 45 ++ .../locale/ko_KR/LC_MESSAGES/releasenotes.po | 47 ++ .../locale/pt_BR/LC_MESSAGES/releasenotes.po | 46 ++ .../locale/ru/LC_MESSAGES/releasenotes.po | 37 ++ .../locale/zh_CN/LC_MESSAGES/releasenotes.po | 42 ++ releasenotes/source/mitaka.rst | 6 + releasenotes/source/newton.rst | 6 + releasenotes/source/ocata.rst | 6 + releasenotes/source/pike.rst | 6 + releasenotes/source/unreleased.rst | 5 + requirements.txt | 7 + setup.cfg | 46 ++ setup.py | 29 + test | Bin 0 -> 2048 bytes test-requirements.txt | 26 + test-shim.js | 110 ++++ tox.ini | 48 ++ 160 files changed, 12375 insertions(+) create mode 100644 .gitignore create mode 100644 .gitreview create mode 100644 .zuul.yaml create mode 100644 CONTRIBUTING.rst create mode 100644 HACKING.rst create mode 100644 LICENSE create mode 100644 README.rst create mode 100644 babel-django.cfg create mode 100644 designatedashboard/__init__.py create mode 100644 designatedashboard/api/__init__.py create mode 100644 designatedashboard/api/designate.py create mode 100644 designatedashboard/api/rest/__init__.py create mode 100644 designatedashboard/api/rest/passthrough.py create mode 100644 designatedashboard/dashboards/__init__.py create mode 100644 designatedashboard/dashboards/project/__init__.py create mode 100644 designatedashboard/dashboards/project/dns_domains/__init__.py create mode 100644 designatedashboard/dashboards/project/dns_domains/forms.py create mode 100644 designatedashboard/dashboards/project/dns_domains/panel.py create mode 100644 designatedashboard/dashboards/project/dns_domains/tables.py create mode 100644 designatedashboard/dashboards/project/dns_domains/templates/dns_domains/_create_domain.html create mode 100644 designatedashboard/dashboards/project/dns_domains/templates/dns_domains/_create_record.html create mode 100644 designatedashboard/dashboards/project/dns_domains/templates/dns_domains/_domain_detail.html create mode 100644 designatedashboard/dashboards/project/dns_domains/templates/dns_domains/_record_detail.html create mode 100644 designatedashboard/dashboards/project/dns_domains/templates/dns_domains/_update_domain.html create mode 100644 designatedashboard/dashboards/project/dns_domains/templates/dns_domains/_update_record.html create mode 100644 designatedashboard/dashboards/project/dns_domains/templates/dns_domains/create_domain.html create mode 100644 designatedashboard/dashboards/project/dns_domains/templates/dns_domains/create_record.html create mode 100644 designatedashboard/dashboards/project/dns_domains/templates/dns_domains/domain_detail.html create mode 100644 designatedashboard/dashboards/project/dns_domains/templates/dns_domains/index.html create mode 100644 designatedashboard/dashboards/project/dns_domains/templates/dns_domains/prefix_field_style.html create mode 100644 designatedashboard/dashboards/project/dns_domains/templates/dns_domains/prefix_html_widget.html create mode 100644 designatedashboard/dashboards/project/dns_domains/templates/dns_domains/record_detail.html create mode 100644 designatedashboard/dashboards/project/dns_domains/templates/dns_domains/records.html create mode 100644 designatedashboard/dashboards/project/dns_domains/templates/dns_domains/update_domain.html create mode 100644 designatedashboard/dashboards/project/dns_domains/templates/dns_domains/update_record.html create mode 100644 designatedashboard/dashboards/project/dns_domains/urls.py create mode 100644 designatedashboard/dashboards/project/dns_domains/utils.py create mode 100644 designatedashboard/dashboards/project/dns_domains/views.py create mode 100644 designatedashboard/dashboards/project/ngdns/__init__.py create mode 100644 designatedashboard/dashboards/project/ngdns/reverse_dns/__init__.py create mode 100644 designatedashboard/dashboards/project/ngdns/reverse_dns/panel.py create mode 100644 designatedashboard/dashboards/project/ngdns/reverse_dns/urls.py create mode 100644 designatedashboard/dashboards/project/ngdns/reverse_dns/views.py create mode 100644 designatedashboard/dashboards/project/ngdns/zones/__init__.py create mode 100644 designatedashboard/dashboards/project/ngdns/zones/panel.py create mode 100644 designatedashboard/dashboards/project/ngdns/zones/urls.py create mode 100644 designatedashboard/dashboards/project/ngdns/zones/views.py create mode 100644 designatedashboard/enabled/_1710_project_dns_panel_group.py create mode 100644 designatedashboard/enabled/_1720_project_dns_panel.py create mode 100644 designatedashboard/enabled/_1721_dns_zones_panel.py create mode 100644 designatedashboard/enabled/_1722_dns_reversedns_panel.py create mode 100644 designatedashboard/enabled/__init__.py create mode 100644 designatedashboard/exceptions.py create mode 100644 designatedashboard/locale/cs/LC_MESSAGES/django.mo create mode 100644 designatedashboard/locale/cs/LC_MESSAGES/django.po create mode 100644 designatedashboard/locale/de/LC_MESSAGES/django.mo create mode 100644 designatedashboard/locale/de/LC_MESSAGES/django.po create mode 100644 designatedashboard/locale/en_GB/LC_MESSAGES/django.mo create mode 100644 designatedashboard/locale/en_GB/LC_MESSAGES/django.po create mode 100644 designatedashboard/locale/es/LC_MESSAGES/django.mo create mode 100644 designatedashboard/locale/es/LC_MESSAGES/django.po create mode 100644 designatedashboard/locale/fr/LC_MESSAGES/django.mo create mode 100644 designatedashboard/locale/fr/LC_MESSAGES/django.po create mode 100644 designatedashboard/locale/id/LC_MESSAGES/django.mo create mode 100644 designatedashboard/locale/id/LC_MESSAGES/django.po create mode 100644 designatedashboard/locale/ja/LC_MESSAGES/django.mo create mode 100644 designatedashboard/locale/ja/LC_MESSAGES/django.po create mode 100644 designatedashboard/locale/ko_KR/LC_MESSAGES/django.mo create mode 100644 designatedashboard/locale/ko_KR/LC_MESSAGES/django.po create mode 100644 designatedashboard/locale/pt_BR/LC_MESSAGES/django.mo create mode 100644 designatedashboard/locale/pt_BR/LC_MESSAGES/django.po create mode 100644 designatedashboard/locale/ru/LC_MESSAGES/django.mo create mode 100644 designatedashboard/locale/ru/LC_MESSAGES/django.po create mode 100644 designatedashboard/locale/tr_TR/LC_MESSAGES/django.mo create mode 100644 designatedashboard/locale/tr_TR/LC_MESSAGES/django.po create mode 100644 designatedashboard/locale/zh_CN/LC_MESSAGES/django.mo create mode 100644 designatedashboard/locale/zh_CN/LC_MESSAGES/django.po create mode 100644 designatedashboard/static/designatedashboard/designatedashboard.module.js create mode 100644 designatedashboard/static/designatedashboard/designatedashboard.scss create mode 100644 designatedashboard/static/designatedashboard/resources/os-designate-floatingip/actions/actions.module.js create mode 100644 designatedashboard/static/designatedashboard/resources/os-designate-floatingip/actions/set.service.js create mode 100644 designatedashboard/static/designatedashboard/resources/os-designate-floatingip/actions/unset.service.js create mode 100644 designatedashboard/static/designatedashboard/resources/os-designate-floatingip/api.service.js create mode 100644 designatedashboard/static/designatedashboard/resources/os-designate-floatingip/details/details.module.js create mode 100644 designatedashboard/static/designatedashboard/resources/os-designate-floatingip/details/drawer.html create mode 100644 designatedashboard/static/designatedashboard/resources/os-designate-floatingip/details/overview.controller.js create mode 100644 designatedashboard/static/designatedashboard/resources/os-designate-floatingip/details/overview.html create mode 100644 designatedashboard/static/designatedashboard/resources/os-designate-floatingip/os-designate-floatingip.module.js create mode 100644 designatedashboard/static/designatedashboard/resources/os-designate-recordset/actions/actions.module.js create mode 100644 designatedashboard/static/designatedashboard/resources/os-designate-recordset/actions/common-forms.service.js create mode 100644 designatedashboard/static/designatedashboard/resources/os-designate-recordset/actions/create.service.js create mode 100644 designatedashboard/static/designatedashboard/resources/os-designate-recordset/actions/delete.service.js create mode 100644 designatedashboard/static/designatedashboard/resources/os-designate-recordset/actions/update.service.js create mode 100644 designatedashboard/static/designatedashboard/resources/os-designate-recordset/api.service.js create mode 100644 designatedashboard/static/designatedashboard/resources/os-designate-recordset/details/details.module.js create mode 100644 designatedashboard/static/designatedashboard/resources/os-designate-recordset/details/drawer.html create mode 100644 designatedashboard/static/designatedashboard/resources/os-designate-recordset/details/overview.controller.js create mode 100644 designatedashboard/static/designatedashboard/resources/os-designate-recordset/details/overview.html create mode 100644 designatedashboard/static/designatedashboard/resources/os-designate-recordset/details/zone-recordsets.controller.js create mode 100644 designatedashboard/static/designatedashboard/resources/os-designate-recordset/details/zone-recordsets.html create mode 100644 designatedashboard/static/designatedashboard/resources/os-designate-recordset/os-designate-recordset.module.js create mode 100644 designatedashboard/static/designatedashboard/resources/os-designate-zone/actions/actions.module.js create mode 100644 designatedashboard/static/designatedashboard/resources/os-designate-zone/actions/actions.module.spec.js create mode 100644 designatedashboard/static/designatedashboard/resources/os-designate-zone/actions/common-forms.service.js create mode 100644 designatedashboard/static/designatedashboard/resources/os-designate-zone/actions/create.html create mode 100644 designatedashboard/static/designatedashboard/resources/os-designate-zone/actions/create.service.js create mode 100644 designatedashboard/static/designatedashboard/resources/os-designate-zone/actions/delete.service.js create mode 100644 designatedashboard/static/designatedashboard/resources/os-designate-zone/actions/update.service.js create mode 100644 designatedashboard/static/designatedashboard/resources/os-designate-zone/api.service.js create mode 100644 designatedashboard/static/designatedashboard/resources/os-designate-zone/details/details.module.js create mode 100644 designatedashboard/static/designatedashboard/resources/os-designate-zone/details/drawer.html create mode 100644 designatedashboard/static/designatedashboard/resources/os-designate-zone/details/overview.controller.js create mode 100644 designatedashboard/static/designatedashboard/resources/os-designate-zone/details/overview.html create mode 100644 designatedashboard/static/designatedashboard/resources/os-designate-zone/os-designate-zone.module.js create mode 100644 designatedashboard/static/designatedashboard/resources/resources.module.js create mode 100644 designatedashboard/static/designatedashboard/resources/util.service.js create mode 100644 designatedashboard/static/designatedashboard/reverse_dns.html create mode 100644 designatedashboard/static/designatedashboard/zones.html create mode 100644 designatedashboard/tests/.secret_key_store create mode 100644 designatedashboard/tests/__init__.py create mode 100644 designatedashboard/tests/base.py create mode 100644 designatedashboard/tests/settings.py create mode 100644 designatedashboard/tests/test_designatedashboard.py create mode 100644 designatedashboard/tests/test_ptr_record_form.py create mode 100644 designatedashboard/tests/test_spf_record_form.py create mode 100644 designatedashboard/tests/test_sshfp_record_form.py create mode 100755 doc/source/conf.py create mode 100644 doc/source/contributor/index.rst create mode 100644 doc/source/index.rst create mode 100644 doc/source/install/index.rst create mode 100644 doc/source/readme.rst create mode 100644 doc/source/user/index.rst create mode 100644 karma.conf.js create mode 100755 manage.py create mode 100644 openstack-common.conf create mode 100644 package.json create mode 100644 releasenotes/notes/.placeholder create mode 100644 releasenotes/source/conf.py create mode 100644 releasenotes/source/index.rst create mode 100644 releasenotes/source/locale/cs/LC_MESSAGES/releasenotes.po create mode 100644 releasenotes/source/locale/de/LC_MESSAGES/releasenotes.po create mode 100644 releasenotes/source/locale/en_GB/LC_MESSAGES/releasenotes.po create mode 100644 releasenotes/source/locale/fr/LC_MESSAGES/releasenotes.po create mode 100644 releasenotes/source/locale/id/LC_MESSAGES/releasenotes.po create mode 100644 releasenotes/source/locale/ko_KR/LC_MESSAGES/releasenotes.po create mode 100644 releasenotes/source/locale/pt_BR/LC_MESSAGES/releasenotes.po create mode 100644 releasenotes/source/locale/ru/LC_MESSAGES/releasenotes.po create mode 100644 releasenotes/source/locale/zh_CN/LC_MESSAGES/releasenotes.po create mode 100644 releasenotes/source/mitaka.rst create mode 100644 releasenotes/source/newton.rst create mode 100644 releasenotes/source/ocata.rst create mode 100644 releasenotes/source/pike.rst create mode 100644 releasenotes/source/unreleased.rst create mode 100644 requirements.txt create mode 100644 setup.cfg create mode 100644 setup.py create mode 100644 test create mode 100644 test-requirements.txt create mode 100644 test-shim.js create mode 100644 tox.ini diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..aea1652 --- /dev/null +++ b/.gitignore @@ -0,0 +1,33 @@ +*.pyc +*.dat +TAGS +*.egg-info +*.egg +.eggs +build +.coverage +.coverage.* +.tox +cover +venv +.venv +*.sublime-workspace +*.sqlite +*.sqlite3 +var/* +AUTHORS +ChangeLog +doc/source/api/* +doc/build/* +dist +*.orig +*.DS_Store +*.idea +.testrepository/* +functionaltests/tempest.log +functionaltests/.testrepository/ +*.ipynb +/.ipynb_checkpoints/* +releasenotes/build +node_modules +npm-debug.log diff --git a/.gitreview b/.gitreview new file mode 100644 index 0000000..0607a92 --- /dev/null +++ b/.gitreview @@ -0,0 +1,4 @@ +[gerrit] +host=review.openstack.org +port=29418 +project=openstack/designate-dashboard.git diff --git a/.zuul.yaml b/.zuul.yaml new file mode 100644 index 0000000..d76eafb --- /dev/null +++ b/.zuul.yaml @@ -0,0 +1,4 @@ +- project: + name: openstack/designate-dashboard + templates: + - designate-devstack-jobs diff --git a/CONTRIBUTING.rst b/CONTRIBUTING.rst new file mode 100644 index 0000000..51e5d63 --- /dev/null +++ b/CONTRIBUTING.rst @@ -0,0 +1,17 @@ +If you would like to contribute to the development of OpenStack, +you must follow the steps in the "If you're a developer, start here" +section of this page: + + http://wiki.openstack.org/HowToContribute + +Once those steps have been completed, changes to OpenStack +should be submitted for review via the Gerrit tool, following +the workflow documented at: + + http://wiki.openstack.org/GerritWorkflow + +Pull requests submitted through GitHub will be ignored. + +Bugs should be filed on Launchpad, not GitHub: + + https://bugs.launchpad.net/designatedashboard \ No newline at end of file diff --git a/HACKING.rst b/HACKING.rst new file mode 100644 index 0000000..e0ddef5 --- /dev/null +++ b/HACKING.rst @@ -0,0 +1,4 @@ +designatedashboard Style Commandments +=============================================== + +Read the OpenStack Style Commandments http://docs.openstack.org/developer/hacking/ \ No newline at end of file diff --git a/LICENSE b/LICENSE new file mode 100644 index 0000000..67db858 --- /dev/null +++ b/LICENSE @@ -0,0 +1,175 @@ + + Apache License + Version 2.0, January 2004 + http://www.apache.org/licenses/ + + TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION + + 1. Definitions. + + "License" shall mean the terms and conditions for use, reproduction, + and distribution as defined by Sections 1 through 9 of this document. + + "Licensor" shall mean the copyright owner or entity authorized by + the copyright owner that is granting the License. + + "Legal Entity" shall mean the union of the acting entity and all + other entities that control, are controlled by, or are under common + control with that entity. For the purposes of this definition, + "control" means (i) the power, direct or indirect, to cause the + direction or management of such entity, whether by contract or + otherwise, or (ii) ownership of fifty percent (50%) or more of the + outstanding shares, or (iii) beneficial ownership of such entity. + + "You" (or "Your") shall mean an individual or Legal Entity + exercising permissions granted by this License. + + "Source" form shall mean the preferred form for making modifications, + including but not limited to software source code, documentation + source, and configuration files. + + "Object" form shall mean any form resulting from mechanical + transformation or translation of a Source form, including but + not limited to compiled object code, generated documentation, + and conversions to other media types. + + "Work" shall mean the work of authorship, whether in Source or + Object form, made available under the License, as indicated by a + copyright notice that is included in or attached to the work + (an example is provided in the Appendix below). + + "Derivative Works" shall mean any work, whether in Source or Object + form, that is based on (or derived from) the Work and for which the + editorial revisions, annotations, elaborations, or other modifications + represent, as a whole, an original work of authorship. For the purposes + of this License, Derivative Works shall not include works that remain + separable from, or merely link (or bind by name) to the interfaces of, + the Work and Derivative Works thereof. + + "Contribution" shall mean any work of authorship, including + the original version of the Work and any modifications or additions + to that Work or Derivative Works thereof, that is intentionally + submitted to Licensor for inclusion in the Work by the copyright owner + or by an individual or Legal Entity authorized to submit on behalf of + the copyright owner. For the purposes of this definition, "submitted" + means any form of electronic, verbal, or written communication sent + to the Licensor or its representatives, including but not limited to + communication on electronic mailing lists, source code control systems, + and issue tracking systems that are managed by, or on behalf of, the + Licensor for the purpose of discussing and improving the Work, but + excluding communication that is conspicuously marked or otherwise + designated in writing by the copyright owner as "Not a Contribution." + + "Contributor" shall mean Licensor and any individual or Legal Entity + on behalf of whom a Contribution has been received by Licensor and + subsequently incorporated within the Work. + + 2. Grant of Copyright License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + copyright license to reproduce, prepare Derivative Works of, + publicly display, publicly perform, sublicense, and distribute the + Work and such Derivative Works in Source or Object form. + + 3. Grant of Patent License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + (except as stated in this section) patent license to make, have made, + use, offer to sell, sell, import, and otherwise transfer the Work, + where such license applies only to those patent claims licensable + by such Contributor that are necessarily infringed by their + Contribution(s) alone or by combination of their Contribution(s) + with the Work to which such Contribution(s) was submitted. If You + institute patent litigation against any entity (including a + cross-claim or counterclaim in a lawsuit) alleging that the Work + or a Contribution incorporated within the Work constitutes direct + or contributory patent infringement, then any patent licenses + granted to You under this License for that Work shall terminate + as of the date such litigation is filed. + + 4. Redistribution. You may reproduce and distribute copies of the + Work or Derivative Works thereof in any medium, with or without + modifications, and in Source or Object form, provided that You + meet the following conditions: + + (a) You must give any other recipients of the Work or + Derivative Works a copy of this License; and + + (b) You must cause any modified files to carry prominent notices + stating that You changed the files; and + + (c) You must retain, in the Source form of any Derivative Works + that You distribute, all copyright, patent, trademark, and + attribution notices from the Source form of the Work, + excluding those notices that do not pertain to any part of + the Derivative Works; and + + (d) If the Work includes a "NOTICE" text file as part of its + distribution, then any Derivative Works that You distribute must + include a readable copy of the attribution notices contained + within such NOTICE file, excluding those notices that do not + pertain to any part of the Derivative Works, in at least one + of the following places: within a NOTICE text file distributed + as part of the Derivative Works; within the Source form or + documentation, if provided along with the Derivative Works; or, + within a display generated by the Derivative Works, if and + wherever such third-party notices normally appear. The contents + of the NOTICE file are for informational purposes only and + do not modify the License. You may add Your own attribution + notices within Derivative Works that You distribute, alongside + or as an addendum to the NOTICE text from the Work, provided + that such additional attribution notices cannot be construed + as modifying the License. + + You may add Your own copyright statement to Your modifications and + may provide additional or different license terms and conditions + for use, reproduction, or distribution of Your modifications, or + for any such Derivative Works as a whole, provided Your use, + reproduction, and distribution of the Work otherwise complies with + the conditions stated in this License. + + 5. Submission of Contributions. Unless You explicitly state otherwise, + any Contribution intentionally submitted for inclusion in the Work + by You to the Licensor shall be under the terms and conditions of + this License, without any additional terms or conditions. + Notwithstanding the above, nothing herein shall supersede or modify + the terms of any separate license agreement you may have executed + with Licensor regarding such Contributions. + + 6. Trademarks. This License does not grant permission to use the trade + names, trademarks, service marks, or product names of the Licensor, + except as required for reasonable and customary use in describing the + origin of the Work and reproducing the content of the NOTICE file. + + 7. Disclaimer of Warranty. Unless required by applicable law or + agreed to in writing, Licensor provides the Work (and each + Contributor provides its Contributions) on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or + implied, including, without limitation, any warranties or conditions + of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A + PARTICULAR PURPOSE. You are solely responsible for determining the + appropriateness of using or redistributing the Work and assume any + risks associated with Your exercise of permissions under this License. + + 8. Limitation of Liability. In no event and under no legal theory, + whether in tort (including negligence), contract, or otherwise, + unless required by applicable law (such as deliberate and grossly + negligent acts) or agreed to in writing, shall any Contributor be + liable to You for damages, including any direct, indirect, special, + incidental, or consequential damages of any character arising as a + result of this License or out of the use or inability to use the + Work (including but not limited to damages for loss of goodwill, + work stoppage, computer failure or malfunction, or any and all + other commercial damages or losses), even if such Contributor + has been advised of the possibility of such damages. + + 9. Accepting Warranty or Additional Liability. While redistributing + the Work or Derivative Works thereof, You may choose to offer, + and charge a fee for, acceptance of support, warranty, indemnity, + or other liability obligations and/or rights consistent with this + License. However, in accepting such obligations, You may act only + on Your own behalf and on Your sole responsibility, not on behalf + of any other Contributor, and only if You agree to indemnify, + defend, and hold each Contributor harmless for any liability + incurred by, or claims asserted against, such Contributor by reason + of your accepting any such warranty or additional liability. diff --git a/README.rst b/README.rst new file mode 100644 index 0000000..4c030ee --- /dev/null +++ b/README.rst @@ -0,0 +1,79 @@ +======================== +Team and repository tags +======================== + +.. image:: http://governance.openstack.org/badges/designate-dashboard.svg + :target: http://governance.openstack.org/reference/tags/index.html + +.. Change things from this point on + +=============================== +designatedashboard +=============================== + +Designate Horizon UI bits + +* Free software: Apache license + +Features +-------- + +* TODO + + +Howto +----- + +1. Package the designatedashboard by running:: + + python setup.py sdist + + This will create a python egg in the dist folder, which can be used to install + on the horizon machine or within horizon's python virtual environment. + + -- or -- + + Install directly from source by running "python setup.py --install" + + Note: On some systems python may throw an error like + + 'Exception: Versioning for this project requires either an sdist tarball, or access + to an upstream git repository' + + this seems to be a result of mismatched pbr versioning. A hacking workaround for development + purposes is replacing the pbr call with a hard-coded version (e.g. '1.0.1') in + designatedashboard/__init__.py. + +2. Copy panel plugin files into your Horizon config. These files can be found in designatedashboard/enabled + and should be copied to /usr/share/openstack-dashboard/openstack_dashboard/local/enabled or the + equivalent directory for your openstack-dashboard install. + +3. Make sure your keystone catalog contains endpoints for service type 'dns'. If no such endpoints are + found, the designatedashboard panels will not render. + +4. (Optional) Copy the designate policy file into horizon's policy files folder, and add this config:: + + 'dns': 'designate_policy.json', + +5. (Optional) Within your horizon settings file(s) (either the local settings or the other settings.py), add + the line below. This will make it so the record create/update screen uses a drop down of your floating ip + addresses instead of a free form text field:: + + DESIGNATE = { 'records_use_fips': True } + + +Test +---- + +* How to run JS tests: + + * Install npm and nodejs=4.8.4 + + $ ``sudo apt-get install npm`` + $ ``curl -sL https://deb.nodesource.com/setup_4.x | sudo -E bash -`` + $ ``sudo apt-get install -y nodejs`` + + 1. ``npm install`` (to create virtual environment and install all dependencies in package.json) + 2. ``npm run lint`` for eslint + 3. ``npm run test`` for JS unit tests + diff --git a/babel-django.cfg b/babel-django.cfg new file mode 100644 index 0000000..e78d6c0 --- /dev/null +++ b/babel-django.cfg @@ -0,0 +1,5 @@ +[extractors] +django = django_babel.extract:extract_django + +[python: **.py] +[django: **/templates/**.html] diff --git a/designatedashboard/__init__.py b/designatedashboard/__init__.py new file mode 100644 index 0000000..a415499 --- /dev/null +++ b/designatedashboard/__init__.py @@ -0,0 +1,19 @@ +# -*- coding: utf-8 -*- + +# Licensed under the Apache License, Version 2.0 (the "License"); you may +# not use this file except in compliance with the License. You may obtain +# a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT +# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the +# License for the specific language governing permissions and limitations +# under the License. + +import pbr.version + + +__version__ = pbr.version.VersionInfo( + 'designate-dashboard').version_string() diff --git a/designatedashboard/api/__init__.py b/designatedashboard/api/__init__.py new file mode 100644 index 0000000..97ba638 --- /dev/null +++ b/designatedashboard/api/__init__.py @@ -0,0 +1 @@ +from designatedashboard.api import designate # noqa diff --git a/designatedashboard/api/designate.py b/designatedashboard/api/designate.py new file mode 100644 index 0000000..6095410 --- /dev/null +++ b/designatedashboard/api/designate.py @@ -0,0 +1,171 @@ +# Copyright 2013 Hewlett-Packard Development Company, L.P. +# +# Licensed under the Apache License, Version 2.0 (the "License"); you may +# not use this file except in compliance with the License. You may obtain +# a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT +# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the +# License for the specific language governing permissions and limitations +# under the License. + +from __future__ import absolute_import + +from designateclient.v1 import Client # noqa +from designateclient.v1.domains import Domain # noqa +from designateclient.v1.records import Record # noqa +from django.conf import settings # noqa + +from horizon import exceptions + +from openstack_dashboard.api.base import url_for # noqa +from oslo_log import log as logging + +LOG = logging.getLogger(__name__) + + +def designateclient(request): + designate_url = "" + try: + designate_url = url_for(request, 'dns') + except exceptions.ServiceCatalogException: + LOG.debug('no dns service configured.') + return None + + insecure = getattr(settings, 'OPENSTACK_SSL_NO_VERIFY', False) + cacert = getattr(settings, 'OPENSTACK_SSL_CACERT', None) + + return Client(endpoint=designate_url, + token=request.user.token.id, + username=request.user.username, + tenant_id=request.user.project_id, + insecure=insecure, + cacert=cacert) + + +def domain_get(request, domain_id): + d_client = designateclient(request) + if d_client is None: + return [] + return d_client.domains.get(domain_id) + + +def domain_list(request): + d_client = designateclient(request) + if d_client is None: + return [] + return d_client.domains.list() + + +def domain_create(request, name, email, ttl=None, description=None): + d_client = designateclient(request) + if d_client is None: + return None + + options = { + 'description': description, + } + + # TTL needs to be optionally added as argument because the client + # won't accept a None value + if ttl is not None: + options['ttl'] = ttl + + domain = Domain(name=name, email=email, **options) + + return d_client.domains.create(domain) + + +def domain_update(request, domain_id, email, ttl, description=None): + d_client = designateclient(request) + if d_client is None: + return None + + # A quirk of the designate client is that you need to start with a + # base record and then update individual fields in order to persist + # the data. The designate client will only send the 'changed' fields. + domain = Domain(id=domain_id, name='', email='') + + domain.email = email + domain.ttl = ttl + domain.description = description + + return d_client.domains.update(domain) + + +def domain_delete(request, domain_id): + d_client = designateclient(request) + if d_client is None: + return [] + return d_client.domains.delete(domain_id) + + +def server_list(request, domain_id): + d_client = designateclient(request) + if d_client is None: + return [] + return d_client.domains.list_domain_servers(domain_id) + + +def record_list(request, domain_id): + d_client = designateclient(request) + if d_client is None: + return [] + return d_client.records.list(domain_id) + + +def record_get(request, domain_id, record_id): + d_client = designateclient(request) + if d_client is None: + return [] + return d_client.records.get(domain_id, record_id) + + +def record_delete(request, domain_id, record_id): + d_client = designateclient(request) + if d_client is None: + return [] + return d_client.records.delete(domain_id, record_id) + + +def record_create(request, domain_id, **kwargs): + d_client = designateclient(request) + if d_client is None: + return [] + + record = Record(**kwargs) + return d_client.records.create(domain_id, record) + + +def record_update(request, domain_id, record_id, **kwargs): + d_client = designateclient(request) + if d_client is None: + return [] + + # A quirk of the designate client is that you need to start with a + # base record and then update individual fields in order to persist + # the data. The designate client will only send the 'changed' fields. + record = Record( + id=record_id, + type='A', + name='', + data='') + + record.type = kwargs.get('type', None) + record.name = kwargs.get('name', None) + record.data = kwargs.get('data', None) + record.priority = kwargs.get('priority', None) + record.ttl = kwargs.get('ttl', None) + record.description = kwargs.get('description', None) + + return d_client.records.update(domain_id, record) + + +def quota_get(request, project_id=None): + if not project_id: + project_id = request.user.project_id + d_client = designateclient(request) + return d_client.quotas.get(project_id) diff --git a/designatedashboard/api/rest/__init__.py b/designatedashboard/api/rest/__init__.py new file mode 100644 index 0000000..6e741ff --- /dev/null +++ b/designatedashboard/api/rest/__init__.py @@ -0,0 +1,16 @@ +# (c) Copyright Hewlett Packard Enterprise Development LP +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +"""REST API for Horizon dashboard Javascript code. +""" +from . import passthrough # noqa diff --git a/designatedashboard/api/rest/passthrough.py b/designatedashboard/api/rest/passthrough.py new file mode 100644 index 0000000..00e1228 --- /dev/null +++ b/designatedashboard/api/rest/passthrough.py @@ -0,0 +1,119 @@ +# Copyright 2016, Hewlett Packard Enterprise Development, LP +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +"""API for the passthrough service. +""" +from django.conf import settings +from django.views import generic +import functools +import requests +from requests.exceptions import HTTPError + +from horizon import exceptions +from openstack_dashboard.api import base +from openstack_dashboard.api.rest import urls +from openstack_dashboard.api.rest import utils as rest_utils +from oslo_log import log as logging + +LOG = logging.getLogger(__name__) + + +def _passthrough_request(request_method, url, + request, data=None, params=None): + """Makes a request to the appropriate service API with an optional payload. + + Should set any necessary auth headers and SSL parameters. + """ + + # Set verify if a CACERT is set and SSL_NO_VERIFY isn't True + verify = getattr(settings, 'OPENSTACK_SSL_CACERT', None) + if getattr(settings, 'OPENSTACK_SSL_NO_VERIFY', False): + verify = False + + service_url = _get_service_url(request, 'dns') + request_url = '{}{}'.format( + service_url, + url if service_url.endswith('/') else ('/' + url) + ) + + response = request_method( + request_url, + headers={'X-Auth-Token': request.user.token.id}, + json=data, + verify=verify, + params=params + ) + + try: + response.raise_for_status() + except HTTPError as e: + LOG.debug(e.response.content) + for error in rest_utils.http_errors: + if (e.response.status_code == getattr(error, 'status_code', 0) and + exceptions.HorizonException in error.__bases__): + raise error + raise + + return response + + +# Create some convenience partial functions +passthrough_get = functools.partial(_passthrough_request, requests.get) +passthrough_post = functools.partial(_passthrough_request, requests.post) +passthrough_put = functools.partial(_passthrough_request, requests.put) +passthrough_patch = functools.partial(_passthrough_request, requests.patch) +passthrough_delete = functools.partial(_passthrough_request, requests.delete) + + +def _get_service_url(request, service): + """Get service's URL from keystone; allow an override in settings""" + service_url = getattr(settings, service.upper() + '_URL', None) + try: + service_url = base.url_for(request, service) + except exceptions.ServiceCatalogException: + pass + # Currently the keystone endpoint is http://host:port/ + # without the version. + return service_url + + +@urls.register +class Passthrough(generic.View): + """Pass-through API for executing service requests. + + Horizon only adds auth and CORS proxying. + """ + url_regex = r'dns/(?P.+)$' + + @rest_utils.ajax() + def get(self, request, path): + return passthrough_get(path, request).json() + + @rest_utils.ajax() + def post(self, request, path): + data = dict(request.DATA) if request.DATA else {} + return passthrough_post(path, request, data).json() + + @rest_utils.ajax() + def put(self, request, path): + data = dict(request.DATA) if request.DATA else {} + return passthrough_put(path, request, data).json() + + @rest_utils.ajax() + def patch(self, request, path): + data = dict(request.DATA) if request.DATA else {} + return passthrough_patch(path, request, data).json() + + @rest_utils.ajax() + def delete(self, request, path): + return passthrough_delete(path, request).json() diff --git a/designatedashboard/dashboards/__init__.py b/designatedashboard/dashboards/__init__.py new file mode 100644 index 0000000..e69de29 diff --git a/designatedashboard/dashboards/project/__init__.py b/designatedashboard/dashboards/project/__init__.py new file mode 100644 index 0000000..e69de29 diff --git a/designatedashboard/dashboards/project/dns_domains/__init__.py b/designatedashboard/dashboards/project/dns_domains/__init__.py new file mode 100644 index 0000000..e69de29 diff --git a/designatedashboard/dashboards/project/dns_domains/forms.py b/designatedashboard/dashboards/project/dns_domains/forms.py new file mode 100644 index 0000000..852c2d9 --- /dev/null +++ b/designatedashboard/dashboards/project/dns_domains/forms.py @@ -0,0 +1,549 @@ +# Copyright 2013 Hewlett-Packard Development Company, L.P. +# +# Licensed under the Apache License, Version 2.0 (the "License"); you may +# not use this file except in compliance with the License. You may obtain +# a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT +# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the +# License for the specific language governing permissions and limitations +# under the License. +import functools +import re +import six + +from designateclient import exceptions as designate_exceptions +from django.core.exceptions import ValidationError # noqa +from django.core import validators +from django.template.loader import render_to_string +from django.utils.translation import ugettext_lazy as _ # noqa + +from horizon import forms +from horizon import messages + +from designatedashboard import api +from designatedashboard.dashboards.project.dns_domains.utils\ + import limit_records_to_fips +from oslo_log import log as logging + + +LOG = logging.getLogger(__name__) + +MAX_TTL = 2147483647 +# These regexes were given to me by Kiall Mac Innes here: +# https://gerrit.hpcloud.net/#/c/25300/2/ +DOMAIN_NAME_REGEX = r'^(?!.{255,})(?:(?!\-)[A-Za-z0-9_\-]{1,63}(?= 500: + msg += " (Request ID: %(request_id)s" + data["request_id"] = ex.request_id + + form.api_error(_(msg) % data) # noqa + + return False + except Exception: + messages.error(request, form.exc_message) + return True + + return wrapped + + +class DomainForm(forms.SelfHandlingForm): + + '''Base class for DomainCreate and DomainUpdate forms. + + Sets-up all of the common form fields. + ''' + + name = forms.RegexField( + label=_("Domain Name"), + regex=DOMAIN_NAME_REGEX, + error_messages={'invalid': _('Enter a valid domain name.')}, + ) + + email = forms.EmailField( + label=_("Email"), + max_length=255, + ) + + ttl = forms.IntegerField( + label=_("TTL (seconds)"), + min_value=1, + max_value=MAX_TTL, + required=False, + ) + + description = forms.CharField( + label=_("Description"), + required=False, + max_length=160, + widget=forms.Textarea(), + ) + + +class DomainCreate(DomainForm): + + '''Form for creating new domain records. + + Name and email address are + required. + ''' + exc_message = _("Unable to create domain.") + + @handle_exc + def handle(self, request, data): + domain = api.designate.domain_create( + request, + name=data['name'], + email=data['email'], + ttl=data['ttl'], + description=data['description']) + messages.success(request, + _('Domain %(name)s created.') % + {"name": domain.name}) + return domain + + +class DomainUpdate(DomainForm): + + '''Form for displaying domain record details and updating them.''' + exc_message = _('Unable to update domain.') + + id = forms.CharField( + required=False, + widget=forms.HiddenInput() + ) + + serial = forms.CharField( + label=_("Serial"), + required=False, + widget=forms.TextInput(attrs={'readonly': 'readonly'}), + ) + + created_at = forms.CharField( + label=_("Created At"), + required=False, + widget=forms.TextInput(attrs={'readonly': 'readonly'}), + ) + + updated_at = forms.CharField( + label=_("Updated At"), + required=False, + widget=forms.TextInput(attrs={'readonly': 'readonly'}), + ) + + def __init__(self, request, *args, **kwargs): + super(DomainUpdate, self).__init__(request, *args, **kwargs) + + # Mark name as read-only + self.fields['name'].required = False + self.fields['name'].widget.attrs['readonly'] = 'readonly' + + self.fields['ttl'].required = True + + # Customize display order for fields + self.fields.keyOrder = [ + 'id', + 'name', + 'serial', + 'email', + 'ttl', + 'description', + 'created_at', + 'updated_at', + ] + + @handle_exc + def handle(self, request, data): + domain = api.designate.domain_update( + request, + domain_id=data['id'], + email=data['email'], + ttl=data['ttl'], + description=data['description']) + messages.success(request, + _('Domain %(name)s updated.') % + {"name": domain.name}) + return domain + + +class PrefixWidget(forms.TextInput): + + def render(self, name, value, attrs=None): + template_name = 'project/dns_domains/prefix_html_widget.html' + result = super(PrefixWidget, self).render(name, value, attrs) + view_data = {'input': result, + 'suffix': getattr(self, "suffix", '')} + return render_to_string(template_name, view_data) + + +class RecordForm(forms.SelfHandlingForm): + + '''Base class for RecordCreate and RecordUpdate forms. + + Sets-up all of + the form fields and implements the complex validation logic. + ''' + + domain_id = forms.CharField( + widget=forms.HiddenInput()) + + domain_name = forms.CharField( + widget=forms.HiddenInput()) + + type = forms.ChoiceField( + label=_("Record Type"), + required=False, + choices=[ + ('a', _('A - Address record')), + ('aaaa', _('AAAA - IPv6 address record')), + ('cname', _('CNAME - Canonical name record')), + ('mx', _('MX - Mail exchange record')), + ('ptr', _('PTR - Pointer record')), + ('spf', _('SPF - Sender Policy Framework')), + ('srv', _('SRV - Service locator')), + ('sshfp', _('SSHFP - SSH Public Key Fingerprint')), + ('txt', _('TXT - Text record')), + ], + widget=forms.Select(attrs={ + 'class': 'switchable', + 'data-slug': 'record_type', + }), + ) + + name = forms.CharField( + required=False, + widget=PrefixWidget(attrs={ + 'class': 'switched', + 'data-switch-on': 'record_type', + 'data-record_type-a': _('Name'), + 'data-record_type-aaaa': _('Name'), + 'data-record_type-cname': _('Name'), + 'data-record_type-mx': _('Name'), + 'data-record_type-ns': _('Name'), + 'data-record_type-ptr': _('Name'), + 'data-record_type-soa': _('Name'), + 'data-record_type-spf': _('Name'), + 'data-record_type-srv': _('Name'), + 'data-record_type-sshfp': _('Name'), + 'data-record_type-txt': _('Name'), + }), + ) + + data = forms.CharField( + required=False, + widget=forms.TextInput(attrs={ + 'class': 'switched', + 'data-switch-on': 'record_type', + 'data-record_type-a': _('IP Address'), + 'data-record_type-aaaa': _('IP Address'), + 'data-record_type-cname': _('Canonical Name'), + 'data-record_type-ns': _('Name Server'), + 'data-record_type-mx': _('Mail Server'), + 'data-record_type-ptr': _('PTR Domain Name'), + 'data-record_type-soa': _('Value'), + 'data-record_type-srv': _('Value'), + }), + ) + + ip_addr = forms.ChoiceField( + required=False, + widget=forms.Select(attrs={ + 'class': 'switched', + 'data-switch-on': 'record_type', + 'data-record_type-a': _('IP Address'), + 'data-record_type-aaaa': _('IP Address'), + }), + ) + + txt = forms.CharField( + label=_('TXT'), + required=False, + widget=forms.Textarea(attrs={ + 'class': 'switched', + 'data-switch-on': 'record_type', + 'data-record_type-txt': _('Text'), + 'data-record_type-spf': _('Text'), + 'data-record_type-sshfp': _('Text'), + }), + ) + + priority = forms.IntegerField( + min_value=0, + max_value=65535, + required=False, + widget=forms.TextInput(attrs={ + 'class': 'switched', + 'data-switch-on': 'record_type', + 'data-record_type-mx': _('Priority'), + 'data-record_type-srv': _('Priority'), + }), + ) + + ttl = forms.IntegerField( + label=_('TTL'), + min_value=1, + max_value=MAX_TTL, + required=False, + widget=forms.TextInput(attrs={ + 'class': 'switched', + 'data-switch-on': 'record_type', + 'data-record_type-a': _('TTL'), + 'data-record_type-aaaa': _('TTL'), + 'data-record_type-cname': _('TTL'), + 'data-record_type-mx': _('TTL'), + 'data-record_type-ptr': _('TTL'), + 'data-record_type-soa': _('TTL'), + 'data-record_type-spf': _('TTL'), + 'data-record_type-srv': _('TTL'), + 'data-record_type-sshfp': _('TTL'), + 'data-record_type-txt': _('TTL'), + }), + ) + + description = forms.CharField( + label=_("Description"), + required=False, + max_length=160, + widget=forms.Textarea(), + ) + + def __init__(self, request, *args, **kwargs): + super(RecordForm, self).__init__(request, *args, **kwargs) + initial = kwargs.get('initial', {}) + domain_suffix = "." + initial['domain_name'] + self.fields['name'].widget.suffix = domain_suffix + self.fields['name'].max_length = min(NAME_MAX_LENGTH, + 255 - len(domain_suffix)) + if limit_records_to_fips(): + del self.fields['data'].widget.attrs['data-record_type-a'] + del self.fields['data'].widget.attrs['data-record_type-aaaa'] + self.fields['ip_addr'].choices = \ + self.populate_ip_addr_choices(request, + initial) + else: + del self.fields['ip_addr'] + + def _generate_fip_list(self, fips, instances): + instance_dict = {instance.id: instance for instance in instances} + for fip in fips: + instance_name = _("Unknown instance name") + if getattr(fip, "instance_id", "None") in instance_dict: + instance_name = instance_dict[getattr(fip, "instance_id")].name + yield (fip.ip, "%s (%s)" % (fip.ip, instance_name)) + + def populate_ip_addr_choices(self, request, initial): + results = [(None, _('Select an IP')), ] + if (initial.get('ip_addr') and + initial['ip_addr'] not in [fip.ip for fip in initial['fips']]): + """The record is currently using an ip not in the list + of fips - this can happen when instance goes away or in + multi region setups + """ + results.append((initial['ip_addr'], initial['ip_addr'])) + results.extend(self._generate_fip_list(initial['fips'], + initial['instances'])) + if len(results) == 1: + messages.warning(request, _("There are no floating IP addresses " + "currently in use to select from.")) + return results + + def clean_type(self): + '''Type value needs to be uppercased before it is sent to the API.''' + return self.cleaned_data['type'].upper() + + def clean(self): + '''Handles the validation logic for the domain record form. + + Validation gets pretty complicated due to the fact that the different + record types (A, AAAA, MX, etc) have different requirements for + each of the fields. + ''' + + cleaned_data = super(RecordForm, self).clean() + record_type = cleaned_data['type'] + domain_name = cleaned_data['domain_name'] + if limit_records_to_fips(): + ip_addr = cleaned_data.pop('ip_addr') + if (record_type in ['AAAA', 'A'] and limit_records_to_fips()): + cleaned_data['data'] = ip_addr + + # Name field + if self._is_field_blank(cleaned_data, 'name'): + if record_type in ['CNAME', 'SRV']: + self._add_required_field_error('name') + elif record_type in ['MX', 'A', 'AAAA', 'TXT', 'PTR']: + cleaned_data['name'] = domain_name + else: + if record_type == 'SRV': + if not re.match(SRV_NAME_REGEX, cleaned_data['name']): + self._add_field_error('name', _('Enter a valid SRV name')) + else: + cleaned_data['name'] += domain_name + else: + cleaned_data['name'] += "." + domain_name + if not re.match(WILDCARD_DOMAIN_NAME_REGEX, + cleaned_data['name']): + self._add_field_error('name', + _('Enter a valid hostname. The ' + 'hostname should contain letters ' + 'and numbers, and be no more than ' + '63 characters.')) + # Data field + if self._is_field_blank(cleaned_data, 'data'): + if record_type in ['A', 'AAAA', 'CNAME', 'MX', 'SRV']: + self._add_required_field_error('data') + else: + if record_type == 'A': + try: + validators.validate_ipv4_address(cleaned_data['data']) + except ValidationError: + self._add_field_error('data', + _('Enter a valid IPv4 address')) + + elif record_type == 'AAAA': + try: + validators.validate_ipv6_address(cleaned_data['data']) + except ValidationError: + self._add_field_error('data', + _('Enter a valid IPv6 address')) + + elif record_type in ['CNAME', 'MX', 'PTR']: + if not re.match(DOMAIN_NAME_REGEX, cleaned_data['data']): + self._add_field_error('data', _('Enter a valid hostname')) + + elif record_type == 'SRV': + if not re.match(SRV_DATA_REGEX, cleaned_data['data']): + self._add_field_error('data', + _('Enter a valid SRV record')) + + # Txt field + if self._is_field_blank(cleaned_data, 'txt'): + if record_type == 'TXT': + self._add_required_field_error('txt') + else: + if record_type == 'TXT': + cleaned_data['data'] = cleaned_data['txt'] + + if record_type == 'SSHFP': + if not re.match(SSHFP_DATA_REGEX, cleaned_data['txt']): + self._add_field_error('txt', + _('Enter a valid SSHFP record')) + cleaned_data['data'] = cleaned_data['txt'] + + cleaned_data.pop('txt') + + # Priority field + # Check against '' instead of using _is_field_blank because we need to + # allow a valud of 0. + if ('priority' not in cleaned_data or + cleaned_data['priority'] == '' or + cleaned_data['priority'] is None): + if record_type in ['MX', 'SRV']: + self._add_required_field_error('priority') + + # Rename 'id' to 'record_id' + if 'id' in cleaned_data: + cleaned_data['record_id'] = cleaned_data.pop('id') + + # Remove domain_name + cleaned_data.pop('domain_name') + + return cleaned_data + + def _add_required_field_error(self, field): + '''Set a required field error on the specified field.''' + self._add_field_error(field, _('This field is required')) + + def _add_field_error(self, field, msg): + '''Set the specified msg as an error on the field.''' + self._errors[field] = self.error_class([msg]) + + def _is_field_blank(self, cleaned_data, field): + '''Returns a flag indicating whether the specified field is blank.''' + return field in cleaned_data and not cleaned_data[field] + + +class RecordCreate(RecordForm): + + '''Form for creating a new domain record.''' + exc_message = _('Unable to create record.') + + @handle_exc + def handle(self, request, data): + record = api.designate.record_create(request, **data) + messages.success(request, + _('Domain record %(name)s created.') % + {"name": record.name}) + return record + + +class RecordUpdate(RecordForm): + + '''Form for editing a domain record.''' + exc_message = _('Unable to create record.') + + id = forms.CharField(widget=forms.HiddenInput()) + + def __init__(self, request, *args, **kwargs): + super(RecordUpdate, self).__init__(request, *args, **kwargs) + + # Force the type field to be read-only + self.fields['type'].widget.attrs['readonly'] = 'readonly' + + if self['type'].value() in ('soa', 'ns'): + self.fields['type'].choices.append(('ns', _('NS'))) + self.fields['type'].choices.append(('soa', _('SOA'))) + + self.fields['name'].widget.attrs['readonly'] = 'readonly' + self.fields['data'].widget.attrs['readonly'] = 'readonly' + self.fields['description'].widget.attrs['readonly'] = 'readonly' + self.fields['ttl'].widget.attrs['readonly'] = 'readonly' + + # Filter the choice list so that it only contains the type for + # the current record. Ideally, we would just disable the select + # field, but that has the unfortunate side-effect of breaking + # the 'selectable' javascript code. + self.fields['type'].choices = ( + [choice for choice in self.fields['type'].choices + if choice[0] == self.initial['type']]) + + @handle_exc + def handle(self, request, data): + + if data['type'] in ('SOA', 'NS'): + return True + + record = api.designate.record_update(request, **data) + + messages.success(request, + _('Domain record %(name)s updated.') % + {"name": record.name}) + + return record diff --git a/designatedashboard/dashboards/project/dns_domains/panel.py b/designatedashboard/dashboards/project/dns_domains/panel.py new file mode 100644 index 0000000..a7df87f --- /dev/null +++ b/designatedashboard/dashboards/project/dns_domains/panel.py @@ -0,0 +1,26 @@ +# Copyright 2013 Hewlett-Packard Development Company, L.P. +# +# Licensed under the Apache License, Version 2.0 (the "License"); you may +# not use this file except in compliance with the License. You may obtain +# a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT +# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the +# License for the specific language governing permissions and limitations +# under the License. +from django.utils.translation import ugettext_lazy as _ # noqa + +import horizon +from openstack_dashboard.dashboards.project import dashboard + + +class DNSDomains(horizon.Panel): + name = _("Domains") + slug = 'dns_domains' + permissions = ('openstack.services.dns',) + + +dashboard.Project.register(DNSDomains) diff --git a/designatedashboard/dashboards/project/dns_domains/tables.py b/designatedashboard/dashboards/project/dns_domains/tables.py new file mode 100644 index 0000000..07b4f8e --- /dev/null +++ b/designatedashboard/dashboards/project/dns_domains/tables.py @@ -0,0 +1,224 @@ +# Copyright 2013 Hewlett-Packard Development Company, L.P. +# +# Licensed under the Apache License, Version 2.0 (the "License"); you may +# not use this file except in compliance with the License. You may obtain +# a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT +# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the +# License for the specific language governing permissions and limitations +# under the License. +from django.core import urlresolvers +from django.utils.translation import ugettext_lazy as _ # noqa + +from horizon import messages +from horizon import tables +from horizon.utils import memoized + +from designatedashboard import api + +from openstack_dashboard import policy +from oslo_log import log as logging + +LOG = logging.getLogger(__name__) + +EDITABLE_RECORD_TYPES = ( + "A", + "AAAA", + "CNAME", + "MX", + "PTR", + "SPF", + "SRV", + "SSHFP", + "TXT", +) + + +class CreateDomain(tables.LinkAction): + + '''Link action for navigating to the CreateDomain view.''' + name = "create_domain" + verbose_name = _("Create Domain") + url = "horizon:project:dns_domains:create_domain" + classes = ("ajax-modal", "btn-create") + policy_rules = (("dns", "create_domain"),) + + @memoized.memoized_method + def allowed(self, request, datum): + if policy.check((("dns", "get_quota"),), request): + try: + if self.table: + quota = api.designate.quota_get(request) + return quota['domains'] > len(self.table.data) + except Exception: + msg = _("The quotas could not be retrieved.") + messages.warning(request, msg) + return True + + +class EditDomain(tables.LinkAction): + + '''Link action for navigating to the UpdateDomain view.''' + name = "edit_domain" + verbose_name = _("Edit Domain") + url = "horizon:project:dns_domains:update_domain" + classes = ("ajax-modal", "btn-edit") + policy_rules = (("dns", "update_domain"),) + + +class ManageRecords(tables.LinkAction): + + '''Link action for navigating to the ManageRecords view.''' + name = "manage_records" + verbose_name = _("Manage Records") + url = "horizon:project:dns_domains:records" + classes = ("btn-edit") + policy_rules = (("dns", "get_records"),) + + +class DeleteDomain(tables.BatchAction): + + '''Batch action for deleting domains.''' + name = "delete" + action_present = _("Delete") + action_past = _("Deleted") + data_type_singular = _("Domain") + data_type_plural = _("Domains") + classes = ('btn-danger', 'btn-delete') + policy_rules = (("dns", "delete_domain"),) + + def action(self, request, domain_id): + api.designate.domain_delete(request, domain_id) + + +class CreateRecord(tables.LinkAction): + + '''Link action for navigating to the CreateRecord view.''' + name = "create_record" + verbose_name = _("Create Record") + classes = ("ajax-modal", "btn-create") + policy_rules = (("dns", "create_record"),) + + def get_link_url(self, datum=None): + url = "horizon:project:dns_domains:create_record" + return urlresolvers.reverse(url, kwargs=self.table.kwargs) + + +class EditRecord(tables.LinkAction): + + '''Link action for navigating to the UpdateRecord view.''' + name = "edit_record" + verbose_name = _("Edit Record") + classes = ("ajax-modal", "btn-edit") + policy_rules = (("dns", "update_record"),) + + def get_link_url(self, datum=None): + url = "horizon:project:dns_domains:update_record" + kwargs = { + 'domain_id': datum.domain_id, + 'record_id': datum.id, + } + + return urlresolvers.reverse(url, kwargs=kwargs) + + def allowed(self, request, record=None): + return record.type in EDITABLE_RECORD_TYPES + + +class DeleteRecord(tables.DeleteAction): + + '''Link action for navigating to the UpdateRecord view.''' + data_type_singular = _("Record") + policy_rules = (("dns", "delete_record"),) + + def delete(self, request, record_id): + domain_id = self.table.kwargs['domain_id'] + return api.designate.record_delete(request, domain_id, record_id) + + def allowed(self, request, record=None): + return record.type in EDITABLE_RECORD_TYPES + + +class BatchDeleteRecord(tables.BatchAction): + + '''Batch action for deleting domain records.''' + + name = "delete" + action_present = _("Delete") + action_past = _("Deleted") + data_type_singular = _("Record") + classes = ('btn-danger', 'btn-delete') + policy_rules = (("dns", "delete_record"),) + + def action(self, request, record_id): + domain_id = self.table.kwargs['domain_id'] + api.designate.record_delete(request, domain_id, record_id) + + +class DomainsTable(tables.DataTable): + + '''Data table for displaying domain summary information.''' + + name = tables.Column("name", + verbose_name=_("Name"), + link=("horizon:project:dns_domains:domain_detail")) + + email = tables.Column("email", + verbose_name=_("Email")) + + ttl = tables.Column("ttl", + verbose_name=_("TTL")) + + serial = tables.Column("serial", + verbose_name=_("Serial")) + + class Meta(object): + name = "domains" + verbose_name = _("Domains") + table_actions = (CreateDomain, DeleteDomain,) + row_actions = (ManageRecords, EditDomain, DeleteDomain,) + + +def record__details_link(record): + '''Returns a link to the view for updating DNS records.''' + + return urlresolvers.reverse( + "horizon:project:dns_domains:view_record", + args=(record.domain_id, record.id)) + + +class RecordsTable(tables.DataTable): + + '''Data table for displaying summary information for a domains records.''' + + name = tables.Column("name", + verbose_name=_("Name"), + link=record__details_link, + ) + + type = tables.Column("type", + verbose_name=_("Type") + ) + + data = tables.Column("data", + verbose_name=_("Data") + ) + + priority = tables.Column("priority", + verbose_name=_("Priority"), + ) + + ttl = tables.Column("ttl", + verbose_name=_("TTL") + ) + + class Meta(object): + name = "records" + verbose_name = _("Records") + table_actions = (CreateRecord,) + row_actions = (EditRecord, DeleteRecord,) + multi_select = False diff --git a/designatedashboard/dashboards/project/dns_domains/templates/dns_domains/_create_domain.html b/designatedashboard/dashboards/project/dns_domains/templates/dns_domains/_create_domain.html new file mode 100644 index 0000000..3aa22ca --- /dev/null +++ b/designatedashboard/dashboards/project/dns_domains/templates/dns_domains/_create_domain.html @@ -0,0 +1,38 @@ +{% extends "horizon/common/_modal_form.html" %} +{% load i18n horizon humanize %} + +{% block form_id %}{% endblock %} +{% block form_action %}{% url 'horizon:project:dns_domains:create_domain' %}{% endblock %} + +{% block modal_id %}create_domain_modal{% endblock %} +{% block modal-header %}{% trans "Create Domain" %}{% endblock %} + +{% block modal-body %} +
+
+ {% include "horizon/common/_form_fields.html" %} +
+
+ +
+

{% trans "Description" %}:

+

{% blocktrans %} + The Name field should contain a full-qualified domain name (with + trailing period). + {% endblocktrans %}

+

{% blocktrans %} + The Email field should contain a valid email address to be associated + with the domain. + {% endblocktrans %}

+

{% blocktrans %} + The optional TTL field can be any value between 1 and 2147483647 + seconds. + {% endblocktrans %}

+
+ +{% endblock %} + +{% block modal-footer %} + + {% trans "Cancel" %} +{% endblock %} diff --git a/designatedashboard/dashboards/project/dns_domains/templates/dns_domains/_create_record.html b/designatedashboard/dashboards/project/dns_domains/templates/dns_domains/_create_record.html new file mode 100644 index 0000000..197443b --- /dev/null +++ b/designatedashboard/dashboards/project/dns_domains/templates/dns_domains/_create_record.html @@ -0,0 +1,37 @@ +{% extends "horizon/common/_modal_form.html" %} +{% load i18n horizon humanize %} + +{% block form_id %}{% endblock %} +{% block form_action %}{% url 'horizon:project:dns_domains:create_record' domain.id %}{% endblock %} + +{% block modal_id %}create_record_modal{% endblock %} +{% block modal-header %}{% trans "Create Record for" %} {{ domain.name }}{% endblock %} + +{% block modal-body %} + +
+ {% include 'project/dns_domains/prefix_field_style.html' %} +
+ {% include "horizon/common/_form_fields.html" %} +
+
+ + {% blocktrans %} +

+ TTL + The TTL is the time-to-live for the record, in seconds. +

+

+ See more info on record types. +

+ {% endblocktrans %} + +{% endblock %} diff --git a/designatedashboard/dashboards/project/dns_domains/templates/dns_domains/_domain_detail.html b/designatedashboard/dashboards/project/dns_domains/templates/dns_domains/_domain_detail.html new file mode 100644 index 0000000..2643938 --- /dev/null +++ b/designatedashboard/dashboards/project/dns_domains/templates/dns_domains/_domain_detail.html @@ -0,0 +1,32 @@ +{% load i18n sizeformat %} + +

{% trans "Domain Overview" %}

+ +
+
+
{% trans "ID" %}
+
{{ domain.id|default:_("None") }}
+
{% trans "Name" %}
+
{{ domain.name|default:_("None") }}
+
{% trans "Description" %}
+
{{ domain.description|default:_("None") }}
+
{% trans "Serial" %}
+
{{ domain.serial|yesno|capfirst }}
+
{% trans "Email" %}
+
{{ domain.email|default:_("Unknown") }}
+
{% trans "TTL" %}
+
{{ domain.ttl|default:_("Unknown") }}
+
{% trans "Created" %}
+ {% if domain.created_at %} +
{{ domain.created_at|parse_isotime }}
+ {% else %} +
{% trans "Unknown" %}
+ {% endif %} +
{% trans "Updated" %}
+ {% if domain.updated_at %} +
{{ domain.updated_at|parse_isotime }}
+ {% else %} +
{% trans "Unknown" %}
+ {% endif %} +
+
diff --git a/designatedashboard/dashboards/project/dns_domains/templates/dns_domains/_record_detail.html b/designatedashboard/dashboards/project/dns_domains/templates/dns_domains/_record_detail.html new file mode 100644 index 0000000..eb86adc --- /dev/null +++ b/designatedashboard/dashboards/project/dns_domains/templates/dns_domains/_record_detail.html @@ -0,0 +1,36 @@ +{% load i18n sizeformat %} + +

{% trans "All Records" %}

+ +

{{ record.name|default:_("None") }}

+ +
+
+
{% trans "Name" %}
+
{{ record.name|default:_("None") }}
+
{% trans "ID" %}
+
{{ record.id|default:_("None") }}
+
{% trans "Type" %}
+
{{ record.type|default:_("Unknown") }}
+
{% trans "Description" %}
+
{{ record.description|default:_("None") }}
+
{% trans "Record Data" %}
+
{{ record.data|default:_("None") }}
+
{% trans "Priority" %}
+
{{ record.priority|yesno|capfirst }}
+
{% trans "TTL" %}
+
{{ record.ttl|default:_("None") }}
+
{% trans "Created" %}
+ {% if record.created_at %} +
{{ record.created_at|parse_isotime }}
+ {% else %} +
{% trans "Unknown" %}
+ {% endif %} +
{% trans "Updated" %}
+ {% if record.updated_at %} +
{{ record.updated_at|parse_isotime }}
+ {% else %} +
{% trans "Unknown" %}
+ {% endif %} +
+
diff --git a/designatedashboard/dashboards/project/dns_domains/templates/dns_domains/_update_domain.html b/designatedashboard/dashboards/project/dns_domains/templates/dns_domains/_update_domain.html new file mode 100644 index 0000000..e8f6ba9 --- /dev/null +++ b/designatedashboard/dashboards/project/dns_domains/templates/dns_domains/_update_domain.html @@ -0,0 +1,36 @@ +{% extends "horizon/common/_modal_form.html" %} +{% load i18n %} +{% load url from future %} + +{% block form_id %}update_domain_form{% endblock %} +{% block form_action %}{% url 'horizon:project:dns_domains:update_domain' domain.id %}{% endblock %} + +{% block modal-header %}{% trans "Update Domain" %}{% endblock %} + +{% block modal-body %} +
+
+ {% include "horizon/common/_form_fields.html" %} +
+
+ +
+

{% trans "Description" %}:

+

{% blocktrans %} + From here you can edit the email address and TTL associated with a domain. + {% endblocktrans %}

+

{% blocktrans %} + The Email field should contain a valid email address to be associated + with the domain. + {% endblocktrans %}

+

{% blocktrans %} + The optional TTL field can be any value between 1 and 2147483647 + seconds. + {% endblocktrans %}

+
+{% endblock %} + +{% block modal-footer %} + + {% trans "Cancel" %} +{% endblock %} diff --git a/designatedashboard/dashboards/project/dns_domains/templates/dns_domains/_update_record.html b/designatedashboard/dashboards/project/dns_domains/templates/dns_domains/_update_record.html new file mode 100644 index 0000000..4344f36 --- /dev/null +++ b/designatedashboard/dashboards/project/dns_domains/templates/dns_domains/_update_record.html @@ -0,0 +1,9 @@ +{% extends "project/dns_domains/_create_record.html" %} +{% load i18n %} +{% load url from future %} + +{% block form_id %}update_record_form{% endblock %} +{% block form_action %}{% url 'horizon:project:dns_domains:update_record' record.domain_id record.id %}{% endblock %} + +{% block modal-header %}{% trans "Update Domain Record" %}{% endblock %} + diff --git a/designatedashboard/dashboards/project/dns_domains/templates/dns_domains/create_domain.html b/designatedashboard/dashboards/project/dns_domains/templates/dns_domains/create_domain.html new file mode 100644 index 0000000..fe6380c --- /dev/null +++ b/designatedashboard/dashboards/project/dns_domains/templates/dns_domains/create_domain.html @@ -0,0 +1,11 @@ +{% extends 'base.html' %} +{% load i18n %} +{% block title %}{% trans "Create Domain" %}{% endblock %} + +{% block page_header %} + {% include "horizon/common/_page_header.html" with title=_("Create Domain") %} +{% endblock page_header %} + +{% block main %} + {% include 'project/dns_domains/_create_domain.html' %} +{% endblock %} diff --git a/designatedashboard/dashboards/project/dns_domains/templates/dns_domains/create_record.html b/designatedashboard/dashboards/project/dns_domains/templates/dns_domains/create_record.html new file mode 100644 index 0000000..4414f32 --- /dev/null +++ b/designatedashboard/dashboards/project/dns_domains/templates/dns_domains/create_record.html @@ -0,0 +1,11 @@ +{% extends 'base.html' %} +{% load i18n %} +{% block title %}{% trans "Create Domain Record" %}{% endblock %} + +{% block page_header %} + {% include "horizon/common/_page_header.html" with title=_("Create Domain Record") %} +{% endblock page_header %} + +{% block main %} + {% include 'project/dns_domains/_create_record.html' %} +{% endblock %} diff --git a/designatedashboard/dashboards/project/dns_domains/templates/dns_domains/domain_detail.html b/designatedashboard/dashboards/project/dns_domains/templates/dns_domains/domain_detail.html new file mode 100644 index 0000000..1137660 --- /dev/null +++ b/designatedashboard/dashboards/project/dns_domains/templates/dns_domains/domain_detail.html @@ -0,0 +1,11 @@ +{% extends 'base.html' %} +{% load i18n %} +{% block title %}{% trans 'Domain Detail' %}{% endblock %} + +{% block page_header %} + {% include "horizon/common/_page_header.html" with title=_("Domain") %} +{% endblock page_header %} + +{% block main %} + {% include 'project/dns_domains/_domain_detail.html' %} +{% endblock %} diff --git a/designatedashboard/dashboards/project/dns_domains/templates/dns_domains/index.html b/designatedashboard/dashboards/project/dns_domains/templates/dns_domains/index.html new file mode 100644 index 0000000..a7d629b --- /dev/null +++ b/designatedashboard/dashboards/project/dns_domains/templates/dns_domains/index.html @@ -0,0 +1,11 @@ +{% extends 'base.html' %} +{% load i18n %} +{% block title %}{% trans "Domains" %}{% endblock %} + +{% block page_header %} + {% include "horizon/common/_page_header.html" with title=_("Domains") %} +{% endblock page_header %} + +{% block main %} + {{ table.render }} +{% endblock %} diff --git a/designatedashboard/dashboards/project/dns_domains/templates/dns_domains/prefix_field_style.html b/designatedashboard/dashboards/project/dns_domains/templates/dns_domains/prefix_field_style.html new file mode 100644 index 0000000..90078a1 --- /dev/null +++ b/designatedashboard/dashboards/project/dns_domains/templates/dns_domains/prefix_field_style.html @@ -0,0 +1,4 @@ + diff --git a/designatedashboard/dashboards/project/dns_domains/templates/dns_domains/prefix_html_widget.html b/designatedashboard/dashboards/project/dns_domains/templates/dns_domains/prefix_html_widget.html new file mode 100644 index 0000000..78f1154 --- /dev/null +++ b/designatedashboard/dashboards/project/dns_domains/templates/dns_domains/prefix_html_widget.html @@ -0,0 +1,2 @@ + +{{ input }} diff --git a/designatedashboard/dashboards/project/dns_domains/templates/dns_domains/record_detail.html b/designatedashboard/dashboards/project/dns_domains/templates/dns_domains/record_detail.html new file mode 100644 index 0000000..bb94f9f --- /dev/null +++ b/designatedashboard/dashboards/project/dns_domains/templates/dns_domains/record_detail.html @@ -0,0 +1,11 @@ +{% extends 'base.html' %} +{% load i18n %} +{% block title %}{% trans 'Record Detail' %}{% endblock %} + +{% block page_header %} + {% include "horizon/common/_page_header.html" with title="Record Detail" %} +{% endblock page_header %} + +{% block main %} + {% include 'project/dns_domains/_record_detail.html' %} +{% endblock %} diff --git a/designatedashboard/dashboards/project/dns_domains/templates/dns_domains/records.html b/designatedashboard/dashboards/project/dns_domains/templates/dns_domains/records.html new file mode 100644 index 0000000..b8de38c --- /dev/null +++ b/designatedashboard/dashboards/project/dns_domains/templates/dns_domains/records.html @@ -0,0 +1,32 @@ +{% extends 'base.html' %} +{% load i18n %} +{% block title %}{% trans 'Domain Records' %}{% endblock %} + +{% block page_header %} + {% include "horizon/common/_page_header.html" with title=_("Domain Records") %} +{% endblock page_header %} + +{% block main %} +
+
+
+

+ {% trans "Domains" %} : {{ domain.name }} → + {% trans "Records" %} +

+
+
+ × +
+
+
+

{% trans "Nameservers" %}

+
    + {% for server in servers %} +
  • {{ server.name }}
  • + {% endfor %} +
+
+ + {{ table.render }} +{% endblock %} diff --git a/designatedashboard/dashboards/project/dns_domains/templates/dns_domains/update_domain.html b/designatedashboard/dashboards/project/dns_domains/templates/dns_domains/update_domain.html new file mode 100644 index 0000000..e1a2752 --- /dev/null +++ b/designatedashboard/dashboards/project/dns_domains/templates/dns_domains/update_domain.html @@ -0,0 +1,11 @@ +{% extends 'base.html' %} +{% load i18n %} +{% block title %}{% trans 'Update Domain' %}{% endblock %} + +{% block page_header %} + {% include "horizon/common/_page_header.html" with title="Domain" %} +{% endblock page_header %} + +{% block main %} + {% include 'project/dns_domains/_update_domain.html' %} +{% endblock %} diff --git a/designatedashboard/dashboards/project/dns_domains/templates/dns_domains/update_record.html b/designatedashboard/dashboards/project/dns_domains/templates/dns_domains/update_record.html new file mode 100644 index 0000000..ca2b7a7 --- /dev/null +++ b/designatedashboard/dashboards/project/dns_domains/templates/dns_domains/update_record.html @@ -0,0 +1,11 @@ +{% extends 'base.html' %} +{% load i18n %} +{% block title %}{% trans 'Update Domain Record' %}{% endblock %} + +{% block page_header %} + {% include "horizon/common/_page_header.html" with title="Domain Record" %} +{% endblock page_header %} + +{% block main %} + {% include 'project/dns_domains/_update_record.html' %} +{% endblock %} diff --git a/designatedashboard/dashboards/project/dns_domains/urls.py b/designatedashboard/dashboards/project/dns_domains/urls.py new file mode 100644 index 0000000..1a2231b --- /dev/null +++ b/designatedashboard/dashboards/project/dns_domains/urls.py @@ -0,0 +1,52 @@ +# Copyright 2013 Hewlett-Packard Development Company, L.P. +# +# Licensed under the Apache License, Version 2.0 (the "License"); you may +# not use this file except in compliance with the License. You may obtain +# a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT +# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the +# License for the specific language governing permissions and limitations +# under the License. +from django.conf.urls import url, patterns # noqa + +from .views import CreateDomainView # noqa +from .views import CreateRecordView # noqa +from .views import DomainDetailView # noqa +from .views import IndexView # noqa +from .views import RecordsView # noqa +from .views import UpdateDomainView # noqa +from .views import UpdateRecordView # noqa +from .views import ViewRecordDetailsView # noqa + + +urlpatterns = patterns( + '', + url(r'^$', + IndexView.as_view(), + name='index'), + url(r'^create/$', + CreateDomainView.as_view(), + name='create_domain'), + url(r'^(?P[^/]+)/update$', + UpdateDomainView.as_view(), + name='update_domain'), + url(r'^(?P[^/]+)$', + DomainDetailView.as_view(), + name='domain_detail'), + url(r'^(?P[^/]+)/records$', + RecordsView.as_view(), + name='records'), + url(r'^(?P[^/]+)/records/create$', + CreateRecordView.as_view(), + name='create_record'), + url(r'^(?P[^/]+)/records/(?P[^/]+)/update$', + UpdateRecordView.as_view(), + name='update_record'), + url(r'^(?P[^/]+)/records/(?P[^/]+)/$', + ViewRecordDetailsView.as_view(), + name='view_record'), +) diff --git a/designatedashboard/dashboards/project/dns_domains/utils.py b/designatedashboard/dashboards/project/dns_domains/utils.py new file mode 100644 index 0000000..eaa74c7 --- /dev/null +++ b/designatedashboard/dashboards/project/dns_domains/utils.py @@ -0,0 +1,20 @@ +# Licensed under the Apache License, Version 2.0 (the "License"); you may +# not use this file except in compliance with the License. You may obtain +# a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT +# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the +# License for the specific language governing permissions and limitations +# under the License. +from django.conf import settings + + +def limit_records_to_fips(): + # This method checks the settings to determine if the + # record creation / update screen should limit the ip input + # to be a dropdown of floating ips + return getattr(settings, "DESIGNATE", + {}).get("records_use_fips", False) diff --git a/designatedashboard/dashboards/project/dns_domains/views.py b/designatedashboard/dashboards/project/dns_domains/views.py new file mode 100644 index 0000000..287bba2 --- /dev/null +++ b/designatedashboard/dashboards/project/dns_domains/views.py @@ -0,0 +1,243 @@ +# Copyright 2013 Hewlett-Packard Development Company, L.P. +# +# Licensed under the Apache License, Version 2.0 (the "License"); you may +# not use this file except in compliance with the License. You may obtain +# a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT +# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the +# License for the specific language governing permissions and limitations +# under the License. +from django.core.urlresolvers import reverse, reverse_lazy # noqa +from django.utils.translation import ugettext_lazy as _ # noqa + +from horizon import exceptions +from horizon import forms +from horizon import tables +from horizon.views import HorizonTemplateView # noqa + +from openstack_dashboard.api.network import tenant_floating_ip_list +from openstack_dashboard.api.nova import server_list + +from designatedashboard import api +from designatedashboard.api import rest # noqa + +from .forms import DomainCreate # noqa +from .forms import DomainUpdate # noqa +from .forms import RecordCreate # noqa +from .forms import RecordUpdate # noqa +from .tables import DomainsTable # noqa +from .tables import RecordsTable # noqa +from .utils import limit_records_to_fips # noqa + + +class IndexView(tables.DataTableView): + table_class = DomainsTable + template_name = 'project/dns_domains/index.html' + + def get_data(self): + try: + return api.designate.domain_list(self.request) + except Exception: + exceptions.handle(self.request, + _('Unable to retrieve domain list.')) + return [] + + +class CreateDomainView(forms.ModalFormView): + form_class = DomainCreate + template_name = 'project/dns_domains/create_domain.html' + success_url = reverse_lazy('horizon:project:dns_domains:index') + + def get_object_display(self, obj): + return obj.ip + + +class DomainDetailView(HorizonTemplateView): + template_name = 'project/dns_domains/domain_detail.html' + + def get_context_data(self, **kwargs): + context = super(DomainDetailView, self).get_context_data(**kwargs) + domain_id = self.kwargs['domain_id'] + try: + context["domain"] = api.designate.domain_get(self.request, + domain_id) + table = DomainsTable(self.request) + context["actions"] = table.render_row_actions(context["domain"]) + except Exception: + redirect = reverse('horizon:project:dns_domains:index') + exceptions.handle(self.request, + _('Unable to retrieve domain record.'), + redirect=redirect) + return context + + +class UpdateDomainView(forms.ModalFormView): + form_class = DomainUpdate + template_name = 'project/dns_domains/update_domain.html' + success_url = reverse_lazy('horizon:project:dns_domains:index') + + def get_object(self): + domain_id = self.kwargs['domain_id'] + try: + return api.designate.domain_get(self.request, domain_id) + except Exception: + redirect = reverse('horizon:project:dns_domains:index') + exceptions.handle(self.request, + _('Unable to retrieve domain record.'), + redirect=redirect) + + def get_initial(self): + self.domain = self.get_object() + return self.domain + + def get_context_data(self, **kwargs): + context = super(UpdateDomainView, self).get_context_data(**kwargs) + context["domain"] = self.domain + return context + + +class RecordsView(tables.DataTableView): + table_class = RecordsTable + template_name = 'project/dns_domains/records.html' + + def get_data(self): + domain_id = self.kwargs['domain_id'] + records = [] + try: + self.domain = api.designate.domain_get(self.request, domain_id) + self.servers = api.designate.server_list(self.request, domain_id) + records = api.designate.record_list(self.request, domain_id) + except Exception: + redirect = reverse('horizon:project:dns_domains:index') + exceptions.handle(self.request, + _('Unable to retrieve record list.'), + redirect=redirect) + + return records + + def get_context_data(self, **kwargs): + context = super(RecordsView, self).get_context_data(**kwargs) + context['domain'] = self.domain + context['servers'] = self.servers + + return context + + +class BaseRecordFormView(forms.ModalFormView): + cancel_label = _("Cancel") + + def get_success_url(self): + return reverse('horizon:project:dns_domains:records', + args=(self.kwargs['domain_id'],)) + + def get_domain(self): + domain_id = self.kwargs['domain_id'] + try: + return api.designate.domain_get(self.request, domain_id) + except Exception: + redirect = reverse('horizon:project:dns_domains:records', + args=(self.kwargs['domain_id'],)) + exceptions.handle(self.request, + ('Unable to retrieve domain record.'), + redirect=redirect) + # NotAuthorized errors won't be redirected automatically. Need + # to force the issue + raise exceptions.Http302(redirect) + + def get_initial(self): + self.domain = self.get_domain() + results = {'domain_id': self.domain.id, + 'domain_name': self.domain.name, } + if limit_records_to_fips(): + results.update({'fips': tenant_floating_ip_list(self.request), + 'instances': server_list(self.request)[0]}) + return results + + def get_context_data(self, **kwargs): + """Set the cancel url + + the cancel_url needs a variable in it + so we cannot do this with a simple class attr + this is critical to perform before the super.get_context_data + """ + self.cancel_url = reverse('horizon:project:dns_domains:records', + args=(self.kwargs['domain_id'],)) + context = super(BaseRecordFormView, self).get_context_data(**kwargs) + context['domain'] = self.domain + return context + + +class CreateRecordView(BaseRecordFormView): + form_class = RecordCreate + submit_label = _("Create Record") + template_name = 'project/dns_domains/create_record.html' + + +class ViewRecordDetailsView(HorizonTemplateView): + template_name = 'project/dns_domains/record_detail.html' + + def get_record(self): + domain_id = self.kwargs['domain_id'] + record_id = self.kwargs['record_id'] + try: + return api.designate.record_get(self.request, domain_id, record_id) + except Exception: + redirect = reverse('horizon:project:dns_domains:records', + args=(self.kwargs['domain_id'],)) + exceptions.handle(self.request, + _('Unable to retrieve domain record.'), + redirect=redirect) + + def get_context_data(self, **kwargs): + context = super(ViewRecordDetailsView, self).get_context_data(**kwargs) + self.record = self.get_record() + context["record"] = self.record + context["domain_id"] = self.kwargs['domain_id'] + return context + + +class UpdateRecordView(BaseRecordFormView): + form_class = RecordUpdate + submit_label = _("Update Record") + template_name = 'project/dns_domains/update_record.html' + + def get_record(self): + domain_id = self.kwargs['domain_id'] + record_id = self.kwargs['record_id'] + + try: + return api.designate.record_get(self.request, domain_id, record_id) + except Exception: + redirect = reverse('horizon:project:dns_domains:records', + args=(self.kwargs['domain_id'],)) + exceptions.handle(self.request, + _('Unable to retrieve domain record.'), + redirect=redirect) + + def get_initial(self): + initial = super(UpdateRecordView, self).get_initial() + self.record = self.get_record() + + initial.update({ + 'id': self.record.id, + 'name': self.record.name.replace("." + initial['domain_name'], ''), + 'data': self.record.data, + 'txt': self.record.data, + 'priority': self.record.priority, + 'ttl': self.record.ttl, + 'type': self.record.type.lower(), + 'description': self.record.description, + }) + if limit_records_to_fips(): + initial.update({'ip_addr': self.record.data}) + + return initial + + def get_context_data(self, **kwargs): + context = super(UpdateRecordView, self).get_context_data(**kwargs) + context["record"] = self.record + return context diff --git a/designatedashboard/dashboards/project/ngdns/__init__.py b/designatedashboard/dashboards/project/ngdns/__init__.py new file mode 100644 index 0000000..2c57778 --- /dev/null +++ b/designatedashboard/dashboards/project/ngdns/__init__.py @@ -0,0 +1 @@ +from designatedashboard.api import rest # noqa diff --git a/designatedashboard/dashboards/project/ngdns/reverse_dns/__init__.py b/designatedashboard/dashboards/project/ngdns/reverse_dns/__init__.py new file mode 100644 index 0000000..e69de29 diff --git a/designatedashboard/dashboards/project/ngdns/reverse_dns/panel.py b/designatedashboard/dashboards/project/ngdns/reverse_dns/panel.py new file mode 100644 index 0000000..41f0c7c --- /dev/null +++ b/designatedashboard/dashboards/project/ngdns/reverse_dns/panel.py @@ -0,0 +1,25 @@ +# (c) Copyright 2015 Hewlett-Packard Development Company, L.P. +# +# Licensed under the Apache License, Version 2.0 (the "License"); you may +# not use this file except in compliance with the License. You may obtain +# a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT +# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the +# License for the specific language governing permissions and limitations +# under the License. +from django.utils.translation import ugettext_lazy as _ + +import horizon +from openstack_dashboard.dashboards.project import dashboard + + +class ReverseDns(horizon.Panel): + name = _("Reverse DNS") + slug = 'reverse_dns' + permissions = ('openstack.services.dns',) + +dashboard.Project.register(ReverseDns) diff --git a/designatedashboard/dashboards/project/ngdns/reverse_dns/urls.py b/designatedashboard/dashboards/project/ngdns/reverse_dns/urls.py new file mode 100644 index 0000000..7977633 --- /dev/null +++ b/designatedashboard/dashboards/project/ngdns/reverse_dns/urls.py @@ -0,0 +1,22 @@ +# (c) Copyright 2015 Hewlett-Packard Development Company, L.P. +# +# Licensed under the Apache License, Version 2.0 (the "License"); you may +# not use this file except in compliance with the License. You may obtain +# a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT +# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the +# License for the specific language governing permissions and limitations +# under the License. + +from django.conf.urls import url + +from designatedashboard.dashboards.project.ngdns.reverse_dns import views + + +urlpatterns = [ + url('', views.IndexView.as_view(), name='index'), +] diff --git a/designatedashboard/dashboards/project/ngdns/reverse_dns/views.py b/designatedashboard/dashboards/project/ngdns/reverse_dns/views.py new file mode 100644 index 0000000..e072aea --- /dev/null +++ b/designatedashboard/dashboards/project/ngdns/reverse_dns/views.py @@ -0,0 +1,19 @@ +# (c) Copyright 2015 Hewlett-Packard Development Company, L.P. +# +# Licensed under the Apache License, Version 2.0 (the "License"); you may +# not use this file except in compliance with the License. You may obtain +# a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT +# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the +# License for the specific language governing permissions and limitations +# under the License. + +from django.views import generic + + +class IndexView(generic.TemplateView): + template_name = 'angular.html' diff --git a/designatedashboard/dashboards/project/ngdns/zones/__init__.py b/designatedashboard/dashboards/project/ngdns/zones/__init__.py new file mode 100644 index 0000000..e69de29 diff --git a/designatedashboard/dashboards/project/ngdns/zones/panel.py b/designatedashboard/dashboards/project/ngdns/zones/panel.py new file mode 100644 index 0000000..e3fdaba --- /dev/null +++ b/designatedashboard/dashboards/project/ngdns/zones/panel.py @@ -0,0 +1,25 @@ +# (c) Copyright 2015 Hewlett-Packard Development Company, L.P. +# +# Licensed under the Apache License, Version 2.0 (the "License"); you may +# not use this file except in compliance with the License. You may obtain +# a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT +# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the +# License for the specific language governing permissions and limitations +# under the License. +from django.utils.translation import ugettext_lazy as _ + +import horizon +from openstack_dashboard.dashboards.project import dashboard + + +class Zones(horizon.Panel): + name = _("Zones") + slug = 'dnszones' + permissions = ('openstack.services.dns',) + +dashboard.Project.register(Zones) diff --git a/designatedashboard/dashboards/project/ngdns/zones/urls.py b/designatedashboard/dashboards/project/ngdns/zones/urls.py new file mode 100644 index 0000000..0eacf22 --- /dev/null +++ b/designatedashboard/dashboards/project/ngdns/zones/urls.py @@ -0,0 +1,22 @@ +# (c) Copyright 2015 Hewlett-Packard Development Company, L.P. +# +# Licensed under the Apache License, Version 2.0 (the "License"); you may +# not use this file except in compliance with the License. You may obtain +# a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT +# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the +# License for the specific language governing permissions and limitations +# under the License. + +from django.conf.urls import url + +from designatedashboard.dashboards.project.ngdns.zones import views + + +urlpatterns = [ + url('', views.IndexView.as_view(), name='index'), +] diff --git a/designatedashboard/dashboards/project/ngdns/zones/views.py b/designatedashboard/dashboards/project/ngdns/zones/views.py new file mode 100644 index 0000000..e072aea --- /dev/null +++ b/designatedashboard/dashboards/project/ngdns/zones/views.py @@ -0,0 +1,19 @@ +# (c) Copyright 2015 Hewlett-Packard Development Company, L.P. +# +# Licensed under the Apache License, Version 2.0 (the "License"); you may +# not use this file except in compliance with the License. You may obtain +# a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT +# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the +# License for the specific language governing permissions and limitations +# under the License. + +from django.views import generic + + +class IndexView(generic.TemplateView): + template_name = 'angular.html' diff --git a/designatedashboard/enabled/_1710_project_dns_panel_group.py b/designatedashboard/enabled/_1710_project_dns_panel_group.py new file mode 100644 index 0000000..4473e40 --- /dev/null +++ b/designatedashboard/enabled/_1710_project_dns_panel_group.py @@ -0,0 +1,20 @@ +# Copyright 2013 Hewlett-Packard Development Company, L.P. +# +# Licensed under the Apache License, Version 2.0 (the "License"); you may +# not use this file except in compliance with the License. You may obtain +# a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT +# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the +# License for the specific language governing permissions and limitations +# under the License. + +# The name of the panel group to be added to HORIZON_CONFIG. Required. +PANEL_GROUP = 'dns' +# The display name of the PANEL_GROUP. Required. +PANEL_GROUP_NAME = 'DNS' +# The name of the dashboard the PANEL_GROUP associated with. Required. +PANEL_GROUP_DASHBOARD = 'project' diff --git a/designatedashboard/enabled/_1720_project_dns_panel.py b/designatedashboard/enabled/_1720_project_dns_panel.py new file mode 100644 index 0000000..c1494d8 --- /dev/null +++ b/designatedashboard/enabled/_1720_project_dns_panel.py @@ -0,0 +1,36 @@ +# Copyright 2013 Hewlett-Packard Development Company, L.P. +# +# Licensed under the Apache License, Version 2.0 (the "License"); you may +# not use this file except in compliance with the License. You may obtain +# a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT +# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the +# License for the specific language governing permissions and limitations +# under the License. + +from designatedashboard import exceptions + +# The name of the panel to be added to HORIZON_CONFIG. Required. +PANEL = 'domains' +# The name of the dashboard the PANEL associated with. Required. +PANEL_DASHBOARD = 'project' +# The name of the panel group the PANEL is associated with. +PANEL_GROUP = 'dns' + +ADD_INSTALLED_APPS = ['designatedashboard'] + +ADD_EXCEPTIONS = { + 'recoverable': exceptions.RECOVERABLE, + 'not_found': exceptions.NOT_FOUND, + 'unauthorized': exceptions.UNAUTHORIZED, +} + +# Python panel class of the PANEL to be added. +ADD_PANEL = ( + 'designatedashboard.dashboards.project.dns_domains.panel.DNSDomains') + +DISABLED = True diff --git a/designatedashboard/enabled/_1721_dns_zones_panel.py b/designatedashboard/enabled/_1721_dns_zones_panel.py new file mode 100644 index 0000000..da72368 --- /dev/null +++ b/designatedashboard/enabled/_1721_dns_zones_panel.py @@ -0,0 +1,40 @@ +# Copyright 2013 Hewlett-Packard Development Company, L.P. +# +# Licensed under the Apache License, Version 2.0 (the "License"); you may +# not use this file except in compliance with the License. You may obtain +# a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT +# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the +# License for the specific language governing permissions and limitations +# under the License. + +from designatedashboard import exceptions + +# The name of the panel to be added to HORIZON_CONFIG. Required. +PANEL = 'dnszones' +# The name of the dashboard the PANEL associated with. Required. +PANEL_DASHBOARD = 'project' +# The name of the panel group the PANEL is associated with. +PANEL_GROUP = 'dns' + +ADD_EXCEPTIONS = { + 'recoverable': exceptions.RECOVERABLE, + 'not_found': exceptions.NOT_FOUND, + 'unauthorized': exceptions.UNAUTHORIZED, +} + +ADD_INSTALLED_APPS = ['designatedashboard'] + +# Python panel class of the PANEL to be added. +ADD_PANEL = ( + 'designatedashboard.dashboards.project.ngdns.zones.panel.Zones') + +ADD_ANGULAR_MODULES = ['designatedashboard'] + +ADD_SCSS_FILES = ['designatedashboard/designatedashboard.scss'] + +AUTO_DISCOVER_STATIC_FILES = True diff --git a/designatedashboard/enabled/_1722_dns_reversedns_panel.py b/designatedashboard/enabled/_1722_dns_reversedns_panel.py new file mode 100644 index 0000000..1728715 --- /dev/null +++ b/designatedashboard/enabled/_1722_dns_reversedns_panel.py @@ -0,0 +1,38 @@ +# Copyright 2013 Hewlett-Packard Development Company, L.P. +# +# Licensed under the Apache License, Version 2.0 (the "License"); you may +# not use this file except in compliance with the License. You may obtain +# a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT +# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the +# License for the specific language governing permissions and limitations +# under the License. + +from designatedashboard import exceptions + +# The name of the panel to be added to HORIZON_CONFIG. Required. +PANEL = 'reverse_dns' +# The name of the dashboard the PANEL associated with. Required. +PANEL_DASHBOARD = 'project' +# The name of the panel group the PANEL is associated with. +PANEL_GROUP = 'dns' + +ADD_EXCEPTIONS = { + 'recoverable': exceptions.RECOVERABLE, + 'not_found': exceptions.NOT_FOUND, + 'unauthorized': exceptions.UNAUTHORIZED, +} + +# Python panel class of the PANEL to be added. +ADD_PANEL = ( + 'designatedashboard.dashboards.project.ngdns.reverse_dns.panel.ReverseDns') + +ADD_ANGULAR_MODULES = ['designatedashboard'] + +ADD_SCSS_FILES = ['designatedashboard/designatedashboard.scss'] + +AUTO_DISCOVER_STATIC_FILES = True diff --git a/designatedashboard/enabled/__init__.py b/designatedashboard/enabled/__init__.py new file mode 100644 index 0000000..e69de29 diff --git a/designatedashboard/exceptions.py b/designatedashboard/exceptions.py new file mode 100644 index 0000000..677fba6 --- /dev/null +++ b/designatedashboard/exceptions.py @@ -0,0 +1,29 @@ +# Copyright (c) 2014 Rackspace Hosting. +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or +# implied. +# See the License for the specific language governing permissions and +# limitations under the License. +from designateclient import exceptions as designateclient + +from openstack_dashboard import exceptions + +NOT_FOUND = exceptions.NOT_FOUND + ( + designateclient.ResourceNotFound, + designateclient.NotFound, + ) +RECOVERABLE = exceptions.RECOVERABLE + ( + designateclient.BadRequest, + designateclient.Conflict, + ) +UNAUTHORIZED = exceptions.UNAUTHORIZED + ( + designateclient.Forbidden, + ) diff --git a/designatedashboard/locale/cs/LC_MESSAGES/django.mo b/designatedashboard/locale/cs/LC_MESSAGES/django.mo new file mode 100644 index 0000000000000000000000000000000000000000..b49d51a4915733806c247f36c651aae2deedcaca GIT binary patch literal 6965 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, 2016. #zanata +# Zbyněk Schwarz , 2017. #zanata +msgid "" +msgstr "" +"Project-Id-Version: designate-dashboard 6.0.0.0b2.dev3\n" +"Report-Msgid-Bugs-To: https://bugs.launchpad.net/openstack-i18n/\n" +"POT-Creation-Date: 2017-11-21 15:00+0000\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"PO-Revision-Date: 2017-11-17 07:24+0000\n" +"Last-Translator: Zbyněk Schwarz \n" +"Language-Team: Czech\n" +"Language: cs\n" +"X-Generator: Zanata 3.9.6\n" +"Plural-Forms: nplurals=3; plural=(n==1) ? 0 : (n>=2 && n<=4) ? 1 : 2\n" + +msgid "" +"\n" +" From here you can edit the email address and TTL associated with a " +"domain.\n" +" " +msgstr "" +"\n" +" Zde můžete upravit emailovou adresu a TTL patřící k doméně.\n" +" " + +msgid "" +"\n" +" The Email field should contain a valid email address to be associated\n" +" with the domain.\n" +" " +msgstr "" +"\n" +" Pole Email by mělo obsahovat platnou emailovou adresu, která je spojená\n" +" s doménou.\n" +" " + +msgid "" +"\n" +" The Name field should contain a full-qualified domain name (with\n" +" trailing period).\n" +" " +msgstr "" +"\n" +" Pole Název by mělo obsahovat plně kvalifikovaný název domény (s \n" +" tečkou na \n" +" konci)." + +msgid "" +"\n" +" The optional TTL field can be any value between 1 and 2147483647\n" +" seconds.\n" +" " +msgstr "" +"\n" +" TTL (volitelné) může být hodnota mezi 1 a 2147483647\n" +" sekund.\n" +" " + +msgid "" +"\n" +"

\n" +" TTL\n" +" The TTL is the time-to-live for the record, in seconds.\n" +"

\n" +"

\n" +" See more info on record types.\n" +"

\n" +" " +msgstr "" +"\n" +"

\n" +" TTL\n" +" TTL je time-to-live záznamu, v sekundách.\n" +"

\n" +"

\n" +" Viz více informací o typech záznamů.\n" +"

\n" +" " + +msgid "A - Address record" +msgstr "A - adresní záznam" + +msgid "AAAA - IPv6 address record" +msgstr "AAAA - adresní záznam IPv6" + +msgid "All Records" +msgstr "Všechny záznamy" + +msgid "CNAME - Canonical name record" +msgstr "CNAME - záznam kanonického jména" + +msgid "Cancel" +msgstr "Zrušit" + +msgid "Canonical Name" +msgstr "Kanonické jméno" + +msgid "Create Domain" +msgstr "Vytvořit doménu" + +msgid "Create Domain Record" +msgstr "Vytvořit doménový záznam" + +msgid "Create Record" +msgstr "Vytvořit záznam" + +msgid "Create Record for" +msgstr "Vytvořit záznam pro" + +msgid "Created" +msgstr "Vytvořeno" + +msgid "Created At" +msgstr "Vytvořeno" + +msgid "Data" +msgstr "Data" + +msgid "Delete" +msgstr "Smazat" + +msgid "Deleted" +msgstr "Smazáno" + +msgid "Description" +msgstr "Popis" + +msgid "Domain" +msgstr "Doména" + +#, python-format +msgid "Domain %(name)s created." +msgstr "Doména %(name)s vytvořena." + +#, python-format +msgid "Domain %(name)s updated." +msgstr "Doména %(name)s aktualizována." + +msgid "Domain Detail" +msgstr "Detail domény" + +msgid "Domain Name" +msgstr "Název domény" + +msgid "Domain Overview" +msgstr "Přehled domén" + +msgid "Domain Records" +msgstr "Doménové záznamy" + +#, python-format +msgid "Domain record %(name)s created." +msgstr "Doménový záznam %(name)s vytvořen." + +#, python-format +msgid "Domain record %(name)s updated." +msgstr "Doménový záznam %(name)s aktualizován." + +msgid "Domains" +msgstr "Domény" + +msgid "Edit Domain" +msgstr "Upravit doménu" + +msgid "Edit Record" +msgstr "Upravit záznam" + +msgid "Email" +msgstr "E-mail" + +msgid "Enter a valid IPv4 address" +msgstr "Zadejte platnou IPv4 adresu" + +msgid "Enter a valid IPv6 address" +msgstr "Zadejte platnou IPv6 adresu" + +msgid "Enter a valid SRV name" +msgstr "Zadejte platné SRV jméno" + +msgid "Enter a valid SRV record" +msgstr "Zadejte platný SRV záznam" + +msgid "Enter a valid SSHFP record" +msgstr "Zadejte platný SSHFP záznam" + +msgid "Enter a valid domain name." +msgstr "Zadejte platný název domény." + +msgid "Enter a valid hostname" +msgstr "Zadejte platné jméno hostitele" + +msgid "" +"Enter a valid hostname. The hostname should contain letters and numbers, and " +"be no more than 63 characters." +msgstr "" +"Zadejte platný název hostitele. Název hostitele by měl obsahovat písmena a " +"číslice a musí být maximálně 63 znaků." + +msgid "ID" +msgstr "ID" + +msgid "IP Address" +msgstr "IP adresa" + +msgid "MX - Mail exchange record" +msgstr "MX - mailový záznam" + +msgid "Mail Server" +msgstr "Mailový server" + +msgid "Manage Records" +msgstr "Spravovat záznamy" + +msgid "NS" +msgstr "NS" + +msgid "Name" +msgstr "Název" + +msgid "Name Server" +msgstr "Název serveru" + +msgid "Nameservers" +msgstr "Jmenné servery" + +msgid "None" +msgstr "Žádný" + +msgid "PTR - Pointer record" +msgstr "PTR - záznam ukazatele" + +msgid "PTR Domain Name" +msgstr "PTR doménové jméno" + +msgid "Priority" +msgstr "Priorita" + +msgid "Record" +msgstr "Záznam" + +msgid "Record Data" +msgstr "Zaznamenat data" + +msgid "Record Detail" +msgstr "Zaznamenat detail" + +msgid "Record Type" +msgstr "Typ záznamu" + +msgid "Records" +msgstr "Záznamy" + +msgid "Reverse DNS" +msgstr "Reverzní DNS" + +msgid "SOA" +msgstr "SOA" + +msgid "SPF - Sender Policy Framework" +msgstr "SPF - Sender Policy Framework" + +msgid "SRV - Service locator" +msgstr "SRV - lokátor služby" + +msgid "SSHFP - SSH Public Key Fingerprint" +msgstr "SSHFP - Otisk veřejného klíče SSH" + +msgid "Select an IP" +msgstr "Vyberte IP adresu" + +msgid "Serial" +msgstr "Sériový" + +msgid "TTL" +msgstr "TTL" + +msgid "TTL (seconds)" +msgstr "TTL (ve vteřinách)" + +msgid "TXT" +msgstr "TXT" + +msgid "TXT - Text record" +msgstr "TXT - textový záznam" + +msgid "Text" +msgstr "Text" + +msgid "The quotas could not be retrieved." +msgstr "Nelze získat kvóty." + +msgid "There are no floating IP addresses currently in use to select from." +msgstr "Žádné plovoucí IP nejsou k dispozici pro výběr." + +msgid "This field is required" +msgstr "Toto pole je povinné" + +msgid "Type" +msgstr "Typ" + +msgid "Unable to create domain." +msgstr "Nelze vytvořit doménu." + +msgid "Unable to create record." +msgstr "Nelze vytvořit záznam." + +msgid "Unable to retrieve domain list." +msgstr "Nelze získat seznam domén." + +msgid "Unable to retrieve domain record." +msgstr "Nelze získat doménový záznam." + +msgid "Unable to retrieve record list." +msgstr "Nelze získat záznamy." + +msgid "Unable to update domain." +msgstr "Nelze aktualizovat doménu." + +msgid "Unknown" +msgstr "Neznámé" + +msgid "Unknown instance name" +msgstr "Neznámé jméno instance" + +msgid "Update Domain" +msgstr "Aktualizovat doménu" + +msgid "Update Domain Record" +msgstr "Aktualizovat záznam domény" + +msgid "Update Record" +msgstr "Aktualizovat záznam" + +msgid "Updated" +msgstr "Aktualizováno" + +msgid "Updated At" +msgstr "Aktualizováno" + +msgid "Value" +msgstr "Hodnota" + +msgid "Zones" +msgstr "Zóny" diff --git a/designatedashboard/locale/de/LC_MESSAGES/django.mo b/designatedashboard/locale/de/LC_MESSAGES/django.mo new file mode 100644 index 0000000000000000000000000000000000000000..0f7d07ec9d4be8d897dad2293907f3899f83352f GIT binary patch literal 7264 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 E0m;vzVgLXD literal 0 HcmV?d00001 diff --git a/designatedashboard/locale/de/LC_MESSAGES/django.po b/designatedashboard/locale/de/LC_MESSAGES/django.po new file mode 100644 index 0000000..5c42e43 --- /dev/null +++ b/designatedashboard/locale/de/LC_MESSAGES/django.po @@ -0,0 +1,351 @@ +# Frank Kloeker , 2016. #zanata +# Robert Simai , 2016. #zanata +msgid "" +msgstr "" +"Project-Id-Version: designate-dashboard 3.0.0.0rc2.dev5\n" +"Report-Msgid-Bugs-To: https://bugs.launchpad.net/openstack-i18n/\n" +"POT-Creation-Date: 2016-09-29 13:19+0000\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"PO-Revision-Date: 2016-10-13 04:32+0000\n" +"Last-Translator: Andreas Jaeger \n" +"Language-Team: German\n" +"Language: de\n" +"X-Generator: Zanata 3.7.3\n" +"Plural-Forms: nplurals=2; plural=(n != 1)\n" + +msgid "" +"\n" +" From here you can edit the email address and TTL associated with a " +"domain.\n" +" " +msgstr "" +"\n" +" Hier können Sie die E-Mail-Adresse und TTL mit dazugehöriger Domäne " +"editieren.\n" +" " + +msgid "" +"\n" +" The Email field should contain a valid email address to be associated\n" +" with the domain.\n" +" " +msgstr "" +"\n" +" Das E-Mail-Feld sollte eine gültige E-Mail-Adresse enthalten,\n" +" die der Domäne zugewiesen wird.\n" +" " + +msgid "" +"\n" +" The Name field should contain a full-qualified domain name (with\n" +" trailing period).\n" +" " +msgstr "" +"\n" +" Das Namensfeld sollte einen vollqualifizierten Domänennamen enthalten " +"(mit\n" +" abschließendem Punkt).\n" +" " + +msgid "" +"\n" +" The optional TTL field can be any value between 1 and 2147483647\n" +" seconds.\n" +" " +msgstr "" +"\n" +" Das optionale TTL Feld kann einen Wert zwischen 1 und 2147483647\n" +" Sekunden haben.\n" +" " + +msgid "" +"\n" +"

\n" +" TTL\n" +" The TTL is the time-to-live for the record, in seconds.\n" +"

\n" +"

\n" +" See more info on record types.\n" +"

\n" +" " +msgstr "" +"\n" +"

\n" +" TTL\n" +" Die TTL ist die time-to-live für den Datensatz, in Sekunden.\n" +"

\n" +"

\n" +" Hier finden Sie mehr " +"Informationen zu Datensatztypen.\n" +"

\n" +" " + +msgid "A - Address record" +msgstr "A - Adress Datensatz" + +msgid "AAAA - IPv6 address record" +msgstr "AAAA - IPv6 Adress Datensatz" + +msgid "All Records" +msgstr "Alle Datensätze" + +msgid "CNAME - Canonical name record" +msgstr "CNAME - Canonical Name Datensatz" + +msgid "Cancel" +msgstr "Abbrechen" + +msgid "Canonical Name" +msgstr "Canonical Name" + +msgid "Create Domain" +msgstr "Domäne erstellen" + +msgid "Create Domain Record" +msgstr "Erzeuge Datensatz für Domäne" + +msgid "Create Record" +msgstr "Datensatz erzeugen" + +msgid "Create Record for" +msgstr "Erzeuge Datensatz für" + +msgid "Created" +msgstr "Erstellt" + +msgid "Created At" +msgstr "Erstellt am" + +msgid "Data" +msgstr "Daten" + +msgid "Delete" +msgstr "Löschen" + +msgid "Deleted" +msgstr "Gelöscht" + +msgid "Description" +msgstr "Beschreibung" + +msgid "Domain" +msgstr "Domäne" + +#, python-format +msgid "Domain %(name)s created." +msgstr "Domäne %(name)s erstellt." + +#, python-format +msgid "Domain %(name)s updated." +msgstr "Domäne %(name)s wurde geändert." + +msgid "Domain Detail" +msgstr "Domänen Details" + +msgid "Domain Name" +msgstr "Domänenname" + +msgid "Domain Overview" +msgstr "Domänenübersicht" + +msgid "Domain Records" +msgstr "Domänendatensatz" + +#, python-format +msgid "Domain record %(name)s created." +msgstr "Domändatensatz %(name)s erstellt." + +#, python-format +msgid "Domain record %(name)s updated." +msgstr "Domänendatensatz %(name)s aktualisiert." + +msgid "Domains" +msgstr "Domänen" + +msgid "Edit Domain" +msgstr "Domäne bearbeiten" + +msgid "Edit Record" +msgstr "Datensatze bearbeiten" + +msgid "Email" +msgstr "E-Mail" + +msgid "Enter a valid IPv4 address" +msgstr "Geben Sie eine gültige IPv4 Addresse ein" + +msgid "Enter a valid IPv6 address" +msgstr "Geben Sie eine gültige IPv6 Addresse ein" + +msgid "Enter a valid SRV name" +msgstr "Geben Sie einen gültigen SRV Namen ein" + +msgid "Enter a valid SRV record" +msgstr "Geben Sie einen gültigen SRV Datensatz ein" + +msgid "Enter a valid SSHFP record" +msgstr "Geben Sie einen gültigen SSHFP Datensatz ein" + +msgid "Enter a valid domain name." +msgstr "Geben Sie einen gültigen Domänennamen ein." + +msgid "Enter a valid hostname" +msgstr "Geben Sie einen gültigen Hostnamen ein" + +msgid "" +"Enter a valid hostname. The hostname should contain letters and numbers, and " +"be no more than 63 characters." +msgstr "" +"Geben Sie einen gültien Hostnamen ein. Der Hostname darf nur Buchstaben und " +"Nummern enthalten und darf nicht mehr als 63 Zeichen lang sein." + +msgid "ID" +msgstr "ID" + +msgid "IP Address" +msgstr "IP-Adresse" + +msgid "MX - Mail exchange record" +msgstr "MX - Mail Austausch Datensatz" + +msgid "Mail Server" +msgstr "Mailserver" + +msgid "Manage Records" +msgstr "Datensätze verwalten" + +msgid "NS" +msgstr "NS" + +msgid "Name" +msgstr "Name" + +msgid "Name Server" +msgstr "Nameserver" + +msgid "Nameservers" +msgstr "Namensserver" + +msgid "None" +msgstr "Keine" + +msgid "PTR - Pointer record" +msgstr "PTR - Pointer Datensatz" + +msgid "PTR Domain Name" +msgstr "PTR Domänenname" + +msgid "Priority" +msgstr "Priorität" + +msgid "Record" +msgstr "Datensatz" + +msgid "Record Data" +msgstr "Datensatz" + +msgid "Record Detail" +msgstr "Datensatzdetails" + +msgid "Record Type" +msgstr "Datensatztyp" + +msgid "Records" +msgstr "Datensätze" + +msgid "Reverse DNS" +msgstr "Reverse DNS" + +msgid "SOA" +msgstr "SOA" + +msgid "SPF - Sender Policy Framework" +msgstr "SPF - Sender Policy Framework" + +msgid "SRV - Service locator" +msgstr "SRV - Dienstezeiger" + +msgid "SSHFP - SSH Public Key Fingerprint" +msgstr "SSHFP - SSH Fingerprint für öffentlichen Schlüssel" + +msgid "Select an IP" +msgstr "Wähle eine IP" + +msgid "Serial" +msgstr "Seriennummer" + +msgid "TTL" +msgstr "TTL" + +msgid "TTL (seconds)" +msgstr "TTL (Sekunden)" + +msgid "TXT" +msgstr "TXT" + +msgid "TXT - Text record" +msgstr "TXT - Text Datensatz" + +msgid "Text" +msgstr "Text" + +msgid "The quotas could not be retrieved." +msgstr "Die Kontingente konnten nicht abgerufen werden." + +msgid "There are no floating IP addresses currently in use to select from." +msgstr "Es stehen derzeit keine Floating IP-Adressen zur Auswahl." + +msgid "This field is required" +msgstr "Dieses Feld ist erforderlich" + +msgid "Type" +msgstr "Typ" + +msgid "Unable to create domain." +msgstr "Die Domäne kann nicht erstellt werden." + +msgid "Unable to create record." +msgstr "Datensatz konnte nicht erzeugt werden." + +msgid "Unable to retrieve domain list." +msgstr "Domänen-Liste kann nicht abgerufen werden." + +msgid "Unable to retrieve domain record." +msgstr "Domänendatensatz kann nicht abgerufen werden." + +msgid "Unable to retrieve record list." +msgstr "Die Datensatzliste kann nicht abgerufen werden." + +msgid "Unable to update domain." +msgstr "Die Domäne kann nicht geändert werden." + +msgid "Unknown" +msgstr "Unbekannt" + +msgid "Unknown instance name" +msgstr "Unbekannter Instanzname" + +msgid "Update Domain" +msgstr "Domain ändern" + +msgid "Update Domain Record" +msgstr "Domänendatensatz aktualisieren" + +msgid "Update Record" +msgstr "Datensatz aktualisieren" + +msgid "Updated" +msgstr "Aktualisiert" + +msgid "Updated At" +msgstr "Aktualisiert am" + +msgid "Value" +msgstr "Wert" + +msgid "Zones" +msgstr "Zonen" diff --git a/designatedashboard/locale/en_GB/LC_MESSAGES/django.mo b/designatedashboard/locale/en_GB/LC_MESSAGES/django.mo new file mode 100644 index 0000000000000000000000000000000000000000..5d868462887f6f5487757ee90d97f3214c157176 GIT binary patch literal 6783 zcmeH~-ESR76~Lz@E##KcFQ6$J666Ji7rqqnJG--X z;*g3zz!RVQ+nF;vbLN~gvy(qOyzQ%s@(l7mTGU_noe-U~0mcfjw%`{B>vJK<}#{wMel^}j== zRQE6lkLrH74eo$4&f`$@yCHvSpKTw8JE%Wpc^1k%7a)J?d43*%FF>*5Yw!c`+m=_L zjQdx}pSo^&1By#ZZ_n5FpygipZrZ1y%=-!3zGC?$tG@)7(El9bVpYK?Syw-N430vX z?+g^XT2T7MP}beG`cGMY-nM@kivM1MvaU;zkf@%ipP{Vtx~=~U zzJ>a|OfK;%!FRzPxC@R!OjjxT6kq7gBJE8cw7fSpFEo<<1 zcPVucN`CJkdD}389=r-AzegX*^ZO!{{Qe4)NPd3{ieG;U#b3XI;+JcXC93OC?79mh zc~lQTsqcp3pMEIon1Ev6aVY*c17)0NtiA2xZj1+?-pxFHoK@j^MgED^wiXTR8eHKc; zPeSo?8%q2V%QZ-PXSQ)PXXX9;lE!KeBEEExBglSa4EYcuYx^i7MdCDooRNwx@|}?K z1oAPYhKS#aa@sz8uxNnd2iZ$v-xMNc7Lo6b6p2R}IUp5V#FtMaPa#9LZP2m_r;#4y zAhH(`ABi7uUslfYD{&Oxe;9c$Qj`%MaF6;((Ey)BcH6qduMd$tlRYm*VkQ2UT$8*V zMvBsJA5Ox3_PMuE!YU#$mv~7TM~)#JJ*t5mMI_dwv^s}Kk=&a=jw6%E403ZhWh)g} zM-CwxIf0BKs-&|r6-UduZDOO>qONYap*EhM>a=aNS$6$EyPg-DB++i@>1Ok|c9SG( z`EF`Fz3QiJ?P@QgV_41xEQW3J{$%$4f^Py(C+(;UT2YwNmzP%D!1p#ykw$vn+%$nT zF`GsVy3<4_+-0NhFwa6a2%HODW}|~=htQ$C(IW%fIntQ5`r)GPnAnfJ-rH9bby7bH z-5_&9KACt+3=Y?1IbA~~T{R}u)y!M{)xrIP!vjNu`|Tu&Vf|jRG4@DjEK`jnX&i-% zWB7NZVxKdGxKW($Cs{;Nf7v)`+E$|w6#f%Ye$dUw95OaaSE-|^_8LZ zj4M<@pl7o+ChG8s@tH~b9d^Sg^jkzDi=^#IYs&;`Ln(=&4#x&l^+cAXx7O^qMPok8 ztu=A8ZSe}~@l;K?sjDVTU{Yh@!K4-YS)M36T@JmcN9^oPbSpE)D_i}BqTBIgPpvB^ z3>zgVx@8NoW1Ucu%%0BT_CK0)>ozMIN2lsGn>MMuEUMXPc3aZDtfnW_biGJ!HFK6+osm=2 zJWH=|(G;u58XE-C#A?P3T{O9y6Lq4YvT)0fwq4{o$$F!YLZj-b z1I3XiqI>Lt)~lLln<7lnG|#4m7v)(=7%z0A)Ma;NADJ*p~JTk)k*w<+T6AS>UbD9kKAO#$cco9W+Vgj}UJkC5;1n`~MNqty@-GP2$zm4h?i zlBcqBsM!Cv)O4OD$~N9o^K7dpXX9!47^-vRX`H(Ag>p!*Kg`UY|fj=Kqy$smF3nwlODxpR@vPrHVFqDRhWyxfLys zm5y_ubU8Vkrg4{RdNN!j@@?I7D)i-cb__!}9(&jO)wBtnBZo?7og*eRaX!^K4xiM; z{rk%UrFzhf-N2cO;^m~K!%kLBM*E-C`QvC$sCSO)YHz8y=WSgdvs?7m^>OR^SR9Yp jp;LU$x2}(cTehx`@;~UU>*Ln-aqIdh{~!PVu8;o(SpvJ5 literal 0 HcmV?d00001 diff --git a/designatedashboard/locale/en_GB/LC_MESSAGES/django.po b/designatedashboard/locale/en_GB/LC_MESSAGES/django.po new file mode 100644 index 0000000..22fa28a --- /dev/null +++ b/designatedashboard/locale/en_GB/LC_MESSAGES/django.po @@ -0,0 +1,351 @@ +# OpenStack Infra , 2015. #zanata +# Rob Cresswell , 2015. #zanata +# Andi Chandler , 2016. #zanata +# Andreas Jaeger , 2016. #zanata +msgid "" +msgstr "" +"Project-Id-Version: designate-dashboard 3.0.0.0rc2.dev6\n" +"Report-Msgid-Bugs-To: https://bugs.launchpad.net/openstack-i18n/\n" +"POT-Creation-Date: 2016-10-14 10:51+0000\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"PO-Revision-Date: 2016-10-23 09:03+0000\n" +"Last-Translator: Andreas Jaeger \n" +"Language-Team: English (United Kingdom)\n" +"Language: en-GB\n" +"X-Generator: Zanata 3.7.3\n" +"Plural-Forms: nplurals=2; plural=(n != 1)\n" + +msgid "" +"\n" +" From here you can edit the email address and TTL associated with a " +"domain.\n" +" " +msgstr "" +"\n" +" From here you can edit the email address and TTL associated with a " +"domain.\n" +" " + +msgid "" +"\n" +" The Email field should contain a valid email address to be associated\n" +" with the domain.\n" +" " +msgstr "" +"\n" +" The Email field should contain a valid email address to be associated\n" +" with the domain.\n" +" " + +msgid "" +"\n" +" The Name field should contain a full-qualified domain name (with\n" +" trailing period).\n" +" " +msgstr "" +"\n" +" The Name field should contain a full-qualified domain name (with\n" +" trailing period).\n" +" " + +msgid "" +"\n" +" The optional TTL field can be any value between 1 and 2147483647\n" +" seconds.\n" +" " +msgstr "" +"\n" +" The optional TTL field can be any value between 1 and 2147483647\n" +" seconds.\n" +" " + +msgid "" +"\n" +"

\n" +" TTL\n" +" The TTL is the time-to-live for the record, in seconds.\n" +"

\n" +"

\n" +" See more info on record types.\n" +"

\n" +" " +msgstr "" +"\n" +"

\n" +" TTL\n" +" The TTL is the time-to-live for the record, in seconds.\n" +"

\n" +"

\n" +" See more info on record types.\n" +"

\n" +" " + +msgid "A - Address record" +msgstr "A - Address record" + +msgid "AAAA - IPv6 address record" +msgstr "AAAA - IPv6 address record" + +msgid "All Records" +msgstr "All Records" + +msgid "CNAME - Canonical name record" +msgstr "CNAME - Canonical name record" + +msgid "Cancel" +msgstr "Cancel" + +msgid "Canonical Name" +msgstr "Canonical Name" + +msgid "Create Domain" +msgstr "Create Domain" + +msgid "Create Domain Record" +msgstr "Create Domain Record" + +msgid "Create Record" +msgstr "Create Record" + +msgid "Create Record for" +msgstr "Create Record for" + +msgid "Created" +msgstr "Created" + +msgid "Created At" +msgstr "Created At" + +msgid "Data" +msgstr "Data" + +msgid "Delete" +msgstr "Delete" + +msgid "Deleted" +msgstr "Deleted" + +msgid "Description" +msgstr "Description" + +msgid "Domain" +msgstr "Domain" + +#, python-format +msgid "Domain %(name)s created." +msgstr "Domain %(name)s created." + +#, python-format +msgid "Domain %(name)s updated." +msgstr "Domain %(name)s updated." + +msgid "Domain Detail" +msgstr "Domain Detail" + +msgid "Domain Name" +msgstr "Domain Name" + +msgid "Domain Overview" +msgstr "Domain Overview" + +msgid "Domain Records" +msgstr "Domain Records" + +#, python-format +msgid "Domain record %(name)s created." +msgstr "Domain record %(name)s created." + +#, python-format +msgid "Domain record %(name)s updated." +msgstr "Domain record %(name)s updated." + +msgid "Domains" +msgstr "Domains" + +msgid "Edit Domain" +msgstr "Edit Domain" + +msgid "Edit Record" +msgstr "Edit Record" + +msgid "Email" +msgstr "Email" + +msgid "Enter a valid IPv4 address" +msgstr "Enter a valid IPv4 address" + +msgid "Enter a valid IPv6 address" +msgstr "Enter a valid IPv6 address" + +msgid "Enter a valid SRV name" +msgstr "Enter a valid SRV name" + +msgid "Enter a valid SRV record" +msgstr "Enter a valid SRV record" + +msgid "Enter a valid SSHFP record" +msgstr "Enter a valid SSHFP record" + +msgid "Enter a valid domain name." +msgstr "Enter a valid domain name." + +msgid "Enter a valid hostname" +msgstr "Enter a valid hostname" + +msgid "" +"Enter a valid hostname. The hostname should contain letters and numbers, and " +"be no more than 63 characters." +msgstr "" +"Enter a valid hostname. The hostname should contain letters and numbers, and " +"be no more than 63 characters." + +msgid "ID" +msgstr "ID" + +msgid "IP Address" +msgstr "IP Address" + +msgid "MX - Mail exchange record" +msgstr "MX - Mail exchange record" + +msgid "Mail Server" +msgstr "Mail Server" + +msgid "Manage Records" +msgstr "Manage Records" + +msgid "NS" +msgstr "NS" + +msgid "Name" +msgstr "Name" + +msgid "Name Server" +msgstr "Name Server" + +msgid "Nameservers" +msgstr "Nameservers" + +msgid "None" +msgstr "None" + +msgid "PTR - Pointer record" +msgstr "PTR - Pointer record" + +msgid "PTR Domain Name" +msgstr "PTR Domain Name" + +msgid "Priority" +msgstr "Priority" + +msgid "Record" +msgstr "Record" + +msgid "Record Data" +msgstr "Record Data" + +msgid "Record Detail" +msgstr "Record Detail" + +msgid "Record Type" +msgstr "Record Type" + +msgid "Records" +msgstr "Records" + +msgid "Reverse DNS" +msgstr "Reverse DNS" + +msgid "SOA" +msgstr "SOA" + +msgid "SPF - Sender Policy Framework" +msgstr "SPF - Sender Policy Framework" + +msgid "SRV - Service locator" +msgstr "SRV - Service locator" + +msgid "SSHFP - SSH Public Key Fingerprint" +msgstr "SSHFP - SSH Public Key Fingerprint" + +msgid "Select an IP" +msgstr "Select an IP" + +msgid "Serial" +msgstr "Serial" + +msgid "TTL" +msgstr "TTL" + +msgid "TTL (seconds)" +msgstr "TTL (seconds)" + +msgid "TXT" +msgstr "TXT" + +msgid "TXT - Text record" +msgstr "TXT - Text record" + +msgid "Text" +msgstr "Text" + +msgid "The quotas could not be retrieved." +msgstr "The quotas could not be retrieved." + +msgid "There are no floating IP addresses currently in use to select from." +msgstr "There are no floating IP addresses currently in use to select from." + +msgid "This field is required" +msgstr "This field is required" + +msgid "Type" +msgstr "Type" + +msgid "Unable to create domain." +msgstr "Unable to create domain." + +msgid "Unable to create record." +msgstr "Unable to create record." + +msgid "Unable to retrieve domain list." +msgstr "Unable to retrieve domain list." + +msgid "Unable to retrieve domain record." +msgstr "Unable to retrieve domain record." + +msgid "Unable to retrieve record list." +msgstr "Unable to retrieve record list." + +msgid "Unable to update domain." +msgstr "Unable to update domain." + +msgid "Unknown" +msgstr "Unknown" + +msgid "Unknown instance name" +msgstr "Unknown instance name" + +msgid "Update Domain" +msgstr "Update Domain" + +msgid "Update Domain Record" +msgstr "Update Domain Record" + +msgid "Update Record" +msgstr "Update Record" + +msgid "Updated" +msgstr "Updated" + +msgid "Updated At" +msgstr "Updated At" + +msgid "Value" +msgstr "Value" + +msgid "Zones" +msgstr "Zones" diff --git a/designatedashboard/locale/es/LC_MESSAGES/django.mo b/designatedashboard/locale/es/LC_MESSAGES/django.mo new file mode 100644 index 0000000000000000000000000000000000000000..a920e4e0ca873afb5bcb82b5517c66c272b4b405 GIT binary patch literal 7000 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, 2015. #zanata +# OpenStack Infra , 2015. #zanata +# Andreas Jaeger , 2016. #zanata +# Eugènia Torrella , 2016. #zanata +# Alberto Molina Coballes , 2017. #zanata +# Zeus Arias Lucero , 2017. #zanata +msgid "" +msgstr "" +"Project-Id-Version: designate-dashboard 5.0.0.0rc2.dev4\n" +"Report-Msgid-Bugs-To: https://bugs.launchpad.net/openstack-i18n/\n" +"POT-Creation-Date: 2017-08-23 14:30+0000\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"PO-Revision-Date: 2017-08-24 09:39+0000\n" +"Last-Translator: Zeus Arias Lucero \n" +"Language-Team: Spanish\n" +"Language: es\n" +"X-Generator: Zanata 3.9.6\n" +"Plural-Forms: nplurals=2; plural=(n != 1)\n" + +msgid "" +"\n" +" From here you can edit the email address and TTL associated with a " +"domain.\n" +" " +msgstr "" +"\n" +" Desde aquí puede editar la dirección de correo y el TTL asociado a un " +"dominio.\n" +" " + +msgid "" +"\n" +" The Name field should contain a full-qualified domain name (with\n" +" trailing period).\n" +" " +msgstr "" +"\n" +" El campo nombre debe contener un full-qualified domain name (con \n" +" punto final).\n" +" " + +msgid "" +"\n" +" The optional TTL field can be any value between 1 and 2147483647\n" +" seconds.\n" +" " +msgstr "" +"\n" +" El campo opcional TTL puede tener un valor entre 1 y 2147483647\n" +" segundos.\n" +" " + +msgid "" +"\n" +"

\n" +" TTL\n" +" The TTL is the time-to-live for the record, in seconds.\n" +"

\n" +"

\n" +" See more info on record types.\n" +"

\n" +" " +msgstr "" +"\n" +"

\n" +" TTL\n" +" El TTL es el tiempo de vida del registro en segundos.\n" +"

\n" +"

\n" +" Vea más info sobre tipos de registros.\n" +"

\n" +" " + +msgid "A - Address record" +msgstr "A - Registro de dirección IPv4" + +msgid "AAAA - IPv6 address record" +msgstr "AAAA - Registro de dirección IPv6" + +msgid "All Records" +msgstr "Todos los registros" + +msgid "CNAME - Canonical name record" +msgstr "CNAME - Registro de nombre canónico" + +msgid "Cancel" +msgstr "Cancelar " + +msgid "Canonical Name" +msgstr "Nombre canónico" + +msgid "Create Domain" +msgstr "Crear dominio" + +msgid "Create Domain Record" +msgstr "Crear registro del dominio" + +msgid "Create Record" +msgstr "Crear registro" + +msgid "Create Record for" +msgstr "Crear registro para" + +msgid "Created" +msgstr "Creado" + +msgid "Created At" +msgstr "Creado el" + +msgid "Data" +msgstr "Datos" + +msgid "Delete" +msgstr "Eliminar" + +msgid "Deleted" +msgstr "Eliminado" + +msgid "Description" +msgstr "Descripción" + +msgid "Domain" +msgstr "Dominio" + +#, python-format +msgid "Domain %(name)s created." +msgstr "Dominio %(name)s creado." + +#, python-format +msgid "Domain %(name)s updated." +msgstr "Dominio %(name)s actualizado." + +msgid "Domain Detail" +msgstr "Detalles del dominio" + +msgid "Domain Name" +msgstr "Nombre de dominio" + +msgid "Domain Overview" +msgstr "Visión general del dominio" + +msgid "Domain Records" +msgstr "Registros del dominio" + +#, python-format +msgid "Domain record %(name)s created." +msgstr "Registro de dominio %(name)s creado" + +#, python-format +msgid "Domain record %(name)s updated." +msgstr "Registro de dominio %(name)s actualizado" + +msgid "Domains" +msgstr "Dominios" + +msgid "Edit Domain" +msgstr "Editar dominio" + +msgid "Edit Record" +msgstr "Editar registro" + +msgid "Email" +msgstr "Correo electrónico" + +msgid "Enter a valid IPv4 address" +msgstr "Introduzca una dirección IPv4 válida" + +msgid "Enter a valid IPv6 address" +msgstr "Introduzca una dirección IPv6 válida" + +msgid "Enter a valid SRV name" +msgstr "Introduzca un nombre SRV válido" + +msgid "Enter a valid SRV record" +msgstr "Introduzca un registro SRV válido" + +msgid "Enter a valid SSHFP record" +msgstr "Introduzca un registro SSHFP válido" + +msgid "Enter a valid domain name." +msgstr "Introduzca un nombre de dominio válido." + +msgid "Enter a valid hostname" +msgstr "Introduzca un hostname válido" + +msgid "" +"Enter a valid hostname. The hostname should contain letters and numbers, and " +"be no more than 63 characters." +msgstr "" +"Introduzca un hostname válido. El hostname puede incluir letras, números y " +"no tener más de 63 caracteres." + +msgid "ID" +msgstr "ID" + +msgid "IP Address" +msgstr "Dirección IP" + +msgid "MX - Mail exchange record" +msgstr "MX - Registro de Mail exchange" + +msgid "Mail Server" +msgstr "Servidor de correo" + +msgid "Manage Records" +msgstr "Gestionar registros" + +msgid "NS" +msgstr "NS" + +msgid "Name" +msgstr "Nombre" + +msgid "Name Server" +msgstr "Servidor de nombres" + +msgid "Nameservers" +msgstr "Servidores de nombres" + +msgid "None" +msgstr "Ninguno" + +msgid "PTR - Pointer record" +msgstr "PTR - Registro de puntero" + +msgid "PTR Domain Name" +msgstr "Nombre de dominio PTR" + +msgid "Priority" +msgstr "Prioridad" + +msgid "Record" +msgstr "Registro" + +msgid "Record Data" +msgstr "Registro" + +msgid "Record Detail" +msgstr "Detalles del registro" + +msgid "Record Type" +msgstr "Tipo de registro" + +msgid "Records" +msgstr "Registros" + +msgid "Reverse DNS" +msgstr "DNS inverso" + +msgid "SOA" +msgstr "SOA" + +msgid "SPF - Sender Policy Framework" +msgstr "SPF - Sender Policy Framework" + +msgid "SRV - Service locator" +msgstr "SRV - Servicio locator" + +msgid "SSHFP - SSH Public Key Fingerprint" +msgstr "SSHFP - Huella digital de clave SSH pública" + +msgid "Select an IP" +msgstr "Seleccione una IP" + +msgid "Serial" +msgstr "Serial" + +msgid "TTL" +msgstr "TTL" + +msgid "TTL (seconds)" +msgstr "TTL (segundos)" + +msgid "TXT" +msgstr "TXT" + +msgid "TXT - Text record" +msgstr "TXT - Registro de texto" + +msgid "Text" +msgstr "Texto" + +msgid "The quotas could not be retrieved." +msgstr "No ha sido posible obtener las cuotas." + +msgid "There are no floating IP addresses currently in use to select from." +msgstr "Actualmente no hay direcciones IP flotantes en uso para seleccionar." + +msgid "This field is required" +msgstr "Este campo es obligatorio" + +msgid "Type" +msgstr "Tipo" + +msgid "Unable to create domain." +msgstr "No ha sido posible crear el dominio." + +msgid "Unable to create record." +msgstr "No ha sido posible crear el registro." + +msgid "Unable to retrieve domain list." +msgstr "No ha sido posible obtener la lista de dominios." + +msgid "Unable to retrieve domain record." +msgstr "No ha sido posible obtener el registro del dominio." + +msgid "Unable to retrieve record list." +msgstr "No ha sido posible obtener la lista de registros." + +msgid "Unable to update domain." +msgstr "No ha sido posible actualizar el dominio." + +msgid "Unknown" +msgstr "Desconocido" + +msgid "Unknown instance name" +msgstr "Nombre de instancia desconicodo" + +msgid "Update Domain" +msgstr "Actualizar dominio" + +msgid "Update Domain Record" +msgstr "Actualizar registro del dominio" + +msgid "Update Record" +msgstr "Actualizar registro" + +msgid "Updated" +msgstr "Actualizada" + +msgid "Updated At" +msgstr "Actualizado el" + +msgid "Value" +msgstr "Valor" + +msgid "Zones" +msgstr "Zonas" diff --git a/designatedashboard/locale/fr/LC_MESSAGES/django.mo b/designatedashboard/locale/fr/LC_MESSAGES/django.mo new file mode 100644 index 0000000000000000000000000000000000000000..95b355df71d4bfeb6bcf144a549f1b9c211870f5 GIT binary patch literal 7398 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, 2015 +# Andreas Jaeger , 2016. #zanata +# Gael Rehault , 2016. #zanata +# Gérald LONLAS , 2016. #zanata +# Gaelle , 2017. #zanata +# JF Taltavull , 2017. #zanata +msgid "" +msgstr "" +"Project-Id-Version: designate-dashboard 6.0.0.0b2.dev3\n" +"Report-Msgid-Bugs-To: https://bugs.launchpad.net/openstack-i18n/\n" +"POT-Creation-Date: 2017-11-21 15:00+0000\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"PO-Revision-Date: 2017-11-14 07:52+0000\n" +"Last-Translator: Gaelle \n" +"Language: fr\n" +"Plural-Forms: nplurals=2; plural=(n > 1);\n" +"Generated-By: Babel 2.0\n" +"X-Generator: Zanata 3.9.6\n" +"Language-Team: French\n" + +msgid "" +"\n" +" From here you can edit the email address and TTL associated with a " +"domain.\n" +" " +msgstr "" +"\n" +" Ici vous pouvez éditer l'adresse email e t le TTL associés à un " +"domaine.\n" +" " + +msgid "" +"\n" +" The Email field should contain a valid email address to be associated\n" +" with the domain.\n" +" " +msgstr "" +"\n" +" Le champ Email doit contenir un email valide qui sera associé \n" +" avec le domaine.\n" +" " + +msgid "" +"\n" +" The Name field should contain a full-qualified domain name (with\n" +" trailing period).\n" +" " +msgstr "" +"\n" +" Le champ Nom doit contenir un nom de domaine (FQDN avec le\n" +" point \".\" de terminaison).\n" +" " + +msgid "" +"\n" +" The optional TTL field can be any value between 1 and 2147483647\n" +" seconds.\n" +" " +msgstr "" +"\n" +" La valeur du champ optionnel TTL doit être comprise entre 1 et " +"2147483647\n" +" secondes.\n" +" " + +msgid "" +"\n" +"

\n" +" TTL\n" +" The TTL is the time-to-live for the record, in seconds.\n" +"

\n" +"

\n" +" See more info on record types.\n" +"

\n" +" " +msgstr "" +"\n" +"

\n" +" TTL\n" +" Le TTL est le time-to-live de l'enregistrement, en secondes.\n" +"

\n" +"

\n" +" Voir plus de détails sur les types " +"d'enregistrement.\n" +"

\n" +" " + +msgid "A - Address record" +msgstr "A - Enregistrement d'Adresse" + +msgid "AAAA - IPv6 address record" +msgstr "AAAA - Enregistrement d'Adresse IPv6" + +msgid "All Records" +msgstr "Tous les enregistrements" + +msgid "CNAME - Canonical name record" +msgstr "CNAME - Enregistrement de Nom Canonique" + +msgid "Cancel" +msgstr "Annuler" + +msgid "Canonical Name" +msgstr "Nom canonique" + +msgid "Create Domain" +msgstr "Créer un domaine" + +msgid "Create Domain Record" +msgstr "Créer un enregistrement de domaine" + +msgid "Create Record" +msgstr "Créer un enregistrement" + +msgid "Create Record for" +msgstr "Créer un enregistrement pour" + +msgid "Created" +msgstr "Créé" + +msgid "Created At" +msgstr "Créé le" + +msgid "Data" +msgstr "Données" + +msgid "Delete" +msgstr "Supprimer" + +msgid "Deleted" +msgstr "Supprimé" + +msgid "Description" +msgstr "Description" + +msgid "Domain" +msgstr "Domaine" + +#, python-format +msgid "Domain %(name)s created." +msgstr "Domaine %(name)s créé." + +#, python-format +msgid "Domain %(name)s updated." +msgstr "Domaine %(name)s mis à jour." + +msgid "Domain Detail" +msgstr "Détail du domaine" + +msgid "Domain Name" +msgstr "Nom de domaine" + +msgid "Domain Overview" +msgstr "Vue d'ensemble du domaine" + +msgid "Domain Records" +msgstr "Enregistrements de domaine" + +#, python-format +msgid "Domain record %(name)s created." +msgstr "Enregistrement Domaine %(name)s créé" + +#, python-format +msgid "Domain record %(name)s updated." +msgstr "Enregistrement Domaine %(name)s mis à jour." + +msgid "Domains" +msgstr "Domaines" + +msgid "Edit Domain" +msgstr "Editer le domaine" + +msgid "Edit Record" +msgstr "Editer l'enregistrement" + +msgid "Email" +msgstr "Courriel" + +msgid "Enter a valid IPv4 address" +msgstr "Entrer une adresse IPv4 valide" + +msgid "Enter a valid IPv6 address" +msgstr "Entrer une adresse IPv6 valide" + +msgid "Enter a valid SRV name" +msgstr "Entrer un nom SRV valide" + +msgid "Enter a valid SRV record" +msgstr "Entrer un enregistrement SRV valide" + +msgid "Enter a valid SSHFP record" +msgstr "Entrer un enregistrement SSHFP valide" + +msgid "Enter a valid domain name." +msgstr "Entrer un nom de domaine valide." + +msgid "Enter a valid hostname" +msgstr "Entrer un nom d'hôte valide" + +msgid "" +"Enter a valid hostname. The hostname should contain letters and numbers, and " +"be no more than 63 characters." +msgstr "" +"Entrer un nom d'hôte valide. Le nom d'hôte ne doit contenir que des lettres " +"et des chiffres, et ne doit pas dépasser 63 caractères." + +msgid "ID" +msgstr "ID" + +msgid "IP Address" +msgstr "Adresse IP" + +msgid "MX - Mail exchange record" +msgstr "MX - Enregistrement de Serveur de messagerie" + +msgid "Mail Server" +msgstr "Serveur de messagerie" + +msgid "Manage Records" +msgstr "Gérer les enregistrements" + +msgid "NS" +msgstr "NS" + +msgid "Name" +msgstr "Nom" + +msgid "Name Server" +msgstr "Nom de serveur" + +msgid "Nameservers" +msgstr "Serveurs de nom" + +msgid "None" +msgstr "Aucun" + +msgid "PTR - Pointer record" +msgstr "PTR - Enregistrement de Pointeur" + +msgid "PTR Domain Name" +msgstr "Nom de domaine PTR" + +msgid "Priority" +msgstr "Priorité" + +msgid "Record" +msgstr "Enregistrement" + +msgid "Record Data" +msgstr "Enregistrer les données" + +msgid "Record Detail" +msgstr "Détail de l'enregistrement" + +msgid "Record Type" +msgstr "Type d'enregistrement" + +msgid "Records" +msgstr "Enregistrements" + +msgid "Reverse DNS" +msgstr "Reverse DNS" + +msgid "SOA" +msgstr "SOA" + +msgid "SPF - Sender Policy Framework" +msgstr "SPF - Politique d’émission" + +msgid "SRV - Service locator" +msgstr "SRV - Enregistrement de Service" + +msgid "SSHFP - SSH Public Key Fingerprint" +msgstr "SSHFP - Somme de contrôle de la clé SSH publique" + +msgid "Select an IP" +msgstr "Sélectionner une IP" + +msgid "Serial" +msgstr "Numéro de série" + +msgid "TTL" +msgstr "TTL" + +msgid "TTL (seconds)" +msgstr "TTL (secondes)" + +msgid "TXT" +msgstr "TXT" + +msgid "TXT - Text record" +msgstr "TXT - Enregistrement Texte" + +msgid "Text" +msgstr "Texte" + +msgid "The quotas could not be retrieved." +msgstr "Les quotas n'ont pas pu être récupérés." + +msgid "There are no floating IP addresses currently in use to select from." +msgstr "Aucune adresse IP flottante en cours d'utilisation à sélectionner." + +msgid "This field is required" +msgstr "Ce champ est requis" + +msgid "Type" +msgstr "Type" + +msgid "Unable to create domain." +msgstr "Impossible de créer le domaine." + +msgid "Unable to create record." +msgstr "Impossible de créer un enregistrement" + +msgid "Unable to retrieve domain list." +msgstr "Impossible de récupérer la liste des domaines." + +msgid "Unable to retrieve domain record." +msgstr "Impossible de récupérer l'enregistrement de domaine." + +msgid "Unable to retrieve record list." +msgstr "Impossible de récupérer la liste des enregistrements." + +msgid "Unable to update domain." +msgstr "Impossible de mettre à jour le domaine." + +msgid "Unknown" +msgstr "Inconnu" + +msgid "Unknown instance name" +msgstr "Nom d'instance inconnu" + +msgid "Update Domain" +msgstr "Mettre à jour le domaine" + +msgid "Update Domain Record" +msgstr "Mettre à jour l'enregistrement de domaine" + +msgid "Update Record" +msgstr "Mettre à jour l'enregistrement" + +msgid "Updated" +msgstr "Mis à jour" + +msgid "Updated At" +msgstr "Mis à jour le" + +msgid "Value" +msgstr "Valeur" + +msgid "Zones" +msgstr "Zones" diff --git a/designatedashboard/locale/id/LC_MESSAGES/django.mo b/designatedashboard/locale/id/LC_MESSAGES/django.mo new file mode 100644 index 0000000000000000000000000000000000000000..46ae6703b34b27ef71a86b4f5f31aec18c6b2a9e GIT binary patch literal 7183 zcmchbYlvh=6~`+vSxv`iOw7YaVr5OV+uiP-SN74Foz3jdOm^Iv-FCZovu>27rf*I6 zwQt|sdml5IfCfy6A3_90k;Mn%7eQ7C1O!n;661%U5JfPGLeNc8`&s@px& zJDcQ#@$S_8Zar?*sZ-~iIZ56Mzk-0T=%1{ea-PRuKxnqMgLPER4l3x ziu;-X-vJ%~x!%1XN~JF&#M6502aZW;0y@Sg#tNm2y%P|;#0BV>YoC6evg8b`^zA| z{~CB0d>W*_u7a@ammu|ahvN*$^SBqh4P1Bi^REBfAm#okNcnyVa{a%6)WgfJe*Lv} z+-)HBTmpH1Q;s$8dzT4uH+T~5|AEkFz>_dy4}2V?-QRFM<^%(f_T9xIwC_)VJh$^8 z_4NeE`F{dZ{-1-`s`v+31b0AK%29Q!fmmA1gWUfqkot%~&Xa;X#|J>TmUsxH-X8;R z0zU_G-3#EY;4>i4`41rV_IKBR6O_$ycY-{J2FQ8VK&VQ59Hg9&ft2SfgD!{p&}82)XT#l z_5F2_-#rd;{P#i1_XAgd7NmXu4CHye08$@+0_VXUsBqpTko_y*yTCSxAMp@=-41>Q zd@uNv_wSDc^?Y*_8}BL)bAw9y{x#0_XM9iQ9g`PL*aV)bIQGNZ}~$# z+=oKDUT}>kK-vu-?xTXTpB48U20w^$5M_^Rn{sS|izxJ~yHR$baPRal>WhTLa|?3T*~1Vqy?rY;O)+LZMwPqilXox(aQgj&cY^q8vjxfFcUg zJPUEulN}vvxe;Zur9!DaKb2`mOWjj`AeHCEI!UAoJ=tt7NtGl~%U7xP^5Q zfnmi=m`~fp_w(laRbL05Ogd2pw4yM@SbVgm0^b{1B8}vV-n4)d(JVuOF164x)zk74 z>#Sx$P(GJoH4O0F6f)!&MNaJ2NMr2P58JY@V?Xjr7w;zOr+yTwz$n2kMr~2xaD&^) zG%D%3)}b6X+L{=j+B>yxa?jLWw@9L~e=ix%J>8!%UDHV#M`3#g`kk)2chf;NQt5tT zP9*hvx|~Mkz+c0Lqu4aYm^JoxOQ?0z0@GD9cfLYHYdNiCC)TS6c6QRVU#nJiSXuYG zejf*}DpB07vQ}O4lk{w~dUozu z%;`X<+Q9>pR_q%)5pFpPc}J0QmJ-=A!gv+eKkUf*9(`)iF{j}uK|aiE#LablO~-4# zUeEh--MDV6i{ZqhD~|S$ES!jW`o9z3ygRiRcNX(us^fwG;=HEv^SZceaFiomjT5KL zxoy*Ir5o*N99gJ4YZ~eD%&D5$DkEK;C`qaHOEgy`ohu(S8lZ5l*r$zpVb)v0`)=dZ z@b@s1#?;de{D04+Y;{zuS{z;xi*sVJo*TDVJ`G!4W)#);V^rAI`7TUj0|%*Nv8+Oc z9;@a=9BT-3-25fo5WXj7H2R}Zi+b|}rmjc6IS!{aHaaUXGt?2vVn5vw?gTBQG2Q&# znwhIL;plE~0z$Y1jst-j$7e;OzJMhgI`m)w^(gRL8*(9rVCzxb6+Bb+!Zcf222o3; zaGZuYIkY#9$a=PdZ{$&p&k$*K+>ar>XuyYBDRc|xMi$aleGJeNhOf_IrqCX7;s_$HaUVYD7X1VijMNf|h;OP(}wDEI#@HEFFxxW+9t&$W7t zji-1SiZig&MAYNxV>sXPqE|izbHWu`lY{%C?5U))5~W$mQgjB1iMSAwf+1r3!zWnrt+S6(I5X*KF=Zlu*M`{Vng(&VNH5v9zH90VTm$a`3#tL~lH>YA> zCVuG4+0aAAr}`?DJq<7BWuXGqL*3?!9z2=8i+I>o==3lOO*T;Hsn5|UK-y_Pz*e&a z4jV&JTc_H<4qc+?W-C1T)#1PGr?zP=gMta5uV-3lcfrj@eI7E>5vv(&ChIF1`mh>A zHY838IUzN3QBLUT)bHkML5lRM86j3v9yH#OTe86{OtUV+Qa6fsV zk`yla9UR;LH!Xy_DoEaEe1@$5leCccL@!fz#?q@KiHG0-qdB5vdZzf-kIgDeVp;cA zGAPI8K_-yoIi!n&vS>z_Z8-BTi9`IzG;MyhpL??C`TPbV+B6WsGH;VqOi7r#`H1Rg ziD>G$V_uL}c9B*lzHrjGJARN<+F1vw8dOa)CLE5EvYLVk14F=twg86g%Cu6^Ta%pwccecEKg_$A5uBCXLEWhiz$Ro zRGRg(u=JXeUFX71&&Y)FnM;3V5t$|vVU?7S)$*@pXRw1R-433}-#B8`8+Mrdc<3$0 z00zBG5Fo3IX2Z{Nhv2|L5-nfPEV_>Wz32=xXro_b(ny^1RN7oLub4<5WfJWS#@=K2 zkAdC_@(s~6R>aUfFcEY8qnmehqTq+j4vYnReme^l-Oyf2J>8>p6>&(l8F4t!gU;e0 z!ywF!KRWCJd0t^-uRY6r@vVL497elv*->2B`6{m5JmnafJ0E0SdHXf;F>a!57wegD zg~&QsCr_wh5cE0E@4|RGnO_o2t<^Bm9_h~$>N)$jY_X5K#|G^|K;146hgs(Xq14UW zf$6g+&hqv_a$Za_51E7K^Z&WrBDi#T7#xpgoUzT{;ZRIpN;Z#|#Rrhzl*E4l7)w~+ literal 0 HcmV?d00001 diff --git a/designatedashboard/locale/id/LC_MESSAGES/django.po b/designatedashboard/locale/id/LC_MESSAGES/django.po new file mode 100644 index 0000000..25ba70e --- /dev/null +++ b/designatedashboard/locale/id/LC_MESSAGES/django.po @@ -0,0 +1,350 @@ +# OpenStack Infra , 2015. #zanata +# Andreas Jaeger , 2016. #zanata +# suhartono , 2016. #zanata +msgid "" +msgstr "" +"Project-Id-Version: designate-dashboard 3.0.0.0rc2.dev5\n" +"Report-Msgid-Bugs-To: https://bugs.launchpad.net/openstack-i18n/\n" +"POT-Creation-Date: 2016-09-29 13:19+0000\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"PO-Revision-Date: 2016-10-13 04:32+0000\n" +"Last-Translator: suhartono \n" +"Language-Team: Indonesian\n" +"Language: id\n" +"X-Generator: Zanata 3.7.3\n" +"Plural-Forms: nplurals=1; plural=0\n" + +msgid "" +"\n" +" From here you can edit the email address and TTL associated with a " +"domain.\n" +" " +msgstr "" +"\n" +" Dari sini Anda dapat mengedit alamat email dan TTL terkait dengan " +"domain.\n" +" " + +msgid "" +"\n" +" The Email field should contain a valid email address to be associated\n" +" with the domain.\n" +" " +msgstr "" +"\n" +"Kolom email harus berisi alamat email yang valid terkait\n" +"       dengan domain.\n" +" " + +msgid "" +"\n" +" The Name field should contain a full-qualified domain name (with\n" +" trailing period).\n" +" " +msgstr "" +"\n" +" The Name field should contain a full-qualified domain name (with\n" +" trailing period).\n" +" " + +msgid "" +"\n" +" The optional TTL field can be any value between 1 and 2147483647\n" +" seconds.\n" +" " +msgstr "" +"\n" +" Kolom TTL opsional dapat berupa nilai antara 1 dan 2147483647\n" +" detik.\n" +" " + +msgid "" +"\n" +"

\n" +" TTL\n" +" The TTL is the time-to-live for the record, in seconds.\n" +"

\n" +"

\n" +" See more info on record types.\n" +"

\n" +" " +msgstr "" +"\n" +"

\n" +" TTL\n" +" TTL adalah time-to-live untuk rekor, dalam hitungan detik.\n" +"

\n" +"

\n" +" Lihat more info pada tipe rekor.\n" +"

\n" +" " + +msgid "A - Address record" +msgstr "A - Address record" + +msgid "AAAA - IPv6 address record" +msgstr "AAAA - IPv6 address record" + +msgid "All Records" +msgstr "All Records (semua rekor)" + +msgid "CNAME - Canonical name record" +msgstr "CNAME - Canonical name record" + +msgid "Cancel" +msgstr "Batal" + +msgid "Canonical Name" +msgstr "Canonical Name (nama canonical)" + +msgid "Create Domain" +msgstr "Membuat Domain" + +msgid "Create Domain Record" +msgstr "Create Domain Record (buat rekor domain)" + +msgid "Create Record" +msgstr "Buat rekor" + +msgid "Create Record for" +msgstr "Buat rekor" + +msgid "Created" +msgstr "Created (dibuat)" + +msgid "Created At" +msgstr "Created At (dibuat pada)" + +msgid "Data" +msgstr "Data" + +msgid "Delete" +msgstr "Hapus" + +msgid "Deleted" +msgstr "Terhapus" + +msgid "Description" +msgstr "Deskripsi" + +msgid "Domain" +msgstr "Domain" + +#, python-format +msgid "Domain %(name)s created." +msgstr "Domain %(name)s dibuat." + +#, python-format +msgid "Domain %(name)s updated." +msgstr "Domain %(name)s diperbaharui." + +msgid "Domain Detail" +msgstr "Domain Detail (rincian domain)" + +msgid "Domain Name" +msgstr "Nama domain" + +msgid "Domain Overview" +msgstr "Domain Overview (ikhtisar domain)" + +msgid "Domain Records" +msgstr "Domain Records (rekor domain)" + +#, python-format +msgid "Domain record %(name)s created." +msgstr "Rekor domain %(name)s dibuat." + +#, python-format +msgid "Domain record %(name)s updated." +msgstr "Rekor domain %(name)s diperbaharui" + +msgid "Domains" +msgstr "Domain" + +msgid "Edit Domain" +msgstr "Mengedit domain" + +msgid "Edit Record" +msgstr "Mengedit recor" + +msgid "Email" +msgstr "Surat elektronik" + +msgid "Enter a valid IPv4 address" +msgstr "Masukkan alamat IPv4 yang valid" + +msgid "Enter a valid IPv6 address" +msgstr "Masukkan alamat IPv6 yang valid" + +msgid "Enter a valid SRV name" +msgstr "Masukkan nama SRV valid" + +msgid "Enter a valid SRV record" +msgstr "Masukkan rekor SRV valid" + +msgid "Enter a valid SSHFP record" +msgstr "Masukkan rekor SSHFP valid" + +msgid "Enter a valid domain name." +msgstr "Masukkan nama domain yang valid." + +msgid "Enter a valid hostname" +msgstr "Masukkan hostname yang valid" + +msgid "" +"Enter a valid hostname. The hostname should contain letters and numbers, and " +"be no more than 63 characters." +msgstr "" +"Masukkan hostname yang valid. Hostname harus berisi huruf dan angka, dan " +"tidak lebih dari 63 karakter." + +msgid "ID" +msgstr "ID" + +msgid "IP Address" +msgstr "IP Address (alamat IP)" + +msgid "MX - Mail exchange record" +msgstr "MX - Mail exchange record" + +msgid "Mail Server" +msgstr "Mail Server (server mail)" + +msgid "Manage Records" +msgstr "Mengelola rekor" + +msgid "NS" +msgstr "NS" + +msgid "Name" +msgstr "Name (nama)" + +msgid "Name Server" +msgstr "Name Server (server nama)" + +msgid "Nameservers" +msgstr "Nameservers" + +msgid "None" +msgstr "None (tak satupun)" + +msgid "PTR - Pointer record" +msgstr "PTR - Pointer record" + +msgid "PTR Domain Name" +msgstr "PTR Domain Name (nama domain PTR)" + +msgid "Priority" +msgstr "Priority (prioritas)" + +msgid "Record" +msgstr "Record (rekor)" + +msgid "Record Data" +msgstr "Record Data (data rekam)" + +msgid "Record Detail" +msgstr "Record Detail (rincian rekor)" + +msgid "Record Type" +msgstr "Record Type (tipe rekam)" + +msgid "Records" +msgstr "Records (rekor)" + +msgid "Reverse DNS" +msgstr "Reverse DNS " + +msgid "SOA" +msgstr "SOA" + +msgid "SPF - Sender Policy Framework" +msgstr "SPF - Sender Policy Framework" + +msgid "SRV - Service locator" +msgstr "SRV - Service locator" + +msgid "SSHFP - SSH Public Key Fingerprint" +msgstr "SSHFP - SSH Public Key Fingerprint" + +msgid "Select an IP" +msgstr "Pilih IP" + +msgid "Serial" +msgstr "Serial (serial)" + +msgid "TTL" +msgstr "TTL" + +msgid "TTL (seconds)" +msgstr "TTL (detik)" + +msgid "TXT" +msgstr "TXT" + +msgid "TXT - Text record" +msgstr "TXT - Text record" + +msgid "Text" +msgstr "Text" + +msgid "The quotas could not be retrieved." +msgstr "Kuota tidak dapat diambil." + +msgid "There are no floating IP addresses currently in use to select from." +msgstr "Tidak ada alamat IP mengambang saat ini yang digunakan untuk memilih." + +msgid "This field is required" +msgstr "Bagian ini diperlukan" + +msgid "Type" +msgstr "Tipe" + +msgid "Unable to create domain." +msgstr "Tidak dapat membuat domain." + +msgid "Unable to create record." +msgstr "Tidak dapat membuat rekor." + +msgid "Unable to retrieve domain list." +msgstr "Tidak dapat mengambil daftar domain." + +msgid "Unable to retrieve domain record." +msgstr "Tidak dapat mengambil rekor domain." + +msgid "Unable to retrieve record list." +msgstr "Tidak dapat mengambil daftar rekor." + +msgid "Unable to update domain." +msgstr "Tidak dapat memperbarui domain." + +msgid "Unknown" +msgstr "Unknown (tidak diketahui)" + +msgid "Unknown instance name" +msgstr " nama instance tidak diketahui" + +msgid "Update Domain" +msgstr "Update Domain (pembaharui domain)" + +msgid "Update Domain Record" +msgstr "Update Domain Record (pembaharuan rekor domain)" + +msgid "Update Record" +msgstr "Pembaruan rekor" + +msgid "Updated" +msgstr "Updated (sudah di perbaharui)" + +msgid "Updated At" +msgstr "Diperbarui pada" + +msgid "Value" +msgstr "Value (nilai)" + +msgid "Zones" +msgstr "Zones (zona)" diff --git a/designatedashboard/locale/ja/LC_MESSAGES/django.mo b/designatedashboard/locale/ja/LC_MESSAGES/django.mo new file mode 100644 index 0000000000000000000000000000000000000000..1681c82ae8a0aeb4bf9c00a0183ecf97d10631c3 GIT binary patch literal 7759 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, 2016. #zanata +# Mie Yamamoto , 2016. #zanata +# Yoshiki Eguchi , 2016. #zanata +msgid "" +msgstr "" +"Project-Id-Version: designate-dashboard 3.0.0.0rc2.dev5\n" +"Report-Msgid-Bugs-To: https://bugs.launchpad.net/openstack-i18n/\n" +"POT-Creation-Date: 2016-09-29 13:19+0000\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"PO-Revision-Date: 2016-10-13 04:32+0000\n" +"Last-Translator: Akihiro Motoki \n" +"Language-Team: Japanese\n" +"Language: ja\n" +"X-Generator: Zanata 3.7.3\n" +"Plural-Forms: nplurals=1; plural=0\n" + +msgid "" +"\n" +" From here you can edit the email address and TTL associated with a " +"domain.\n" +" " +msgstr "" +"\n" +" ここでドメインに関連付けられたメールアドレスと TTL を編集できます。\n" +" " + +msgid "" +"\n" +" The Email field should contain a valid email address to be associated\n" +" with the domain.\n" +" " +msgstr "" +"\n" +" メールフィールドには、ドメインに関連付ける\n" +" 有効なメールアドレスを指定する必要があります。\n" +" " + +msgid "" +"\n" +" The Name field should contain a full-qualified domain name (with\n" +" trailing period).\n" +" " +msgstr "" +"\n" +" 名前フィールドには、\n" +" (末尾がピリオドの) 完全修飾ドメイン名を指定する必要があります。\n" +" " + +msgid "" +"\n" +" The optional TTL field can be any value between 1 and 2147483647\n" +" seconds.\n" +" " +msgstr "" +"\n" +" オプションの TTL フィールドは\n" +" 1 秒から 2147483647 秒までの任意の値にできます。\n" +" " + +msgid "" +"\n" +"

\n" +" TTL\n" +" The TTL is the time-to-live for the record, in seconds.\n" +"

\n" +"

\n" +" See more info on record types.\n" +"

\n" +" " +msgstr "" +"\n" +"

\n" +" TTL\n" +" TTL はこのレコードの生存時間 (time-to-live) で、単位は秒です。\n" +"

\n" +"

\n" +" レコードタイプの詳細はこちらを参照" +"してください。\n" +"

\n" +" " + +msgid "A - Address record" +msgstr "A - アドレスレコード" + +msgid "AAAA - IPv6 address record" +msgstr "AAAA - IPv6 アドレスレコード" + +msgid "All Records" +msgstr "全レコード" + +msgid "CNAME - Canonical name record" +msgstr "CNAME - 別名レコード" + +msgid "Cancel" +msgstr "取り消し" + +msgid "Canonical Name" +msgstr "正規名" + +msgid "Create Domain" +msgstr "ドメインの作成" + +msgid "Create Domain Record" +msgstr "ドメインレコードの作成" + +msgid "Create Record" +msgstr "レコードの作成" + +msgid "Create Record for" +msgstr "レコードの作成: ドメイン" + +msgid "Created" +msgstr "作成時刻" + +msgid "Created At" +msgstr "作成時刻" + +msgid "Data" +msgstr "データ" + +msgid "Delete" +msgstr "削除" + +msgid "Deleted" +msgstr "削除" + +msgid "Description" +msgstr "説明" + +msgid "Domain" +msgstr "ドメイン" + +#, python-format +msgid "Domain %(name)s created." +msgstr "ドメイン %(name)s が作成されました。" + +#, python-format +msgid "Domain %(name)s updated." +msgstr "ドメイン %(name)s が更新されました。" + +msgid "Domain Detail" +msgstr "ドメインの詳細" + +msgid "Domain Name" +msgstr "ドメイン名" + +msgid "Domain Overview" +msgstr "ドメインの概要" + +msgid "Domain Records" +msgstr "ドメインレコード" + +#, python-format +msgid "Domain record %(name)s created." +msgstr "ドメインレコード %(name)s が作成されました。" + +#, python-format +msgid "Domain record %(name)s updated." +msgstr "ドメインレコード %(name)s が更新されました。" + +msgid "Domains" +msgstr "ドメイン" + +msgid "Edit Domain" +msgstr "ドメインの編集" + +msgid "Edit Record" +msgstr "レコードの編集" + +msgid "Email" +msgstr "メール" + +msgid "Enter a valid IPv4 address" +msgstr "有効な IPv4 アドレスを入力してください。" + +msgid "Enter a valid IPv6 address" +msgstr "有効な IPv6 アドレスを入力してください。" + +msgid "Enter a valid SRV name" +msgstr "有効な SRV 名を入力してください。" + +msgid "Enter a valid SRV record" +msgstr "有効な SRV レコードを入力してください。" + +msgid "Enter a valid SSHFP record" +msgstr "有効な SSHFP レコードを入力してください。" + +msgid "Enter a valid domain name." +msgstr "有効なドメイン名を入力してください。" + +msgid "Enter a valid hostname" +msgstr "有効なホスト名を入力してください。" + +msgid "" +"Enter a valid hostname. The hostname should contain letters and numbers, and " +"be no more than 63 characters." +msgstr "" +"有効なホスト名を入力してください。ホスト名は、文字と数字で 63 文字以内に設定" +"してください。" + +msgid "ID" +msgstr "ID" + +msgid "IP Address" +msgstr "IP アドレス" + +msgid "MX - Mail exchange record" +msgstr "MX - Mail exchange レコード" + +msgid "Mail Server" +msgstr "メールサーバー" + +msgid "Manage Records" +msgstr "レコードの管理" + +msgid "NS" +msgstr "NS" + +msgid "Name" +msgstr "名前" + +msgid "Name Server" +msgstr "ネームサーバー" + +msgid "Nameservers" +msgstr "ネームサーバー" + +msgid "None" +msgstr "なし" + +msgid "PTR - Pointer record" +msgstr "PTR - ポインターレコード" + +msgid "PTR Domain Name" +msgstr "PTR ドメイン名" + +msgid "Priority" +msgstr "優先度" + +msgid "Record" +msgstr "レコード" + +msgid "Record Data" +msgstr "レコードデータ" + +msgid "Record Detail" +msgstr "レコードの詳細" + +msgid "Record Type" +msgstr "レコード種別" + +msgid "Records" +msgstr "レコード" + +msgid "SOA" +msgstr "SOA" + +msgid "SPF - Sender Policy Framework" +msgstr "SPF - Sender Policy Framework" + +msgid "SRV - Service locator" +msgstr "SRV - サービスロケーター" + +msgid "SSHFP - SSH Public Key Fingerprint" +msgstr "SSHFP - SSH 公開鍵フィンガープリント" + +msgid "Select an IP" +msgstr "IP を選択してください" + +msgid "Serial" +msgstr "シリアル" + +msgid "TTL" +msgstr "TTL" + +msgid "TTL (seconds)" +msgstr "TTL (秒)" + +msgid "TXT" +msgstr "TXT" + +msgid "TXT - Text record" +msgstr "TXT - テキストレコード" + +msgid "Text" +msgstr "テキスト" + +msgid "The quotas could not be retrieved." +msgstr "クォータを取得できませんでした。" + +msgid "There are no floating IP addresses currently in use to select from." +msgstr "選択可能な現在使用中の Floating IP がありません。" + +msgid "This field is required" +msgstr "このフィールドは必須です" + +msgid "Type" +msgstr "種別" + +msgid "Unable to create domain." +msgstr "ドメインを作成できません。" + +msgid "Unable to create record." +msgstr "レコードを作成できません。" + +msgid "Unable to retrieve domain list." +msgstr "ドメインの一覧を取得できません。" + +msgid "Unable to retrieve domain record." +msgstr "ドメインレコードを取得できません。" + +msgid "Unable to retrieve record list." +msgstr "レコードの一覧を取得できません。" + +msgid "Unable to update domain." +msgstr "ドメインを更新できません。" + +msgid "Unknown" +msgstr "不明" + +msgid "Unknown instance name" +msgstr "不明なインスタンス名" + +msgid "Update Domain" +msgstr "ドメインの更新" + +msgid "Update Domain Record" +msgstr "ドメインレコードの更新" + +msgid "Update Record" +msgstr "レコードの更新" + +msgid "Updated" +msgstr "更新時刻" + +msgid "Updated At" +msgstr "最終更新" + +msgid "Value" +msgstr "値" + +msgid "Zones" +msgstr "ゾーン" diff --git a/designatedashboard/locale/ko_KR/LC_MESSAGES/django.mo b/designatedashboard/locale/ko_KR/LC_MESSAGES/django.mo new file mode 100644 index 0000000000000000000000000000000000000000..854f3e744f8279ddad9ec8316eff7f343f3d8d0e GIT binary patch literal 7499 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 literal 0 HcmV?d00001 diff --git a/designatedashboard/locale/ko_KR/LC_MESSAGES/django.po b/designatedashboard/locale/ko_KR/LC_MESSAGES/django.po new file mode 100644 index 0000000..08577b0 --- /dev/null +++ b/designatedashboard/locale/ko_KR/LC_MESSAGES/django.po @@ -0,0 +1,350 @@ +# Andreas Jaeger , 2016. #zanata +# Ian Y. Choi , 2016. #zanata +# Sungjin Kang , 2016. #zanata +msgid "" +msgstr "" +"Project-Id-Version: designate-dashboard 3.0.0.0rc2.dev5\n" +"Report-Msgid-Bugs-To: https://bugs.launchpad.net/openstack-i18n/\n" +"POT-Creation-Date: 2016-09-29 13:19+0000\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"PO-Revision-Date: 2016-09-20 12:11+0000\n" +"Last-Translator: Ian Y. Choi \n" +"Language-Team: Korean (South Korea)\n" +"Language: ko-KR\n" +"X-Generator: Zanata 3.7.3\n" +"Plural-Forms: nplurals=1; plural=0\n" + +msgid "" +"\n" +" From here you can edit the email address and TTL associated with a " +"domain.\n" +" " +msgstr "" +"\n" +" 여기서부터 도메인과 연결된 이메일 주소와 TTL을 편집할 수 있습니다.\n" +" " + +msgid "" +"\n" +" The Email field should contain a valid email address to be associated\n" +" with the domain.\n" +" " +msgstr "" +"\n" +" 이메일 피드는 도메인에 연결된 유료한 이메일 주소 값을\n" +" 포함해야 합니다.\n" +" " + +msgid "" +"\n" +" The Name field should contain a full-qualified domain name (with\n" +" trailing period).\n" +" " +msgstr "" +"\n" +" 이름 필드는 정규화된 도메인 이름 (끝 마침표와 함께)을\n" +" 포함해야 합니다.\n" +" " + +msgid "" +"\n" +" The optional TTL field can be any value between 1 and 2147483647\n" +" seconds.\n" +" " +msgstr "" +"\n" +" 옵션인 TTL 필드는 1 에서 2147483647 초 사이 임의의 값을\n" +" 사용할 수 있습니다.\n" +" " + +msgid "" +"\n" +"

\n" +" TTL\n" +" The TTL is the time-to-live for the record, in seconds.\n" +"

\n" +"

\n" +" See more info on record types.\n" +"

\n" +" " +msgstr "" +"\n" +"

\n" +" TTL\n" +" TTL은 초 단위의 레코드에 대한 time-to-live 값입니다.\n" +"

\n" +"

\n" +" 레코드 유형에 관한 자세한 정보를 살펴봅니다.\n" +"

\n" +" " + +msgid "A - Address record" +msgstr "A - 주소 레코드" + +msgid "AAAA - IPv6 address record" +msgstr "AAAA - IPv6 주소 레코드" + +msgid "All Records" +msgstr "모든 레코드" + +msgid "CNAME - Canonical name record" +msgstr "CNAME - 대체 이름 레코드" + +msgid "Cancel" +msgstr "취소" + +msgid "Canonical Name" +msgstr "대체 이름" + +msgid "Create Domain" +msgstr "도메인 생성" + +msgid "Create Domain Record" +msgstr "도메인 레코드 생성" + +msgid "Create Record" +msgstr "레코드 생성" + +msgid "Create Record for" +msgstr "다음에 대한 레코드 생성" + +msgid "Created" +msgstr "생성됨" + +msgid "Created At" +msgstr "생성 시점" + +msgid "Data" +msgstr "데이터" + +msgid "Delete" +msgstr "삭제" + +msgid "Deleted" +msgstr "삭제됨" + +msgid "Description" +msgstr "설명" + +msgid "Domain" +msgstr "도메인" + +#, python-format +msgid "Domain %(name)s created." +msgstr "도메인 %(name)s 이 생성되었습니다." + +#, python-format +msgid "Domain %(name)s updated." +msgstr "도메인 %(name)s 이 수정되었습니다." + +msgid "Domain Detail" +msgstr "도메인 세부 사항" + +msgid "Domain Name" +msgstr "도메인 이름" + +msgid "Domain Overview" +msgstr "도메인 개요" + +msgid "Domain Records" +msgstr "도메인 레코드" + +#, python-format +msgid "Domain record %(name)s created." +msgstr "도메인 레코드 %(name)s 가 생성되었습니다." + +#, python-format +msgid "Domain record %(name)s updated." +msgstr "도메인 레코드 %(name)s 가 업데이트되었습니다." + +msgid "Domains" +msgstr "도메인" + +msgid "Edit Domain" +msgstr "도메인 수정" + +msgid "Edit Record" +msgstr "레코드 수정" + +msgid "Email" +msgstr "이메일" + +msgid "Enter a valid IPv4 address" +msgstr "유효한 IPv4 주소를 입력합니다" + +msgid "Enter a valid IPv6 address" +msgstr "유요한 IPv6 주소를 입력합니다" + +msgid "Enter a valid SRV name" +msgstr "유효한 SRV 이름을 입력합니다" + +msgid "Enter a valid SRV record" +msgstr "유효한 SRV 레코드를 입력합니다" + +msgid "Enter a valid SSHFP record" +msgstr "유효한 SSHFP 레코드를 입력합니다" + +msgid "Enter a valid domain name." +msgstr "유효한 도메인 이름을 입력하시오." + +msgid "Enter a valid hostname" +msgstr "유효한 호스트명을 입력합니다" + +msgid "" +"Enter a valid hostname. The hostname should contain letters and numbers, and " +"be no more than 63 characters." +msgstr "" +"유효한 호스트명을 입력합니다. 호스트명은 문자 및 숫자로 구성되어야 하며, 63 " +"문자를 초과할 수 없습니다." + +msgid "ID" +msgstr "ID" + +msgid "IP Address" +msgstr "IP 주소" + +msgid "MX - Mail exchange record" +msgstr "MX - 메일 교환 레코드" + +msgid "Mail Server" +msgstr "메일 서버" + +msgid "Manage Records" +msgstr "레코드 관리" + +msgid "NS" +msgstr "NS" + +msgid "Name" +msgstr "이름" + +msgid "Name Server" +msgstr "네임 서버" + +msgid "Nameservers" +msgstr "이름 서버" + +msgid "None" +msgstr "없음" + +msgid "PTR - Pointer record" +msgstr "PTR - 포인터 레코드" + +msgid "PTR Domain Name" +msgstr "PTR 도메인 이름" + +msgid "Priority" +msgstr "우선순위" + +msgid "Record" +msgstr "레코드" + +msgid "Record Data" +msgstr "레코드 데이터" + +msgid "Record Detail" +msgstr "레코드 세부 사항" + +msgid "Record Type" +msgstr "레코드 타입" + +msgid "Records" +msgstr "레코드" + +msgid "Reverse DNS" +msgstr "Reverse DNS" + +msgid "SOA" +msgstr "SOA" + +msgid "SPF - Sender Policy Framework" +msgstr "SPF - 발송자 정책 프레임워크" + +msgid "SRV - Service locator" +msgstr "SRV - 서비스 위치" + +msgid "SSHFP - SSH Public Key Fingerprint" +msgstr "SSHFP - SSH 공개키 Fingerprint" + +msgid "Select an IP" +msgstr "IP를 선택합니다" + +msgid "Serial" +msgstr "시리얼" + +msgid "TTL" +msgstr "TTL" + +msgid "TTL (seconds)" +msgstr "TTL (초)" + +msgid "TXT" +msgstr "TXT" + +msgid "TXT - Text record" +msgstr "TXT - 텍스트 레코드" + +msgid "Text" +msgstr "텍스트" + +msgid "The quotas could not be retrieved." +msgstr "할당량을 가져올 수 없습니다." + +msgid "There are no floating IP addresses currently in use to select from." +msgstr "선택할 수 있는 사용 중인 floating IP 주소가 없습니다." + +msgid "This field is required" +msgstr "해당 필드는 필수입니다" + +msgid "Type" +msgstr "타입" + +msgid "Unable to create domain." +msgstr "도메인을 생성할 수 없습니다." + +msgid "Unable to create record." +msgstr "레코드를 생성할 수 없습니다." + +msgid "Unable to retrieve domain list." +msgstr "도메인 목록을 가져올 수 없습니다." + +msgid "Unable to retrieve domain record." +msgstr "도메인 레코드를 가져올 수 없습니다." + +msgid "Unable to retrieve record list." +msgstr "레코드 목록을 가져올 수 없습니다." + +msgid "Unable to update domain." +msgstr "도메인을 업데이트할 수 없습니다." + +msgid "Unknown" +msgstr "알 수 없음" + +msgid "Unknown instance name" +msgstr "알려지지 않은 인스턴스 이름" + +msgid "Update Domain" +msgstr "도메인 업데이트" + +msgid "Update Domain Record" +msgstr "도메인 레코드 업데이트" + +msgid "Update Record" +msgstr "레코드 업데이트" + +msgid "Updated" +msgstr "업데이트됨" + +msgid "Updated At" +msgstr "갱신 시점" + +msgid "Value" +msgstr "값" + +msgid "Zones" +msgstr "존" diff --git a/designatedashboard/locale/pt_BR/LC_MESSAGES/django.mo b/designatedashboard/locale/pt_BR/LC_MESSAGES/django.mo new file mode 100644 index 0000000000000000000000000000000000000000..45113d43589246fe33edf8ebb553c9bbe489381f GIT binary patch literal 7192 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 b8{+lZ-R((Ouju6naQ%?i+eGsEAXWbbP3DP8 literal 0 HcmV?d00001 diff --git a/designatedashboard/locale/pt_BR/LC_MESSAGES/django.po b/designatedashboard/locale/pt_BR/LC_MESSAGES/django.po new file mode 100644 index 0000000..c79632c --- /dev/null +++ b/designatedashboard/locale/pt_BR/LC_MESSAGES/django.po @@ -0,0 +1,352 @@ +# Andreas Jaeger , 2016. #zanata +# Eric Baum , 2016. #zanata +# Marcio , 2017. #zanata +msgid "" +msgstr "" +"Project-Id-Version: designate-dashboard 4.0.0.0rc2.dev10\n" +"Report-Msgid-Bugs-To: https://bugs.launchpad.net/openstack-i18n/\n" +"POT-Creation-Date: 2017-03-10 19:52+0000\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"PO-Revision-Date: 2017-03-15 07:16+0000\n" +"Last-Translator: Marcio \n" +"Language-Team: Portuguese (Brazil)\n" +"Language: pt-BR\n" +"X-Generator: Zanata 3.9.6\n" +"Plural-Forms: nplurals=2; plural=(n != 1)\n" + +msgid "" +"\n" +" From here you can edit the email address and TTL associated with a " +"domain.\n" +" " +msgstr "" +"\n" +" Aqui você pode editar o endereço de e-mail e o TTL associado ao " +"domínio.\n" +" " + +msgid "" +"\n" +" The Email field should contain a valid email address to be associated\n" +" with the domain.\n" +" " +msgstr "" +"\n" +" O campo de Email deve conter um endereço de e-mail válido para ser\n" +" associado ao domínio.\n" +" " + +msgid "" +"\n" +" The Name field should contain a full-qualified domain name (with\n" +" trailing period).\n" +" " +msgstr "" +"\n" +" O campo de nome deve conter um nome de domínio completamente " +"qualificado\n" +" ( com ponto final).\n" +" " + +msgid "" +"\n" +" The optional TTL field can be any value between 1 and 2147483647\n" +" seconds.\n" +" " +msgstr "" +"\n" +" O campo opcional de TTL pode ter qualquer valor entre 1 e 2147483647\n" +" segundos.\n" +" " + +msgid "" +"\n" +"

\n" +" TTL\n" +" The TTL is the time-to-live for the record, in seconds.\n" +"

\n" +"

\n" +" See more info on record types.\n" +"

\n" +" " +msgstr "" +"\n" +"

\n" +" TTL\n" +" O TTL é o tempo de vida do registro, em segundos.\n" +"

\n" +"

\n" +" Veja para mais informações nos tipos de " +"registro.\n" +"

\n" +" " + +msgid "A - Address record" +msgstr "A - Registro do endereço" + +msgid "AAAA - IPv6 address record" +msgstr "AAAA - Registro do endereço IPv6" + +msgid "All Records" +msgstr "Todos os Registros" + +msgid "CNAME - Canonical name record" +msgstr "CNAME - Registro Canonical name " + +msgid "Cancel" +msgstr "Cancelar" + +msgid "Canonical Name" +msgstr "Canonical Name" + +msgid "Create Domain" +msgstr "Criar Domínio" + +msgid "Create Domain Record" +msgstr "Criar Registro do Domínio" + +msgid "Create Record" +msgstr "Criar Registro" + +msgid "Create Record for" +msgstr "Criar Registro para" + +msgid "Created" +msgstr "Criado" + +msgid "Created At" +msgstr "Criado em" + +msgid "Data" +msgstr "Dados" + +msgid "Delete" +msgstr "Remover" + +msgid "Deleted" +msgstr "Removido" + +msgid "Description" +msgstr "Descrição" + +msgid "Domain" +msgstr "Domínio" + +#, python-format +msgid "Domain %(name)s created." +msgstr "Domínio %(name)s criado." + +#, python-format +msgid "Domain %(name)s updated." +msgstr "Domínio %(name)s atualizado." + +msgid "Domain Detail" +msgstr "Detalhes do Domínio" + +msgid "Domain Name" +msgstr "Nome do Domínio" + +msgid "Domain Overview" +msgstr "Visão Geral do Domínio" + +msgid "Domain Records" +msgstr "Registros do Domínio" + +#, python-format +msgid "Domain record %(name)s created." +msgstr "Registro de domínio %(name)s criado." + +#, python-format +msgid "Domain record %(name)s updated." +msgstr "Registro de domínio %(name)s atualizado." + +msgid "Domains" +msgstr "Domínios" + +msgid "Edit Domain" +msgstr "Editar Domínio" + +msgid "Edit Record" +msgstr "Editar Registro" + +msgid "Email" +msgstr "Email" + +msgid "Enter a valid IPv4 address" +msgstr "Entre com um endereço IPv4 válido" + +msgid "Enter a valid IPv6 address" +msgstr "Entre com um endereço IPv6 válido" + +msgid "Enter a valid SRV name" +msgstr "Entre com um nome SRV válido." + +msgid "Enter a valid SRV record" +msgstr "Entre com um registro SRV válido" + +msgid "Enter a valid SSHFP record" +msgstr "Entre com um registro SSHFP válido" + +msgid "Enter a valid domain name." +msgstr "Entre com um nome de domínio válido." + +msgid "Enter a valid hostname" +msgstr "Entre com um nome de host válido" + +msgid "" +"Enter a valid hostname. The hostname should contain letters and numbers, and " +"be no more than 63 characters." +msgstr "" +"Insira um nome de host válido. O Nome de host deve conter letras e números, " +"e não pode ter mais de 63 caracteres." + +msgid "ID" +msgstr "ID" + +msgid "IP Address" +msgstr "Endereço IP" + +msgid "MX - Mail exchange record" +msgstr "MX - Registro Mail exchange" + +msgid "Mail Server" +msgstr "Servidor de Email" + +msgid "Manage Records" +msgstr "Gerenciar Registros" + +msgid "NS" +msgstr "NS" + +msgid "Name" +msgstr "Nome" + +msgid "Name Server" +msgstr "Servidor de Nome" + +msgid "Nameservers" +msgstr "Servidores de Nome" + +msgid "None" +msgstr "Nenhum" + +msgid "PTR - Pointer record" +msgstr "PTR - Registro Pointer record" + +msgid "PTR Domain Name" +msgstr "PTR Nome do Domínio" + +msgid "Priority" +msgstr "Prioridade" + +msgid "Record" +msgstr "Registro" + +msgid "Record Data" +msgstr "Dados do Registro" + +msgid "Record Detail" +msgstr "Detalhes do Registro" + +msgid "Record Type" +msgstr "Tipo de Registro" + +msgid "Records" +msgstr "Registros" + +msgid "Reverse DNS" +msgstr "DNS Reverso" + +msgid "SOA" +msgstr "SOA" + +msgid "SPF - Sender Policy Framework" +msgstr "SPF - Sender Policy Framework" + +msgid "SRV - Service locator" +msgstr "SRV - Service locator" + +msgid "SSHFP - SSH Public Key Fingerprint" +msgstr "SSHFP - SSH Public Key Fingerprint" + +msgid "Select an IP" +msgstr "Selecione um IP" + +msgid "Serial" +msgstr "Serial" + +msgid "TTL" +msgstr "TTL" + +msgid "TTL (seconds)" +msgstr "TTL (segundos)" + +msgid "TXT" +msgstr "TXT" + +msgid "TXT - Text record" +msgstr "TXT - Text record" + +msgid "Text" +msgstr "Text" + +msgid "The quotas could not be retrieved." +msgstr "Não foi possível recuperar as cotas." + +msgid "There are no floating IP addresses currently in use to select from." +msgstr "Nenhum endereço IP flutuante atualmente em uso para ser selecionado." + +msgid "This field is required" +msgstr "Este campo é necessário" + +msgid "Type" +msgstr "Tipo" + +msgid "Unable to create domain." +msgstr "Não foi possível criar o domínio." + +msgid "Unable to create record." +msgstr "Não é possível criar o registro." + +msgid "Unable to retrieve domain list." +msgstr "Não foi possível recuperar a lista de domínios." + +msgid "Unable to retrieve domain record." +msgstr "Não foi possível recuperar o registro de domínio." + +msgid "Unable to retrieve record list." +msgstr "Não é possível recuperar lista de registros." + +msgid "Unable to update domain." +msgstr "Não foi possível atualizar o domínio." + +msgid "Unknown" +msgstr "Desconhecido" + +msgid "Unknown instance name" +msgstr "Nome de Instância desconhecida" + +msgid "Update Domain" +msgstr "Atualizar Domínio" + +msgid "Update Domain Record" +msgstr "Atualizar Registro do Domínio" + +msgid "Update Record" +msgstr "Atualizar Registro" + +msgid "Updated" +msgstr "Atualizado" + +msgid "Updated At" +msgstr "Atualizado em" + +msgid "Value" +msgstr "Valor" + +msgid "Zones" +msgstr "Zonas" diff --git a/designatedashboard/locale/ru/LC_MESSAGES/django.mo b/designatedashboard/locale/ru/LC_MESSAGES/django.mo new file mode 100644 index 0000000000000000000000000000000000000000..f4cf4ec7e111b29c1d55753e60cf6466d4c1185d GIT binary patch literal 8541 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 literal 0 HcmV?d00001 diff --git a/designatedashboard/locale/ru/LC_MESSAGES/django.po b/designatedashboard/locale/ru/LC_MESSAGES/django.po new file mode 100644 index 0000000..7352687 --- /dev/null +++ b/designatedashboard/locale/ru/LC_MESSAGES/django.po @@ -0,0 +1,361 @@ +# Translations template for designate-dashboard. +# Copyright (C) 2015 ORGANIZATION +# This file is distributed under the same license as the designate-dashboard +# project. +# +# Translators: +# Denis Gubanov , 2015 +# Andreas Jaeger , 2016. #zanata +# Ivan Startsev , 2016. #zanata +# Ilya Alekseyev , 2017. #zanata +msgid "" +msgstr "" +"Project-Id-Version: designate-dashboard 4.0.0.0rc2.dev10\n" +"Report-Msgid-Bugs-To: https://bugs.launchpad.net/openstack-i18n/\n" +"POT-Creation-Date: 2017-03-10 19:52+0000\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"PO-Revision-Date: 2017-03-23 08:04+0000\n" +"Last-Translator: Ilya Alekseyev \n" +"Language: ru\n" +"Plural-Forms: nplurals=4; plural=(n%10==1 && n%100!=11 ? 0 : n%10>=2 && n" +"%10<=4 && (n%100<12 || n%100>14) ? 1 : n%10==0 || (n%10>=5 && n%10<=9) || (n" +"%100>=11 && n%100<=14)? 2 : 3);\n" +"Generated-By: Babel 2.0\n" +"X-Generator: Zanata 3.9.6\n" +"Language-Team: Russian\n" + +msgid "" +"\n" +" From here you can edit the email address and TTL associated with a " +"domain.\n" +" " +msgstr "" +"\n" +" Отсюда вы можете отредактировать адрес почты и ассоциированный с " +"доменом TTL \n" +" " + +msgid "" +"\n" +" The Email field should contain a valid email address to be associated\n" +" with the domain.\n" +" " +msgstr "" +"\n" +" Поле Почта должно содержать валидный почтовый адрес\n" +" который может быть ассоциирован с доменом.\n" +" " + +msgid "" +"\n" +" The Name field should contain a full-qualified domain name (with\n" +" trailing period).\n" +" " +msgstr "" +"\n" +" Поле Имя должно содержать полностью определённое имя домена (FQDN) " +"( с\n" +" завершающей точкой).\n" +" " + +msgid "" +"\n" +" The optional TTL field can be any value between 1 and 2147483647\n" +" seconds.\n" +" " +msgstr "" +"\n" +" Опциональное поле TTL может принимать любое значение \n" +" между 1 и 2147483647 секундами.\n" +" " + +msgid "" +"\n" +"

\n" +" TTL\n" +" The TTL is the time-to-live for the record, in seconds.\n" +"

\n" +"

\n" +" See more info on record types.\n" +"

\n" +" " +msgstr "" +"\n" +"

\n" +" TTL\n" +" TTL - это time-to-live для записи, в секундах.\n" +"

\n" +"

\n" +" See больше информации на типах записи.\n" +"

\n" +" " + +msgid "A - Address record" +msgstr "A - IPv4 адрес" + +msgid "AAAA - IPv6 address record" +msgstr "AAAA - IPv6 адрес" + +msgid "All Records" +msgstr "Все Записи" + +msgid "CNAME - Canonical name record" +msgstr "CNAME - каноническое имя" + +msgid "Cancel" +msgstr "Отмена" + +msgid "Canonical Name" +msgstr "Каноническое имя" + +msgid "Create Domain" +msgstr "Создать домен" + +msgid "Create Domain Record" +msgstr "Создать Доменную Запись" + +msgid "Create Record" +msgstr "Создать запись" + +msgid "Create Record for" +msgstr "Создать запись для" + +msgid "Created" +msgstr "Создано" + +msgid "Created At" +msgstr "Создано" + +msgid "Data" +msgstr "Данные" + +msgid "Delete" +msgstr "Удалить" + +msgid "Deleted" +msgstr "Удалено" + +msgid "Description" +msgstr "Описание" + +msgid "Domain" +msgstr "Домен" + +#, python-format +msgid "Domain %(name)s created." +msgstr "Домен %(name)s создан." + +#, python-format +msgid "Domain %(name)s updated." +msgstr "Домен %(name)s обновлен." + +msgid "Domain Detail" +msgstr "Детали Домена" + +msgid "Domain Name" +msgstr "Имя домена" + +msgid "Domain Overview" +msgstr "Обзор Домена" + +msgid "Domain Records" +msgstr "Записи Домена" + +#, python-format +msgid "Domain record %(name)s created." +msgstr "Доменная запись %(name)s создана." + +#, python-format +msgid "Domain record %(name)s updated." +msgstr "Доменная запись %(name)s обновлена." + +msgid "Domains" +msgstr "Домены" + +msgid "Edit Domain" +msgstr "Редактировать домен" + +msgid "Edit Record" +msgstr "Редактировать запись" + +msgid "Email" +msgstr "Email" + +msgid "Enter a valid IPv4 address" +msgstr "Введите корректный IPv4 адрес" + +msgid "Enter a valid IPv6 address" +msgstr "Введите корректный IPv6 адрес" + +msgid "Enter a valid SRV name" +msgstr "Введите верное SRV имя" + +msgid "Enter a valid SRV record" +msgstr "Введите корректную SRV запись" + +msgid "Enter a valid SSHFP record" +msgstr "Введите корректныю SSHFP запись" + +msgid "Enter a valid domain name." +msgstr "Введите корректное имя домена" + +msgid "Enter a valid hostname" +msgstr "Введите корректное имя узла" + +msgid "" +"Enter a valid hostname. The hostname should contain letters and numbers, and " +"be no more than 63 characters." +msgstr "" +"Введите верное имя узла. Имя узла должно содержать буквы, цифры и быть не " +"длиннее 63 символов." + +msgid "ID" +msgstr "ID" + +msgid "IP Address" +msgstr "IP адрес" + +msgid "MX - Mail exchange record" +msgstr "MX - почтовый узел" + +msgid "Mail Server" +msgstr "Почтовый сервер" + +msgid "Manage Records" +msgstr "Управление записями" + +msgid "NS" +msgstr "NS" + +msgid "Name" +msgstr "Имя" + +msgid "Name Server" +msgstr "Сервер имен" + +msgid "Nameservers" +msgstr "сервера" + +msgid "None" +msgstr "Нет" + +msgid "PTR - Pointer record" +msgstr "PTR - указатель на каноническое имя" + +msgid "PTR Domain Name" +msgstr "Имя домена PTR" + +msgid "Priority" +msgstr "Приоритет" + +msgid "Record" +msgstr "Запись" + +msgid "Record Data" +msgstr "Запись Данных" + +msgid "Record Detail" +msgstr "Детали Записи" + +msgid "Record Type" +msgstr "Тип записи" + +msgid "Records" +msgstr "Записи" + +msgid "Reverse DNS" +msgstr "Обратный DNS" + +msgid "SOA" +msgstr "SOA" + +msgid "SPF - Sender Policy Framework" +msgstr "SPF - инфраструктура политики отправителя" + +msgid "SRV - Service locator" +msgstr "SRV - указатель сервиса" + +msgid "SSHFP - SSH Public Key Fingerprint" +msgstr "SSHFP - отпечаток публичного ключа SSH" + +msgid "Select an IP" +msgstr "Выберите IP" + +msgid "Serial" +msgstr "Серийный номер" + +msgid "TTL" +msgstr "TTL" + +msgid "TTL (seconds)" +msgstr "TTL (секунды)" + +msgid "TXT" +msgstr "TXT" + +msgid "TXT - Text record" +msgstr "TXT - тестовая запись" + +msgid "Text" +msgstr "Текст" + +msgid "The quotas could not be retrieved." +msgstr "Невозможно получить квоты." + +msgid "There are no floating IP addresses currently in use to select from." +msgstr "Отсутствуют используемые плавающие IP-адреса для выбора." + +msgid "This field is required" +msgstr "Это поле обязательно" + +msgid "Type" +msgstr "Тип" + +msgid "Unable to create domain." +msgstr "Невозможно создать домен." + +msgid "Unable to create record." +msgstr "Невозможно создать запись." + +msgid "Unable to retrieve domain list." +msgstr "Невозможно получить список доменов." + +msgid "Unable to retrieve domain record." +msgstr "Невозможно получить список записей." + +msgid "Unable to retrieve record list." +msgstr "Невозможно получить список записей." + +msgid "Unable to update domain." +msgstr "Невозможно обновить домен." + +msgid "Unknown" +msgstr "Неизвестно" + +msgid "Unknown instance name" +msgstr "Неизвестное имя инстанса" + +msgid "Update Domain" +msgstr "Обновить Домен" + +msgid "Update Domain Record" +msgstr "Обновить Доменную Запись" + +msgid "Update Record" +msgstr "Обновить запись" + +msgid "Updated" +msgstr "Обновлено" + +msgid "Updated At" +msgstr "Обновлено" + +msgid "Value" +msgstr "Значение" + +msgid "Zones" +msgstr "Зоны" diff --git a/designatedashboard/locale/tr_TR/LC_MESSAGES/django.mo b/designatedashboard/locale/tr_TR/LC_MESSAGES/django.mo new file mode 100644 index 0000000000000000000000000000000000000000..8a1bf5e0f1a6200dad3ef5bca274342bd4c86088 GIT binary patch literal 6928 zcmcJTTZ~;*8OIk2C<7>h7rcOra@{yeYf?U{^{zApHMuXrG1L_pZ6&BF!=H%ym(%EpHfG_--8Ws=>19s z;FI73;9KB7!TT>$>T0kypwuV81@Ln4i{N$OSHVlcr@;?^KL@V_p94P({?_+j0Y60l zA0RB%C2Yc{x)Qt?ycU#wt^8yQ z_*0L+24&yZLH^X69^V2*B?DI!=ex?|ZQw^4p8|#N=lu8yk6-unr@L4=ZX zjes8qcYwn0C@6B(L7A6=a_)|&FM52$kADLc{e2shb3F}W5_J|7J)Hw(-`{}zsh52J zRZ!0Prtkj?d@ub=VJ`L>06zk*2iJi+K}1&ulzn4R=667rs;=)p3X1(62Sx7hfwKOG z;BN37DEhh#VTGqTN#B3c(|-ku+%JM6-ycBX|2I(d@Rsjib5$|# z22k|80Tlaf_BaNQGWqH z3H}unyNx{@c#uU^PdG}zvn^G#|xmy{RVgw zcm+Wye1}1?!#F7Or$DjOJ-&Yw6!{+jMeZ0B`{v+g@GgU z%S-y3K+(^B5Y?&%DE5BX_rDB^9e)7I{Ga*$3!upLGKi_wYoO@kZBWjADVyT9F7~7^ zkDOO@bTduPTA}TsZKds`-9i&zx|b%8*kcpzsC4`zu_DjywEJjdG?Alxj`)uo%K=dI z`2bDyHAR!>5Ka6?9`UOx?GEYqXE%5c?M~VjKPLK>r$*aLTTi=-b{kE6L-dAv-E)k$ z=tFcS_7EMH&vrggle)DW0JqV^59AR$4AI1I#MbhN&xoDHx5RHpY2`EGKMsN${r85F z2oBREW+d+98K>=|iCt%Dchkg%gowJICXe{e1Z_WUk~U52eGdE1AUH$YMboqcv>h}x zpxrZ-CN15tsny-2qw6NtHVAW_H>|cT6GqwuL29#1n>f(5+J0@aEUAYkw}Czx<_&Fh zkT5Z>x(&;1YpkDi>*qro1v+aa9iX1XIdfTb!bD-v$0ASkob6%YMRaULpbIfNU|LpR zfX{plQRywx+wEawlh)xEd3cRJGcEKieoVJG_CKIp$4LDVQp z4>MZdn1P4x1H<+?(8f?_4wTHp>LAQ6wVb)r`-RMSY=Xb1Yvr{LrE|zPzTn_DJQscU!$Guw>bJREwqkCsgY+3QRk`zVykUL|h zb{`m@o@Cx`6DM(4#~RL(ekP-J8>v;IIELDtT13?o&P&(!{J!O2!DVev)a*xs67_hl zCQNSBgpF)&Jp^D@PebP?%Cjp#-@IPr+>q(I6DFv7`YNH*4kV^l2osh-iOOm2Abzid zCv18mv?ohisT)r&bg{Z|KgWK0ALC3-O8k5A-M3eZ3uiSM=Qdr5U+lHHwAcBh6;t++ zW)B^4c6*26Lb`s!?A=o{-kbVJfwQXHt=dOwBv~$6zrb)+OK_Ew`V$m(<(x&24&XmrrrjOyP(~?E)L(Ga7YzU3RYxF+LGBT}A+bPnkSz@T3qg#TXP8ym>k@BSJ^**6(P_<(< zX|Sl-hw@U3@~y;-4|bB=5MAz)iIZH?a%%H5_+Qi0g_kHlxTY8Q z(Li$J5xET2{rG97X42&I*tfDbs2staxI)MD%KcFZOxBo7Od9Cht3&c{ZloI66Qcu% zY&%Kw%5=6629;f%g{)Fb#Xpi z*p8@Lwz){pxB?H?rYfVW^HLLM^ERza#`Pq?v&Qu3T$m$5g%}Fu%zYA!jOd}!vD>#6 z66`k_%w&^Df@VzbPTEqd%ysqsq=ww+?R7U&nd?>--N;=9$gRC|U_XhaLk2xj^S;(_}nqHZD5a$Z7LT|Wmk7h?=o{X(j%PxSjAIB4$pm4 z^^WS66)u&UHLWpS>!i&vYv}dzVY?d!%F1t7Ck+T@6Y&ZO3d*W;r<%MqCdER;9Q9-v zhngBjf;_OxrwDrnf~B*IHjZr9&M_ZniNnR!VoA!?Nxub~FP}M|2e~OsVOqCMy|MB=gk95g5kJ;lbMBOB>1^q&m;)VbELpN~YMMO-Te(H$6hHg z^|U>AD(z5s;|Ybz)bi75_MdA$AEY%<_#KqP4(o@T4nh`|DQ>~Pj-j~YF-Jbo#g{8AnCPxx1g~;RG z6cy&nXYe#X!V!uuNe7YMv-C{LGIX_cwisVg857rV@Y=#X3$2_lLpkF-LBs$~qZ0%P zStAcq<<&XiQ-hkUxD=rF^n|zkY>+Zk99b`r0pjz>Qi@)PETxMJ>33Kvv}R$^ccQxT z=E8I_#|5{}I&r7o`QOP^J=?Q_y0a-^O4%9ki_xlYu6E+?Z?ISmcAt}8a6#U4_X2NaSIvr8n^XhwXj{% zg=LMBqpyI)m8F73XUnIfrg+TKqsIGwPZ3MzAepx?Ao6h7I#TskvqcE%@z^x@U94wJ z+A_^>F;u%q&{UQg{wepDD`|TQT9?&(>)HFvBC14uIk)SmeH61|IKveY6XZ~;?8An| z)z;+yG~~|k7l6A#<&x!0uBLU4-c6cSwG01pQ)9|M5=`2`Nr-Z8$J91kw)6?qRzzMG zpF$a=mgO^D3V5&9)nG0O&Q?-siUz%)Zb`M+s96Nfnj`ql`c@UkbNj7k1ltN>&jBRk zwWtWAv4e2C@PbM9B5CO)e827KGzb8v7VVLYa|o)b_fBAA4$EEiTH`OYw4M2 JfnNiu`Y), 2015 +# Mücahit Büyükyılmaz , 2015. #zanata +# OpenStack Infra , 2015. #zanata +# Andreas Jaeger , 2016. #zanata +# işbaran akçayır , 2017. #zanata +msgid "" +msgstr "" +"Project-Id-Version: designate-dashboard 5.0.0.0b2.dev8\n" +"Report-Msgid-Bugs-To: https://bugs.launchpad.net/openstack-i18n/\n" +"POT-Creation-Date: 2017-05-18 22:12+0000\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"PO-Revision-Date: 2017-05-22 08:57+0000\n" +"Last-Translator: Copied by Zanata \n" +"Language: tr-TR\n" +"Plural-Forms: nplurals=1; plural=0;\n" +"Generated-By: Babel 2.0\n" +"X-Generator: Zanata 3.9.6\n" +"Language-Team: Turkish (Turkey)\n" + +msgid "" +"\n" +" From here you can edit the email address and TTL associated with a " +"domain.\n" +" " +msgstr "" +"\n" +" Burdan alan ile ilişkili eposta adresini ve TTL değerini " +"düzenleyebilirsiniz.\n" +" " + +msgid "" +"\n" +" The Email field should contain a valid email address to be associated\n" +" with the domain.\n" +" " +msgstr "" +"\n" +" Eposta alanı alan ile ilişkilendirilecek geçerli bir\n" +" eposta adresi içermelidir.\n" +" " + +msgid "" +"\n" +" The Name field should contain a full-qualified domain name (with\n" +" trailing period).\n" +" " +msgstr "" +"\n" +" İsim alanı tam-nitelikli alan adı içermelidir\n" +" (sonunda nokta ile).\n" +" " + +msgid "" +"\n" +" The optional TTL field can be any value between 1 and 2147483647\n" +" seconds.\n" +" " +msgstr "" +"\n" +" İsteğe bağlı TTL alanı 1 ve 2147483647 saniye arasında\n" +" bir sayı olabilir.\n" +" " + +msgid "" +"\n" +"

\n" +" TTL\n" +" The TTL is the time-to-live for the record, in seconds.\n" +"

\n" +"

\n" +" See more info on record types.\n" +"

\n" +" " +msgstr "" +"\n" +"

\n" +" TTL\n" +" TTL kaydın yaşam süresidir, saniye olarak.\n" +"

\n" +"

\n" +" Kayıt türleri hakkında daha fazla " +"bilgi için bakınız.\n" +"

\n" +" " + +msgid "A - Address record" +msgstr "A - Adres kaydı" + +msgid "AAAA - IPv6 address record" +msgstr "AAAA - IPv6 adres kaydı" + +msgid "All Records" +msgstr "Tüm Kayıtlar" + +msgid "CNAME - Canonical name record" +msgstr "CNAME - Meşru isim kaydı" + +msgid "Cancel" +msgstr "İptal" + +msgid "Canonical Name" +msgstr "Meşru isim" + +msgid "Create Domain" +msgstr "Alan Oluştur" + +msgid "Create Domain Record" +msgstr "Alan Kaydı Oluştur" + +msgid "Create Record" +msgstr "Kayıt Oluştur" + +msgid "Create Record for" +msgstr "Kayıt oluştur" + +msgid "Created" +msgstr "Oluşturuldu" + +msgid "Created At" +msgstr "Oluşturulduğu zaman" + +msgid "Data" +msgstr "Veri" + +msgid "Delete" +msgstr "Sil" + +msgid "Deleted" +msgstr "Silindi" + +msgid "Description" +msgstr "Açıklama" + +msgid "Domain" +msgstr "Alan" + +#, python-format +msgid "Domain %(name)s created." +msgstr "%(name)s alanı oluşturuldu" + +#, python-format +msgid "Domain %(name)s updated." +msgstr "%(name)s alanı güncellendi" + +msgid "Domain Detail" +msgstr "Alan Ayrıntısı" + +msgid "Domain Name" +msgstr "Alan Adı" + +msgid "Domain Overview" +msgstr "Alan Genel Görünümü" + +msgid "Domain Records" +msgstr "Alan Kayıtları" + +#, python-format +msgid "Domain record %(name)s created." +msgstr "%(name)s alan kaydı oluşturuldu" + +#, python-format +msgid "Domain record %(name)s updated." +msgstr "%(name)s alan kaydı güncellendi" + +msgid "Domains" +msgstr "Alanlar" + +msgid "Edit Domain" +msgstr "Alan Değiştir" + +msgid "Edit Record" +msgstr "Kaydı Düzenle" + +msgid "Email" +msgstr "E-posta" + +msgid "Enter a valid IPv4 address" +msgstr "Geçerli bir IPv4 adresi girin" + +msgid "Enter a valid IPv6 address" +msgstr "Geçerli bir IPv6 adresi girin" + +msgid "Enter a valid SRV name" +msgstr "Geçerli bir SRV adı girin" + +msgid "Enter a valid SRV record" +msgstr "Geçerli bir SRV kaydı girin" + +msgid "Enter a valid SSHFP record" +msgstr "Geçerli bir SSHFP kaydı girin" + +msgid "Enter a valid domain name." +msgstr "Geçerli bir alan adı girin" + +msgid "Enter a valid hostname" +msgstr "Geçerli bir sunucu adı girin" + +msgid "" +"Enter a valid hostname. The hostname should contain letters and numbers, and " +"be no more than 63 characters." +msgstr "" +"Geçerli bir sunucu adı girin. Sunucu adı 63 karakterden az olmayan harf ve " +"sayılar içermektedir." + +msgid "ID" +msgstr "KİMLİK" + +msgid "IP Address" +msgstr "IP Adresi" + +msgid "MX - Mail exchange record" +msgstr "MX - Posta değişim kaydı" + +msgid "Mail Server" +msgstr "Posta Sunucusu" + +msgid "Manage Records" +msgstr "Kayıtları Yönet" + +msgid "NS" +msgstr "NS" + +msgid "Name" +msgstr "İsim" + +msgid "Name Server" +msgstr "İsim Sunucusu" + +msgid "Nameservers" +msgstr "İsim sunucular" + +msgid "None" +msgstr "Yok" + +msgid "PTR - Pointer record" +msgstr "PTR - İşaretçi kaydı" + +msgid "PTR Domain Name" +msgstr "PTR Alan Adı" + +msgid "Priority" +msgstr "Öncelik" + +msgid "Record" +msgstr "Kayıt" + +msgid "Record Data" +msgstr "Kayıt Verisi" + +msgid "Record Detail" +msgstr "Kayıt Ayrıntısı" + +msgid "Record Type" +msgstr "Kayıt Tipi" + +msgid "Records" +msgstr "Kayıtlar" + +msgid "Reverse DNS" +msgstr "Ters DNS" + +msgid "SOA" +msgstr "SOA" + +msgid "SPF - Sender Policy Framework" +msgstr "SPF - Gönderen İlkesi Çatısı" + +msgid "SRV - Service locator" +msgstr "SRV - Servis konumlandırıcı" + +msgid "SSHFP - SSH Public Key Fingerprint" +msgstr "SSHFP - SSH Açık Anahtar Parmakizi" + +msgid "Select an IP" +msgstr "Bir IP seçiniz" + +msgid "Serial" +msgstr "Seri" + +msgid "TTL" +msgstr "TTL" + +msgid "TTL (seconds)" +msgstr "TTL (saniye)" + +msgid "TXT" +msgstr "TXT" + +msgid "TXT - Text record" +msgstr "TXT - Yazı Kaydı" + +msgid "Text" +msgstr "Yazı" + +msgid "The quotas could not be retrieved." +msgstr "Kotalar alınamadı." + +msgid "There are no floating IP addresses currently in use to select from." +msgstr "Seçilecek kullanılabilir değişken IP adresi yok." + +msgid "This field is required" +msgstr "Bu alanın doldurulması gerekmektedir." + +msgid "Type" +msgstr "Tip" + +msgid "Unable to create domain." +msgstr "Alan yaratılamıyor" + +msgid "Unable to create record." +msgstr "Kayıt oluşturulamıyor" + +msgid "Unable to retrieve domain list." +msgstr "Alan listesi alınamıyor." + +msgid "Unable to retrieve domain record." +msgstr "Alan kaydı alınamadı." + +msgid "Unable to retrieve record list." +msgstr "Kayıt listesi alınamadı." + +msgid "Unable to update domain." +msgstr "Alan güncellenemiyor." + +msgid "Unknown" +msgstr "Bilinmeyen" + +msgid "Unknown instance name" +msgstr "Bilinmeyen mesafe adı" + +msgid "Update Domain" +msgstr "Alanı Güncelle" + +msgid "Update Domain Record" +msgstr "Alan Kaydını Güncelle" + +msgid "Update Record" +msgstr "Kaydı Güncelle" + +msgid "Updated" +msgstr "Güncellendi" + +msgid "Updated At" +msgstr "Güncellendiği Zaman" + +msgid "Value" +msgstr "Değer" + +msgid "Zones" +msgstr "Bölgeler" diff --git a/designatedashboard/locale/zh_CN/LC_MESSAGES/django.mo b/designatedashboard/locale/zh_CN/LC_MESSAGES/django.mo new file mode 100644 index 0000000000000000000000000000000000000000..34173bdbd05e694d3f99ab68080a75f284705efb GIT binary patch literal 6537 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 ND`*>K!8VZt{|oY|ii!XL literal 0 HcmV?d00001 diff --git a/designatedashboard/locale/zh_CN/LC_MESSAGES/django.po b/designatedashboard/locale/zh_CN/LC_MESSAGES/django.po new file mode 100644 index 0000000..ff78785 --- /dev/null +++ b/designatedashboard/locale/zh_CN/LC_MESSAGES/django.po @@ -0,0 +1,353 @@ +# OpenStack Infra , 2015. #zanata +# Andreas Jaeger , 2016. #zanata +# Gaoxiao Zhu , 2016. #zanata +# Linda , 2016. #zanata +# Wu Han , 2016. #zanata +# ZHIYUAN SU , 2016. #zanata +# vuuv , 2016. #zanata +# zzxwill , 2016. #zanata +# vuuv , 2017. #zanata +msgid "" +msgstr "" +"Project-Id-Version: designate-dashboard 4.0.0.0rc2.dev10\n" +"Report-Msgid-Bugs-To: https://bugs.launchpad.net/openstack-i18n/\n" +"POT-Creation-Date: 2017-03-10 19:52+0000\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"PO-Revision-Date: 2017-03-23 08:04+0000\n" +"Last-Translator: vuuv \n" +"Language-Team: Chinese (China)\n" +"Language: zh-CN\n" +"X-Generator: Zanata 3.9.6\n" +"Plural-Forms: nplurals=1; plural=0\n" + +msgid "" +"\n" +" From here you can edit the email address and TTL associated with a " +"domain.\n" +" " +msgstr "" +"\n" +"从这里可以编辑邮件地址和关联到一个域的TTL\n" +" " + +msgid "" +"\n" +" The Email field should contain a valid email address to be associated\n" +" with the domain.\n" +" " +msgstr "" +"\n" +"电子邮件项应包括一个有效的邮件地址来\n" +"跟域名进行关联。\n" +" " + +msgid "" +"\n" +" The Name field should contain a full-qualified domain name (with\n" +" trailing period).\n" +" " +msgstr "" +"\n" +"名称项应该包含一个完全合格的域名(有一个\n" +"尾随句点)。\n" +" " + +msgid "" +"\n" +" The optional TTL field can be any value between 1 and 2147483647\n" +" seconds.\n" +" " +msgstr "" +"\n" +"可选的TTL项可以是1到2147483647之间的任何\n" +"秒数。\n" +" " + +msgid "" +"\n" +"

\n" +" TTL\n" +" The TTL is the time-to-live for the record, in seconds.\n" +"

\n" +"

\n" +" See more info on record types.\n" +"

\n" +" " +msgstr "" +"\n" +"

\n" +"TTL\n" +"TTL是指 记录的存活时间或者存在时间,单位为秒。\n" +"

\n" +"

\n" +"见记录类型的 更多信息。\n" +"

\n" +" " + +msgid "A - Address record" +msgstr "A-地址记录" + +msgid "AAAA - IPv6 address record" +msgstr "AAAA-IPv6地址记录" + +msgid "All Records" +msgstr "所有记录" + +msgid "CNAME - Canonical name record" +msgstr "CNAME-标准名称记录" + +msgid "Cancel" +msgstr "取消" + +msgid "Canonical Name" +msgstr "规范名称" + +msgid "Create Domain" +msgstr "创建域" + +msgid "Create Domain Record" +msgstr "创建域记录" + +msgid "Create Record" +msgstr "创建记录" + +msgid "Create Record for" +msgstr "创建记录为:" + +msgid "Created" +msgstr "已创建" + +msgid "Created At" +msgstr "创建于" + +msgid "Data" +msgstr "数据" + +msgid "Delete" +msgstr "删除" + +msgid "Deleted" +msgstr "已删除" + +msgid "Description" +msgstr "描述" + +msgid "Domain" +msgstr "域" + +#, python-format +msgid "Domain %(name)s created." +msgstr "域 %(name)s 已创建" + +#, python-format +msgid "Domain %(name)s updated." +msgstr "域%(name)s已更新" + +msgid "Domain Detail" +msgstr "域明细" + +msgid "Domain Name" +msgstr "域名" + +msgid "Domain Overview" +msgstr "域概览" + +msgid "Domain Records" +msgstr "域记录" + +#, python-format +msgid "Domain record %(name)s created." +msgstr "创建的域记录数 %(name)s " + +#, python-format +msgid "Domain record %(name)s updated." +msgstr "更新的域记录数 %(name)s" + +msgid "Domains" +msgstr "域" + +msgid "Edit Domain" +msgstr "编辑域" + +msgid "Edit Record" +msgstr "编辑记录" + +msgid "Email" +msgstr "邮箱" + +msgid "Enter a valid IPv4 address" +msgstr "输入一个有效的IPv4地址" + +msgid "Enter a valid IPv6 address" +msgstr "输入一个有效的IPv6地址" + +msgid "Enter a valid SRV name" +msgstr "输入一个有效的SRV名" + +msgid "Enter a valid SRV record" +msgstr "输入一个有效的SRV记录" + +msgid "Enter a valid SSHFP record" +msgstr "输入一个有效的SSHFP记录" + +msgid "Enter a valid domain name." +msgstr "输入一个有效域名" + +msgid "Enter a valid hostname" +msgstr "输入一个有效的主机名" + +msgid "" +"Enter a valid hostname. The hostname should contain letters and numbers, and " +"be no more than 63 characters." +msgstr "输入一个有效的主机名。该主机名应包含字母和数字且不多余63个字符。" + +msgid "ID" +msgstr "标识" + +msgid "IP Address" +msgstr "IP 地址" + +msgid "MX - Mail exchange record" +msgstr "MX-邮件交换记录" + +msgid "Mail Server" +msgstr "邮件服务器" + +msgid "Manage Records" +msgstr "管理记录" + +msgid "NS" +msgstr "NS" + +msgid "Name" +msgstr "名称" + +msgid "Name Server" +msgstr "域名服务器" + +msgid "Nameservers" +msgstr "名称服务器数" + +msgid "None" +msgstr "无" + +msgid "PTR - Pointer record" +msgstr "PTR-指针记录" + +msgid "PTR Domain Name" +msgstr "域名" + +msgid "Priority" +msgstr "优先级" + +msgid "Record" +msgstr "记录" + +msgid "Record Data" +msgstr "记录数据" + +msgid "Record Detail" +msgstr "记录明细" + +msgid "Record Type" +msgstr "记录类型" + +msgid "Records" +msgstr "记录数" + +msgid "Reverse DNS" +msgstr "反向解析域名" + +msgid "SOA" +msgstr "SOA" + +msgid "SPF - Sender Policy Framework" +msgstr "SPF-发送方策略框架" + +msgid "SRV - Service locator" +msgstr "SRV-服务定位器" + +msgid "SSHFP - SSH Public Key Fingerprint" +msgstr "SSHFP-SSH公钥指纹" + +msgid "Select an IP" +msgstr "选择一个IP" + +msgid "Serial" +msgstr "序列号" + +msgid "TTL" +msgstr "TTL" + +msgid "TTL (seconds)" +msgstr "生存时间(秒)" + +msgid "TXT" +msgstr "TXT" + +msgid "TXT - Text record" +msgstr "TXT-文本记录" + +msgid "Text" +msgstr "文本" + +msgid "The quotas could not be retrieved." +msgstr "无法获取配额" + +msgid "There are no floating IP addresses currently in use to select from." +msgstr "没有可以从中选择的使用中的浮动IP地址" + +msgid "This field is required" +msgstr "需要此字段" + +msgid "Type" +msgstr "类型" + +msgid "Unable to create domain." +msgstr "无法创建域" + +msgid "Unable to create record." +msgstr "不能创建记录。" + +msgid "Unable to retrieve domain list." +msgstr "无法获取域列表。" + +msgid "Unable to retrieve domain record." +msgstr "无法获取域记录" + +msgid "Unable to retrieve record list." +msgstr "无法获取记录列表" + +msgid "Unable to update domain." +msgstr "无法更新域" + +msgid "Unknown" +msgstr "未知" + +msgid "Unknown instance name" +msgstr "未知实例名" + +msgid "Update Domain" +msgstr "更新域" + +msgid "Update Domain Record" +msgstr "更新域记录" + +msgid "Update Record" +msgstr "更新记录" + +msgid "Updated" +msgstr "已更新" + +msgid "Updated At" +msgstr "已更新于" + +msgid "Value" +msgstr "值" + +msgid "Zones" +msgstr "区域" diff --git a/designatedashboard/static/designatedashboard/designatedashboard.module.js b/designatedashboard/static/designatedashboard/designatedashboard.module.js new file mode 100644 index 0000000..e51e60f --- /dev/null +++ b/designatedashboard/static/designatedashboard/designatedashboard.module.js @@ -0,0 +1,78 @@ +/** + * (c) Copyright 2015 Hewlett-Packard Development Company, L.P. + * + * Licensed under the Apache License, Version 2.0 (the "License"); you may + * not use this file except in compliance with the License. You may obtain + * a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT + * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the + * License for the specific language governing permissions and limitations + * under the License. + */ + +(function() { + 'use strict'; + + /** + * @ngdoc overview + * @ngname designatedashboard + * + * @description + * Provides the services and widgets required + * to support and display the project search panel. + */ + angular + .module('designatedashboard', [ + 'ngRoute', + 'designatedashboard.resources' + ]) + .constant( + 'designatedashboard.apiPassthroughUrl', '/api/dns/') + .config(config) + .run(run); + + config.$inject = [ + '$provide', + '$routeProvider', + '$windowProvider' + ]; + + /** + * @name designatedashboard.basePath + * @description Base path for the project dashboard + * + * @param {function} $provide ng provide service + * + * @param {function} $routeProvider ng route service + * + * @param {function} $windowProvider NG window provider + * + * @returns {undefined} + */ + function config($provide, $routeProvider, $windowProvider) { + var path = $windowProvider.$get().STATIC_URL + 'designatedashboard/'; + $provide.constant('designatedashboard.basePath', path); + + $routeProvider + .when('/project/dnszones/', { + templateUrl: path + 'zones.html' + }) + .when('/project/reverse_dns/', { + templateUrl: path + 'reverse_dns.html' + }); + } + + run.$inject = [ + 'horizon.framework.conf.resource-type-registry.service', + 'designatedashboard.basePath' + ]; + + function run(registry, basePath) { + //registry.setDefaultSummaryTemplateUrl(basePath + 'table/default-drawer.html'); + } + +})(); diff --git a/designatedashboard/static/designatedashboard/designatedashboard.scss b/designatedashboard/static/designatedashboard/designatedashboard.scss new file mode 100644 index 0000000..e69de29 diff --git a/designatedashboard/static/designatedashboard/resources/os-designate-floatingip/actions/actions.module.js b/designatedashboard/static/designatedashboard/resources/os-designate-floatingip/actions/actions.module.js new file mode 100644 index 0000000..1efe950 --- /dev/null +++ b/designatedashboard/static/designatedashboard/resources/os-designate-floatingip/actions/actions.module.js @@ -0,0 +1,66 @@ +/** + * (c) Copyright 2016 Hewlett Packard Enterprise Development LP + * + * Licensed under the Apache License, Version 2.0 (the "License"); you may + * not use this file except in compliance with the License. You may obtain + * a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT + * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the + * License for the specific language governing permissions and limitations + * under the License. + */ + +(function() { + 'use strict'; + + /** + * @ngdoc overview + * @ngname designatedashboard.resources.os-designate-floatingip.actions + * + * @description + * Provides all of the actions for DNS Floating IPs. + */ + angular.module('designatedashboard.resources.os-designate-floatingip.actions', [ + 'horizon.framework.conf', + 'horizon.app.core' + ]) + .run(run); + + run.$inject = [ + 'horizon.framework.conf.resource-type-registry.service', + 'designatedashboard.resources.os-designate-floatingip.resourceType', + 'designatedashboard.resources.os-designate-floatingip.actions.set', + 'designatedashboard.resources.os-designate-floatingip.actions.unset' + ]; + + function run( + registry, + resourceTypeString, + setAction, + unsetAction) + { + var resourceType = registry.getResourceType(resourceTypeString); + + resourceType + .itemActions + .append({ + id: 'setFloatingIp', + service: setAction, + template: { + text: gettext('Set') + } + }) + .append({ + id: 'unsetFloatingIp', + service: unsetAction, + template: { + text: gettext('Unset') + } + }); + } + +})(); diff --git a/designatedashboard/static/designatedashboard/resources/os-designate-floatingip/actions/set.service.js b/designatedashboard/static/designatedashboard/resources/os-designate-floatingip/actions/set.service.js new file mode 100644 index 0000000..89e63f0 --- /dev/null +++ b/designatedashboard/static/designatedashboard/resources/os-designate-floatingip/actions/set.service.js @@ -0,0 +1,170 @@ +/** + * + * (c) Copyright 2016 Hewlett Packard Enterprise Development Company LP + * + * Licensed under the Apache License, Version 2.0 (the "License"); you may + * not use self file except in compliance with the License. You may obtain + * a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT + * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the + * License for the specific language governing permissions and limitations + * under the License. + */ + +(function () { + 'use strict'; + + angular + .module('designatedashboard.resources.os-designate-floatingip.actions') + .factory('designatedashboard.resources.os-designate-floatingip.actions.set', action); + + action.$inject = [ + '$q', + 'designatedashboard.resources.os-designate-floatingip.api', + 'designatedashboard.resources.os-designate-floatingip.resourceType', + 'designatedashboard.resources.util', + 'horizon.app.core.openstack-service-api.serviceCatalog', + 'horizon.framework.widgets.form.ModalFormService', + 'horizon.framework.widgets.toast.service', + 'horizon.framework.widgets.modal-wait-spinner.service' + ]; + + /** + * @ngDoc factory + * @name designatedashboard.resources.os-designate-floatingip.actions.set + * + * @Description + * Brings up the Set Floating IP modal. + */ + function action($q, + api, + resourceTypeName, + util, + serviceCatalog, + schemaFormModalService, + toast, + waitSpinner) { + var dnsServiceEnabled; + var title = null; // Set once perform is called + var formConfig = { + "schema": { + "type": "object", + "properties": { + "ptrdname": { + "type": "string", + "pattern": /^.+\.$/ + }, + "description": { + "type": "string" + }, + "ttl": { + "type": "integer", + "minimum": 0, + "maximum": 2147483647 + }, + } + }, + "form": [ + { + "key": "ptrdname", + "title": gettext("Domain Name"), + "description": gettext("Domain name ending in '.'"), + "validationMessage": gettext("Domain must end with '.'"), + "placeholder": "smtp.example.com.", + "type": "text", + "required": true + }, + { + "key": "description", + "type": "textarea", + "title": gettext("Description"), + "description": gettext("Details about the PTR record.") + }, + { + "key": "ttl", + "title": gettext("TTL"), + "description": gettext("Time To Live in seconds."), + "type": "number" + } + ] + }; + + var message = { + success: gettext('Domain name PTR %s was successfully set.') + }; + + var service = { + initScope: initScope, + allowed: allowed, + perform: perform + }; + + return service; + + ///////////////// + + function initScope() { + dnsServiceEnabled = serviceCatalog.ifTypeEnabled('dns'); + } + + function allowed(item) { + return $q.all([ + // TODO (tyr) designate currently has no floating ips policy rules + dnsServiceEnabled, + util.notPending(item) + ]); + } + + function perform(item) { + // Initialize the per-item title for use now and during submit + title = gettext("Set Domain Name PTR for ") + item.address; + formConfig.title = title; + + // Get a form model based on the current item + formConfig.model = util.getModel(formConfig.form, item); + + // Initialize default data + formConfig.model.ttl = formConfig.model.ttl || 3600; + + // Remember the ID for use during submit + formConfig.model.floatingIpId = item.id; + + return schemaFormModalService.open(formConfig).then(onSubmit, onCancel); + } + + function onSubmit(context) { + var model = angular.copy(context.model); + var floatingIpId = formConfig.model.floatingIpId; + + waitSpinner.showModalSpinner(title); + return api.set(floatingIpId, model).then(onSuccess, onFailure); + } + + function onCancel() { + waitSpinner.hideModalSpinner(); + } + + function onSuccess(response) { + waitSpinner.hideModalSpinner(); + var floatingIp = response.data; + toast.add('success', interpolate(message.success, [floatingIp.ptrdname])); + + // To make the result of this action generically useful, reformat the return + // from the deleteModal into a standard form + return { + created: [], + updated: [{type: resourceTypeName, id: floatingIp.id}], + deleted: [], + failed: [] + }; + } + + function onFailure() { + waitSpinner.hideModalSpinner(); + } + } +})(); diff --git a/designatedashboard/static/designatedashboard/resources/os-designate-floatingip/actions/unset.service.js b/designatedashboard/static/designatedashboard/resources/os-designate-floatingip/actions/unset.service.js new file mode 100644 index 0000000..11be68a --- /dev/null +++ b/designatedashboard/static/designatedashboard/resources/os-designate-floatingip/actions/unset.service.js @@ -0,0 +1,139 @@ +/** + * + * (c) Copyright 2016 Hewlett Packard Enterprise Development Company LP + * + * Licensed under the Apache License, Version 2.0 (the "License"); you may + * not use self file except in compliance with the License. You may obtain + * a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT + * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the + * License for the specific language governing permissions and limitations + * under the License. + */ + +(function () { + 'use strict'; + + angular + .module('designatedashboard.resources.os-designate-floatingip.actions') + .factory('designatedashboard.resources.os-designate-floatingip.actions.unset', action); + + action.$inject = [ + '$q', + 'designatedashboard.resources.os-designate-floatingip.api', + 'designatedashboard.resources.os-designate-floatingip.resourceType', + 'designatedashboard.resources.util', + 'horizon.app.core.openstack-service-api.serviceCatalog', + 'horizon.framework.util.q.extensions', + 'horizon.framework.widgets.form.ModalFormService', + 'horizon.framework.widgets.toast.service', + 'horizon.framework.widgets.modal-wait-spinner.service' + ]; + + /** + * @ngDoc factory + * @name designatedashboard.resources.os-designate-floatingip.actions.unset + * + * @Description + * Brings up the Unset Floating IP modal. + */ + function action($q, + api, + resourceTypeName, + util, + serviceCatalog, + $qExtensions, + schemaFormModalService, + toast, + waitSpinner) { + var dnsServiceEnabled; + var title = null; // Set on perform + var currentFloatingIpId; // Used to remember the ID we are modifying since it isn't returned by the unset API call + + // Unset it just a simple case of "set", but with ptrdname of 'null' + var formConfig = { + "schema": { + "type": "object", + "properties": { + } + }, + "form": [ + ], + "model": { + } + }; + + var message = { + success: gettext('Domain name PTR successfully unset.') + }; + + var service = { + initScope: initScope, + allowed: allowed, + perform: perform + }; + + return service; + + ///////////////// + + function initScope() { + dnsServiceEnabled = serviceCatalog.ifTypeEnabled('dns'); + } + + function allowed(item) { + return $q.all([ + // TODO (tyr) designate currently has no floating ip policy rules + dnsServiceEnabled, + domainNameSet(item), + util.notPending(item) + ]); + } + + function domainNameSet(item) { + return $qExtensions.booleanAsPromise( + angular.isString(item.ptrdname) + ); + } + + function perform(item) { + title = gettext("Unset Domain Name PTR for ") + item.address; + // Store the zone ID so it can be used on submit + formConfig.model.floatingIpId = item.id; + formConfig.title = title; + return schemaFormModalService.open(formConfig).then(onSubmit, onCancel); + } + + function onSubmit(context) { + waitSpinner.showModalSpinner(title); + currentFloatingIpId = context.model.floatingIpId; + return api.unset(currentFloatingIpId).then(onSuccess, onFailure); + } + + function onCancel() { + waitSpinner.hideModalSpinner(); + } + + function onSuccess(response) { + waitSpinner.hideModalSpinner(); + toast.add('success', message.success); + + // To make the result of this action generically useful, reformat the return + // from the deleteModal into a standard form + return { + created: [], + updated: [{type: resourceTypeName, id: currentFloatingIpId}], + deleted: [], + failed: [] + }; + } + + function onFailure() { + waitSpinner.hideModalSpinner(); + } + } +})(); diff --git a/designatedashboard/static/designatedashboard/resources/os-designate-floatingip/api.service.js b/designatedashboard/static/designatedashboard/resources/os-designate-floatingip/api.service.js new file mode 100644 index 0000000..5881dca --- /dev/null +++ b/designatedashboard/static/designatedashboard/resources/os-designate-floatingip/api.service.js @@ -0,0 +1,121 @@ +/** + * (c) Copyright 2016 Hewlett Packard Enterprise Development LP + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +(function () { + 'use strict'; + + angular + .module('designatedashboard.resources.os-designate-floatingip') + .factory('designatedashboard.resources.os-designate-floatingip.api', apiService); + + apiService.$inject = [ + 'designatedashboard.apiPassthroughUrl', + 'horizon.framework.util.http.service', + 'horizon.framework.widgets.toast.service' + ]; + + /** + * @ngdoc service + * @param {Object} httpService + * @param {Object} toastService + * @name apiService + * @description Provides direct access to Designate Floating IP APIs. + * @returns {Object} The service + */ + function apiService(apiPassthroughUrl, httpService, toastService) { + var service = { + list: list, + get: get, + set: set, + unset: unset + }; + + return service; + + /////////////// + + /** + * @name list + * @description + * Get a list of DNS floating ips. + * + * The listing result is an object with property "items." Each item is + * a floating IP PTR record. + * + * @param {Object} params + * Query parameters. Optional. + * + * @returns {Object} The result of the API call + */ + function list(params) { + var config = params ? {'params': params} : {}; + return httpService.get(apiPassthroughUrl + 'v2/reverse/floatingips', config) + .error(function () { + toastService.add('error', gettext('Unable to retrieve the floating ip PTRs.')); + }); + } + + function get(id, params) { + var config = params ? {'params': params} : {}; + return httpService.get(apiPassthroughUrl + 'v2/reverse/floatingips/' + id, config) + .error(function () { + toastService.add('error', gettext('Unable to get the floating ip PTR ' + id)); + }); + } + + /** + * @name set + * @description + * Set a floating ip PTR record + * + * @param {string} floatingIpID - ID of PTR record to unset + * @param {Object} data + * Specifies the PTR information to set + * + * @returns {Object} The updated DNS floating IP object + */ + function set(floatingIpID, data) { + // The update API will not accept extra data. Restrict the input to only the allowed + // fields + var apiData = { + ptrdname: data.ptrdname, + description: data.description, + ttl: data.ttl + }; + return httpService.patch(apiPassthroughUrl + 'v2/reverse/floatingips/' + floatingIpID, apiData) + .error(function () { + toastService.add('error', gettext('Unable to set the floating IP PTR record.')); + }) + } + + /** + * @name unset + * @description + * Unset a floating ip PTR record + * + * @param {string} floatingIpID - ID of PTR record to unset + * + * @returns {Object} The updated DNS floating IP object + */ + function unset(floatingIpID) { + // Unset is just a special case of 'set' + return set(floatingIpID, { + ptrdname: null, + description: null, + ttl: null + }) + } + } +}()); diff --git a/designatedashboard/static/designatedashboard/resources/os-designate-floatingip/details/details.module.js b/designatedashboard/static/designatedashboard/resources/os-designate-floatingip/details/details.module.js new file mode 100644 index 0000000..99130cd --- /dev/null +++ b/designatedashboard/static/designatedashboard/resources/os-designate-floatingip/details/details.module.js @@ -0,0 +1,66 @@ +/** + * (c) Copyright 2016 Hewlett Packard Enterprise Development LP + * + * Licensed under the Apache License, Version 2.0 (the "License"); you may + * not use this file except in compliance with the License. You may obtain + * a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT + * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the + * License for the specific language governing permissions and limitations + * under the License. + */ + +(function() { + 'use strict'; + + /** + * @ngdoc overview + * @ngname designatedashboard.resources.os-designate-floatingip.details + * + * @description + * Provides details features for floating IPs. + */ + angular.module('designatedashboard.resources.os-designate-floatingip.details', + ['horizon.framework.conf', 'horizon.app.core']) + .run(run); + + run.$inject = [ + 'designatedashboard.resources.os-designate-floatingip.resourceType', + 'designatedashboard.resources.os-designate-floatingip.api', + 'designatedashboard.resources.os-designate-floatingip.basePath', + 'horizon.framework.conf.resource-type-registry.service' + ]; + + function run( + resourceTypeName, + api, + basePath, + registry + ) { + var resourceType = registry.getResourceType(resourceTypeName); + resourceType + .setLoadFunction(loadFunction) + .setSummaryTemplateUrl(basePath + 'details/drawer.html') + .setItemNameFunction(itemNameFunction); + + resourceType.detailsViews + .prepend({ + id: 'floatingIpDetailsOverview', + name: gettext('Overview'), + template: basePath + 'details/overview.html' + }, 0); + + function loadFunction(identifier) { + return api.get(identifier); + } + + function itemNameFunction(floatingIp) { + return floatingIp.address; + } + } + +})(); diff --git a/designatedashboard/static/designatedashboard/resources/os-designate-floatingip/details/drawer.html b/designatedashboard/static/designatedashboard/resources/os-designate-floatingip/details/drawer.html new file mode 100644 index 0000000..d9bc4e9 --- /dev/null +++ b/designatedashboard/static/designatedashboard/resources/os-designate-floatingip/details/drawer.html @@ -0,0 +1,10 @@ + + \ No newline at end of file diff --git a/designatedashboard/static/designatedashboard/resources/os-designate-floatingip/details/overview.controller.js b/designatedashboard/static/designatedashboard/resources/os-designate-floatingip/details/overview.controller.js new file mode 100644 index 0000000..13446bd --- /dev/null +++ b/designatedashboard/static/designatedashboard/resources/os-designate-floatingip/details/overview.controller.js @@ -0,0 +1,46 @@ +/** + * (c) Copyright 2016 Hewlett Packard Enterprise Development LP + * + * Licensed under the Apache License, Version 2.0 (the 'License'); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an 'AS IS' BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +(function() { + "use strict"; + + angular + .module('designatedashboard.resources.os-designate-floatingip.details') + .controller('designatedashboard.resources.os-designate-floatingip.details.overviewController', controller); + + controller.$inject = [ + 'designatedashboard.resources.os-designate-floatingip.resourceType', + 'horizon.framework.conf.resource-type-registry.service', + '$scope' + ]; + + function controller( + resourceTypeCode, + registry, + $scope + ) { + var ctrl = this; + + ctrl.item; + ctrl.resourceType = registry.getResourceType(resourceTypeCode); + + $scope.context.loadPromise.then(onGetResponse); + + function onGetResponse(response) { + ctrl.item = response.data; + } + } + +})(); diff --git a/designatedashboard/static/designatedashboard/resources/os-designate-floatingip/details/overview.html b/designatedashboard/static/designatedashboard/resources/os-designate-floatingip/details/overview.html new file mode 100644 index 0000000..6cef428 --- /dev/null +++ b/designatedashboard/static/designatedashboard/resources/os-designate-floatingip/details/overview.html @@ -0,0 +1,11 @@ +
+ + +
\ No newline at end of file diff --git a/designatedashboard/static/designatedashboard/resources/os-designate-floatingip/os-designate-floatingip.module.js b/designatedashboard/static/designatedashboard/resources/os-designate-floatingip/os-designate-floatingip.module.js new file mode 100644 index 0000000..7872ad9 --- /dev/null +++ b/designatedashboard/static/designatedashboard/resources/os-designate-floatingip/os-designate-floatingip.module.js @@ -0,0 +1,156 @@ +/** + * (c) Copyright 2016 Hewlett Packard Enterprise Development LP + * + * Licensed under the Apache License, Version 2.0 (the "License"); you may + * not use this file except in compliance with the License. You may obtain + * a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT + * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the + * License for the specific language governing permissions and limitations + * under the License. + */ + +(function() { + 'use strict'; + + /** + * @ngdoc overview + * @ngname designatedashboard.resources.os-designate-floatingip + * + * @description + * Provides all of the services and widgets required + * to support and display DNS (designate) floating ip related content. + */ + angular + .module('designatedashboard.resources.os-designate-floatingip', [ + 'ngRoute', + 'designatedashboard.resources.os-designate-floatingip.actions', + 'designatedashboard.resources.os-designate-floatingip.details' + ]) + .constant( + 'designatedashboard.resources.os-designate-floatingip.resourceType', + 'OS::Designate::FloatingIp') + .config(config) + .run(run); + + config.$inject = [ '$provide', '$windowProvider' ]; + + function config($provide, $windowProvider) { + var path = $windowProvider.$get().STATIC_URL + 'designatedashboard/resources/os-designate-floatingip/'; + $provide.constant('designatedashboard.resources.os-designate-floatingip.basePath', path); + } + + run.$inject = [ + 'horizon.app.core.detailRoute', + 'horizon.framework.conf.resource-type-registry.service', + 'designatedashboard.resources.os-designate-floatingip.api', + 'designatedashboard.resources.os-designate-floatingip.resourceType', + 'designatedashboard.resources.util' + ]; + + function run( + detailRoute, + registry, + api, + resourceTypeString, + util) + { + var resourceType = registry.getResourceType(resourceTypeString); + resourceType + .setNames(gettext('Floating IP'), gettext('Floating IPs')) + .setListFunction(listFloatingIps) + .setProperty('id', { + label: gettext('ID') + }) + .setProperty('ptrdname', { + label: gettext('PTR Domain Name'), + filters: ['noName'] + }) + .setProperty('description', { + label: gettext('Description'), + filters: ['noName'] + }) + .setProperty('ttl', { + label: gettext('Time To Live'), + filters: ['noValue'] + }) + .setProperty('address', { + label: gettext('Address'), + filters: ['noName'] + }) + .setProperty('status', { + label: gettext('Status'), + filters: ['lowercase', 'noName'], + values: util.statusMap() + }) + .setProperty('action', { + label: gettext('Action'), + filters: ['lowercase', 'noName'], + values: util.actionMap() + }); + + resourceType + .tableColumns + .append({ + id: 'address', + priority: 1, + sortDefault: true, + template: '{$ item.address $}' + }) + .append({ + id: 'ptrdname', + filters: ['noValue'], + priority: 1, + }) + .append({ + id: 'status', + filters: ['lowercase'], + values: util.statusMap(), + priority: 2 + }); + + resourceType + .filterFacets + .append({ + label: gettext('Address'), + name: 'address', + isServer: false, + singleton: true, + persistent: false + }) + .append({ + label: gettext('PTR Domain Name'), + name: 'ptrdname', + isServer: false, + singleton: true, + persistent: false + }) + .append({ + label: gettext('Status'), + name: 'status', + isServer: false, + singleton: true, + persistent: false, + options: [ + {label: gettext('Active'), key: 'active'}, + {label: gettext('Pending'), key: 'pending'} + ] + }); + + function listFloatingIps() { + return api.list().then(function onList(response) { + // listFunctions are expected to return data in "items" + response.data.items = response.data.floatingips; + + util.addTimestampIds(response.data.items); + + return response; + }); + } + } + +})(); diff --git a/designatedashboard/static/designatedashboard/resources/os-designate-recordset/actions/actions.module.js b/designatedashboard/static/designatedashboard/resources/os-designate-recordset/actions/actions.module.js new file mode 100644 index 0000000..542c689 --- /dev/null +++ b/designatedashboard/static/designatedashboard/resources/os-designate-recordset/actions/actions.module.js @@ -0,0 +1,79 @@ +/** + * (c) Copyright 2016 Hewlett Packard Enterprise Development LP + * + * Licensed under the Apache License, Version 2.0 (the "License"); you may + * not use this file except in compliance with the License. You may obtain + * a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT + * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the + * License for the specific language governing permissions and limitations + * under the License. + */ + +(function () { + 'use strict'; + + /** + * @ngdoc overview + * @ngname designatedashboard.resources.os-designate-recordset.actions + * + * @description + * Provides all of the actions for DNS Recordsets. + */ + angular.module('designatedashboard.resources.os-designate-recordset.actions', [ + 'horizon.framework.conf', + 'horizon.app.core' + ]) + .run(run); + + run.$inject = [ + 'horizon.framework.conf.resource-type-registry.service', + 'designatedashboard.resources.os-designate-recordset.resourceType', + 'designatedashboard.resources.os-designate-recordset.actions.create', + 'designatedashboard.resources.os-designate-recordset.actions.delete', + 'designatedashboard.resources.os-designate-recordset.actions.update' + ]; + + function run(registry, + resourceTypeString, + createAction, + deleteAction, + updateAction) { + var resourceType = registry.getResourceType(resourceTypeString); + + resourceType + .itemActions + .append({ + id: 'updateRecordset', + service: updateAction, + template: { + text: gettext('Update') + } + }) + .append({ + id: 'deleteRecordset', + service: deleteAction, + template: { + text: gettext('Delete'), + type: 'delete' + } + }); + + // Append a record set view to the zones actions + var zoneResourceType = registry.getResourceType("OS::Designate::Zone"); + zoneResourceType + .itemActions + .append({ + id: 'createRecordset', + service: createAction, + template: { + text: gettext('Create Record Set') + } + }); + } + +})(); diff --git a/designatedashboard/static/designatedashboard/resources/os-designate-recordset/actions/common-forms.service.js b/designatedashboard/static/designatedashboard/resources/os-designate-recordset/actions/common-forms.service.js new file mode 100644 index 0000000..3f18bcc --- /dev/null +++ b/designatedashboard/static/designatedashboard/resources/os-designate-recordset/actions/common-forms.service.js @@ -0,0 +1,165 @@ +/** + * + * (c) Copyright 2016 Hewlett Packard Enterprise Development Company LP + * + * Licensed under the Apache License, Version 2.0 (the "License"); you may + * not use self file except in compliance with the License. You may obtain + * a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT + * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the + * License for the specific language governing permissions and limitations + * under the License. + */ + +(function () { + 'use strict'; + + angular + .module('designatedashboard.resources.os-designate-recordset.actions') + .factory('designatedashboard.resources.os-designate-recordset.actions.common-forms', service); + + service.$inject = [ + 'designatedashboard.resources.os-designate-recordset.editableTypes', + 'designatedashboard.resources.os-designate-recordset.typeMap' + ]; + + /** + * Service to return a schema form config for action forms. Especially useful for forms + * like create and update that differ only in the readonly state of certain form fields. + * + * @returns {object} A schema form config + */ + function service(editableTypes, typeMap) { + var service = { + getCreateFormConfig: getCreateFormConfig, + getUpdateFormConfig: getUpdateFormConfig + }; + + return service; + + ///////////////// + + /** + * Returns the create form config + * @returns {{schema, form, model}|*} + */ + function getCreateFormConfig() { + return getCreateUpdateFormConfig(false); + } + + /** + * Return the update form config + * @returns {{schema, form, model}|*} + */ + function getUpdateFormConfig() { + return getCreateUpdateFormConfig(true); + } + + /** + * Return the create/update form. The two forms are identical except for + * during update, some fields are read-only. + * + * @param readonly - sets readonly value on form fields that change between update and create + * @returns {object} a schema form config, including default model + */ + function getCreateUpdateFormConfig(readonly) { + return { + "schema": { + "type": "object", + "properties": { + "name": { + "type": "string", + "pattern": /^.+\.$/ + }, + "description": { + "type": "string" + }, + "type": { + "type": "string", + "enum": editableTypes + }, + "ttl": { + "type": "integer", + "minimum": 1, + "maximum": 2147483647 + }, + "records": { + "type": "array", + "items": { + "type": "object", + "properties": { + "record": { + "type": "string" + } + } + }, + "minItems": 1, + "uniqueItems": true + } + } + }, + "form": [ + { + "key": "type", + "readonly": readonly, + "title": gettext("Type"), + "description": gettext("Select the type of record set"), + "type": "select", + "titleMap": editableTypes.map(function toTitleMap(type) { + return { + "value": type, + "name": typeMap[type] + } + }), + "required": true + }, + { + "key": "name", + "readonly": readonly, + "type": "text", + "title": gettext("Name"), + "description": gettext("DNS name for the record set, ending in '.'"), + "validationMessage": gettext("DNS name must end with '.'"), + "placeholder": "www.example.com.", + "required": true + }, + { + "key": "description", + "type": "textarea", + "title": gettext("Description"), + "description": gettext("Details about the zone.") + }, + { + "key": "ttl", + "title": gettext("TTL"), + "description": gettext("Time To Live in seconds."), + "type": "number", + "required": true + }, + { + "key": "records", + "title": gettext("Records"), + "type": "array", + "description": gettext("Records for the record set."), + "add": gettext("Add Record"), + "items": [ + { + "key": "records[].record", + "title": gettext("Record") + } + ], + "required": true + } + ], + "model": { + "type": "A", + "ttl": 3600 + } + }; + } + } +})(); diff --git a/designatedashboard/static/designatedashboard/resources/os-designate-recordset/actions/create.service.js b/designatedashboard/static/designatedashboard/resources/os-designate-recordset/actions/create.service.js new file mode 100644 index 0000000..84c65e2 --- /dev/null +++ b/designatedashboard/static/designatedashboard/resources/os-designate-recordset/actions/create.service.js @@ -0,0 +1,132 @@ +/** + * + * (c) Copyright 2016 Hewlett Packard Enterprise Development Company LP + * + * Licensed under the Apache License, Version 2.0 (the "License"); you may + * not use self file except in compliance with the License. You may obtain + * a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT + * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the + * License for the specific language governing permissions and limitations + * under the License. + */ + +(function () { + 'use strict'; + + angular + .module('designatedashboard.resources.os-designate-recordset.actions') + .factory('designatedashboard.resources.os-designate-recordset.actions.create', action); + + action.$inject = [ + '$q', + 'designatedashboard.resources.os-designate-recordset.actions.common-forms', + 'designatedashboard.resources.os-designate-recordset.api', + 'designatedashboard.resources.os-designate-recordset.resourceType', + 'horizon.app.core.openstack-service-api.policy', + 'horizon.app.core.openstack-service-api.serviceCatalog', + 'horizon.framework.conf.resource-type-registry.service', + 'horizon.framework.widgets.form.ModalFormService', + 'horizon.framework.widgets.toast.service', + 'horizon.framework.widgets.modal-wait-spinner.service' + ]; + + /** + * @ngDoc factory + * @name designatedashboard.resources.os-designate-recordset.actions.create + * + * @Description + * Brings up the Create Record Set modal. + */ + function action($q, + forms, + api, + resourceTypeName, + policy, + serviceCatalog, + registry, + schemaFormModalService, + toast, + waitSpinner) { + var createRecordSetPolicy, dnsServiceEnabled; + var title = gettext("Create Record Set"); + var message = { + success: gettext('Record Set %s was successfully created.') + }; + + var service = { + initScope: initScope, + allowed: allowed, + perform: perform + }; + + return service; + + ///////////////// + + function initScope() { + createRecordSetPolicy = policy.ifAllowed({rules: [['dns', 'create_recordset']]}); + dnsServiceEnabled = serviceCatalog.ifTypeEnabled('dns'); + } + + function allowed(item) { + return $q.all([ + createRecordSetPolicy, + dnsServiceEnabled + ]); + } + + function perform(item) { + var formConfig = forms.getCreateFormConfig(); + + // Store the zone ID so it can be used on submit + formConfig.model.zoneId = item.id; + + formConfig.title = title; + return schemaFormModalService.open(formConfig).then(onSubmit, onCancel); + } + + function onSubmit(context) { + var model = angular.copy(context.model); + var zoneId = model.zoneId; + delete model.zoneId; + + // schema form doesn't appear to support populating arrays directly + // Map the records objects to simple array of records + var records = context.model.records.map(function (item) { + return item.record; + }); + model.records = records; + + waitSpinner.showModalSpinner(gettext('Creating Record Set')); + return api.create(zoneId, model).then(onSuccess, onFailure); + } + + function onCancel() { + waitSpinner.hideModalSpinner(); + } + + function onSuccess(response) { + waitSpinner.hideModalSpinner(); + var zone = response.data; + toast.add('success', interpolate(message.success, [zone.name])); + + // To make the result of this action generically useful, reformat the return + // from the deleteModal into a standard form + return { + created: [{type: resourceTypeName, id: zone.id}], + updated: [], + deleted: [], + failed: [] + }; + } + + function onFailure() { + waitSpinner.hideModalSpinner(); + } + } +})(); diff --git a/designatedashboard/static/designatedashboard/resources/os-designate-recordset/actions/delete.service.js b/designatedashboard/static/designatedashboard/resources/os-designate-recordset/actions/delete.service.js new file mode 100644 index 0000000..1d36e9a --- /dev/null +++ b/designatedashboard/static/designatedashboard/resources/os-designate-recordset/actions/delete.service.js @@ -0,0 +1,182 @@ +/** + * (c) Copyright 2016 Hewlett Packard Enterprise Development LP + * + * Licensed under the Apache License, Version 2.0 (the "License"); you may + * not use self file except in compliance with the License. You may obtain + * a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT + * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the + * License for the specific language governing permissions and limitations + * under the License. + */ + +(function() { + 'use strict'; + + angular + .module('designatedashboard.resources.os-designate-recordset.actions') + .factory('designatedashboard.resources.os-designate-recordset.actions.delete', action); + + action.$inject = [ + '$q', + 'designatedashboard.resources.os-designate-recordset.api', + 'designatedashboard.resources.os-designate-recordset.editableTypes', + 'designatedashboard.resources.os-designate-recordset.resourceType', + 'horizon.app.core.openstack-service-api.policy', + 'horizon.framework.util.actions.action-result.service', + 'horizon.framework.util.i18n.gettext', + 'horizon.framework.util.q.extensions', + 'horizon.framework.widgets.modal.deleteModalService', + 'horizon.framework.widgets.toast.service' + ]; + + /* + * @ngdoc factory + * @name designatedashboard.resources.os-designate-recordset.actions.delete + * + * @Description + * Brings up the delete recordset confirmation modal dialog. + + * On submit, delete given recordset. + * On cancel, do nothing. + */ + function action( + $q, + recordsetApi, + editableTypes, + resourceType, + policy, + actionResultService, + gettext, + $qExtensions, + deleteModal, + toast + ) { + var scope, context, deletePromise; + var notAllowedMessage = gettext("You are not allowed to delete record sets: %s"); + var allowedRecordsets = []; + + var service = { + initScope: initScope, + allowed: allowed, + perform: perform + }; + + return service; + + ////////////// + + function initScope(newScope) { + scope = newScope; + context = { }; + deletePromise = policy.ifAllowed({rules: [['dns', 'delete_recordset']]}); + } + + function perform(items) { + var recordsets = angular.isArray(items) ? items : [items]; + context.labels = labelize(recordsets.length); + context.deleteEntity = deleteRecordSet; + return $qExtensions.allSettled(recordsets.map(checkPermission)).then(afterCheck); + } + + function allowed(recordset) { + // only row actions pass in recordset + // otherwise, assume it is a batch action + if (recordset) { + return $q.all([ + deletePromise, + editableRecordType(recordset) + ]); + } else { + return policy.ifAllowed({ rules: [['dns', 'delete_recordset']] }); + } + } + + function checkPermission(recordset) { + return {promise: allowed(recordset), context: recordset}; + } + + function afterCheck(result) { + var outcome = $q.reject(); // Reject the promise by default + if (result.fail.length > 0) { + toast.add('error', getMessage(notAllowedMessage, result.fail)); + outcome = $q.reject(result.fail); + } + if (result.pass.length > 0) { + // Remember the record sets we are allowed to delete so that on delete modal submit + // we can map the recordset ID back to the full recordset. Then we can fetch the + // corresponding zone ID + allowedRecordsets = result.pass.map(getEntity) + outcome = deleteModal.open(scope, allowedRecordsets, context).then(createResult); + } + return outcome; + } + + function createResult(deleteModalResult) { + // To make the result of this action generically useful, reformat the return + // from the deleteModal into a standard form + var actionResult = actionResultService.getActionResult(); + deleteModalResult.pass.forEach(function markDeleted(item) { + actionResult.deleted(resourceType, getEntity(item).id); + }); + deleteModalResult.fail.forEach(function markFailed(item) { + actionResult.failed(resourceType, getEntity(item).id); + }); + return actionResult.result; + } + + function labelize(count) { + return { + + title: ngettext( + 'Confirm Delete Record Set', + 'Confirm Delete Record Sets', count), + + message: ngettext( + 'You have selected "%s". Deleted record set is not recoverable.', + 'You have selected "%s". Deleted record sets are not recoverable.', count), + + submit: ngettext( + 'Delete Record Set', + 'Delete Record Sets', count), + + success: ngettext( + 'Deleted Record Set: %s.', + 'Deleted Record Sets: %s.', count), + + error: ngettext( + 'Unable to delete Record Set: %s.', + 'Unable to delete Record Sets: %s.', count) + }; + } + + function editableRecordType(recordset) { + return $qExtensions.booleanAsPromise( + editableTypes.indexOf(recordset.type) > -1 + ); + } + + function deleteRecordSet(recordSetId) { + var recordSet = allowedRecordsets.find(function(element) { + return element.id === recordSetId; + }) + return recordsetApi.deleteRecordSet(recordSet.zone_id, recordSet.id); + } + + function getMessage(message, entities) { + return interpolate(message, [entities.map(getName).join(", ")]); + } + + function getName(result) { + return getEntity(result).name; + } + + function getEntity(result) { + return result.context; + } + } +})(); diff --git a/designatedashboard/static/designatedashboard/resources/os-designate-recordset/actions/update.service.js b/designatedashboard/static/designatedashboard/resources/os-designate-recordset/actions/update.service.js new file mode 100644 index 0000000..caca961 --- /dev/null +++ b/designatedashboard/static/designatedashboard/resources/os-designate-recordset/actions/update.service.js @@ -0,0 +1,159 @@ +/** + * + * (c) Copyright 2016 Hewlett Packard Enterprise Development Company LP + * + * Licensed under the Apache License, Version 2.0 (the "License"); you may + * not use self file except in compliance with the License. You may obtain + * a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT + * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the + * License for the specific language governing permissions and limitations + * under the License. + */ + +(function () { + 'use strict'; + + angular + .module('designatedashboard.resources.os-designate-recordset.actions') + .factory('designatedashboard.resources.os-designate-recordset.actions.update', action); + + action.$inject = [ + '$q', + 'designatedashboard.resources.util', + 'designatedashboard.resources.os-designate-recordset.actions.common-forms', + 'designatedashboard.resources.os-designate-recordset.api', + 'designatedashboard.resources.os-designate-recordset.editableTypes', + 'designatedashboard.resources.os-designate-recordset.resourceType', + 'horizon.app.core.openstack-service-api.policy', + 'horizon.app.core.openstack-service-api.serviceCatalog', + 'horizon.framework.util.q.extensions', + 'horizon.framework.widgets.form.ModalFormService', + 'horizon.framework.widgets.toast.service', + 'horizon.framework.widgets.modal-wait-spinner.service' + ]; + + /** + * @ngDoc factory + * @name designatedashboard.resources.os-designate-recordset.actions.update + * + * @Description + * Brings up the Update modal. + */ + function action($q, + util, + forms, + api, + editableTypes, + resourceTypeName, + policy, + serviceCatalog, + $qExtensions, + schemaFormModalService, + toast, + waitSpinner) { + var updateRecordSetPolicy, dnsServiceEnabled; + var title = gettext("Update Record Set"); + var message = { + success: gettext('Record Set %s was successfully updated.') + }; + + var service = { + initScope: initScope, + allowed: allowed, + perform: perform + }; + + return service; + + ///////////////// + + function initScope() { + updateRecordSetPolicy = policy.ifAllowed({rules: [['dns', 'update_recordset']]}); + dnsServiceEnabled = serviceCatalog.ifTypeEnabled('dns'); + } + + function allowed(recordset) { + // only supports row action (exactly 1 recordset) + if (recordset) { + return $q.all([ + updateRecordSetPolicy, + util.notDeleted(recordset), + util.notPending(recordset), + editableRecordType(recordset) + ]); + } else { + return false; + } + } + + function editableRecordType(recordset) { + return $qExtensions.booleanAsPromise( + editableTypes.indexOf(recordset.type) > -1 + ); + } + + function perform(item) { + var formConfig = forms.getUpdateFormConfig(); + formConfig.title = title; + formConfig.model = util.getModel(formConfig.form, item); + + // Append the id and zoneId so it can be used on submit + formConfig.model.id = item.id; + formConfig.model.zoneId = item.zone_id; + + // schema form doesn't appear to support populating the records array directly + // Map the records objects to record objects + if (item.hasOwnProperty("records")) { + var records = item.records.map(function (item) { + return {"record": item} + }); + formConfig.model.records = records; + } + return schemaFormModalService.open(formConfig).then(onSubmit, onCancel); + } + + function onSubmit(context) { + var model = angular.copy(context.model); + // schema form doesn't appear to support populating the records array directly + // Map the records objects to simple array of records + if (context.model.hasOwnProperty("records")) { + var records = context.model.records.map(function (item) { + return item.record; + }); + model.records = records; + } + + waitSpinner.showModalSpinner(gettext('Updating Record Set')); + + return api.update(model.zoneId, model.id, model).then(onSuccess, onFailure); + } + + function onCancel() { + waitSpinner.hideModalSpinner(); + } + + function onSuccess(response) { + waitSpinner.hideModalSpinner(); + var recordset = response.data; + toast.add('success', interpolate(message.success, [recordset.name])); + + // To make the result of this action generically useful, reformat the return + // from the deleteModal into a standard form + return { + created: [], + updated: [{type: resourceTypeName, id: recordset.id}], + deleted: [], + failed: [] + }; + } + + function onFailure() { + waitSpinner.hideModalSpinner(); + } + } +})(); diff --git a/designatedashboard/static/designatedashboard/resources/os-designate-recordset/api.service.js b/designatedashboard/static/designatedashboard/resources/os-designate-recordset/api.service.js new file mode 100644 index 0000000..25cb51f --- /dev/null +++ b/designatedashboard/static/designatedashboard/resources/os-designate-recordset/api.service.js @@ -0,0 +1,136 @@ +/** + * (c) Copyright 2016 Hewlett Packard Enterprise Development LP + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +(function () { + 'use strict'; + + angular + .module('designatedashboard.resources.os-designate-recordset') + .factory('designatedashboard.resources.os-designate-recordset.api', apiService); + + apiService.$inject = [ + '$q', + 'designatedashboard.apiPassthroughUrl', + 'horizon.framework.util.http.service', + 'horizon.framework.widgets.toast.service' + ]; + + /** + * @ngdoc service + * @param {Object} httpService + * @param {Object} toastService + * @name apiService + * @description Provides direct access to Designate Record Set APIs. + * @returns {Object} The service + */ + function apiService($q, apiPassthroughUrl, httpService, toastService) { + var service = { + get: get, + list: list, + deleteRecordSet: deleteRecordSet, + create: create, + update: update + }; + + return service; + + /////////////// + + /** + * @name list + * @description + * Get a list of record sets. + * + * The listing result is an object with property "items." Each item is + * a record set. + * + * @param {Object} params + * Query parameters. Optional. + * + * @returns {Object} The result of the API call + */ + function list(zoneId, params) { + return httpService.get(apiPassthroughUrl + 'v2/zones/' + zoneId + '/recordsets/', params) + .error(function () { + toastService.add('error', gettext('Unable to retrieve the record sets.')); + }); + } + + /** + * @name get + * @description + * Get a single record set by ID. + * + * @param {string} zoneId + * Specifies the id of the zone containing the record set to request. + * + * @param {string} recordSetId + * Specifies the id of the record set to request. + * + * @returns {Object} The result of the API call + */ + function get(zoneId, recordSetId) { + // Unfortunately routed-details-view is not happy when load fails, which is + // common when then delete action removes a record set. Mask this failure by + // always returning a successful promise instead of terminating the $http promise + // in the .error handler. + return httpService.get(apiPassthroughUrl + 'v2/zones/' + zoneId + '/recordsets/' + recordSetId + '/') + .then(undefined, function onError() { + toastService.add('error', gettext('Unable to retrieve the record set.')); + return $q.when({}); + }); + } + + /** + * @name delete + * @description + * Delete a single record set by ID + * @param {string} zoneId + * The id of the zone containing the recordset + * + * @param {string} recordSetId + * The id of the recordset within the zone + * + * @returns {*} + */ + function deleteRecordSet(zoneId, recordSetId) { + return httpService.delete(apiPassthroughUrl + 'v2/zones/' + zoneId + '/recordsets/' + recordSetId + '/') + .error(function () { + toastService.add('error', gettext('Unable to delete the record set.')); + }); + } + + function create(zoneId, data) { + return httpService.post(apiPassthroughUrl + 'v2/zones/' + zoneId + '/recordsets/', data) + .error(function () { + toastService.add('error', gettext('Unable to create the record set.')); + }); + } + + function update(zoneId, recordSetId, data) { + // The update API will not accept extra data. Restrict the input to only the allowed + // fields + var apiData = { + ttl: data.ttl, + description: data.description, + records: data.records + }; + return httpService.put(apiPassthroughUrl + 'v2/zones/' + zoneId + '/recordsets/' + recordSetId, apiData) + .error(function () { + toastService.add('error', gettext('Unable to update the record set.')); + }); + } + } +}()); diff --git a/designatedashboard/static/designatedashboard/resources/os-designate-recordset/details/details.module.js b/designatedashboard/static/designatedashboard/resources/os-designate-recordset/details/details.module.js new file mode 100644 index 0000000..8c407b2 --- /dev/null +++ b/designatedashboard/static/designatedashboard/resources/os-designate-recordset/details/details.module.js @@ -0,0 +1,109 @@ +/** + * (c) Copyright 2016 Hewlett Packard Enterprise Development LP + * + * Licensed under the Apache License, Version 2.0 (the "License"); you may + * not use this file except in compliance with the License. You may obtain + * a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT + * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the + * License for the specific language governing permissions and limitations + * under the License. + */ + +(function() { + 'use strict'; + + /** + * @ngdoc overview + * @ngname designatedashboard.resources.os-designate-recordset.details + * + * @description + * Provides details features for record sets. + */ + angular.module('designatedashboard.resources.os-designate-recordset.details', + ['horizon.framework.conf', 'horizon.app.core']) + .run(run); + + run.$inject = [ + 'designatedashboard.resources.os-designate-recordset.resourceType', + 'designatedashboard.resources.os-designate-recordset.api', + 'designatedashboard.resources.os-designate-recordset.basePath', + 'horizon.framework.conf.resource-type-registry.service', + ]; + + function run( + recordSetResourceType, + recordSetApi, + basePath, + registry + ) { + var resourceType = registry.getResourceType(recordSetResourceType); + resourceType + .setLoadFunction(loadFunction) + .setPathGenerator(pathGenerator) + .setPathParser(pathParser) + .setSummaryTemplateUrl(basePath + 'details/drawer.html'); + + /** + * + * @param identifier + * The object returned by the pathParser containing the zone ID and record set ID to load + */ + function loadFunction(identifier) { + return recordSetApi.get(identifier.zoneId, identifier.recordSetId); + } + + /** + * Because a record set is contained by a zone, we implement a custom + * pathGenerator to encode the zone ID and record set ID for the generic + * details panel. + * + * @param item + * A record set + * + * @returns {string} In format "/" + */ + function pathGenerator(item) { + return item.zone_id + '/' + item.id; + } + + /** + * Given a path, extract the zone and record set ids + * + * @param path + * created by pathGenerator + * + * @returns {{zoneId: *, recordSetId: *}} + * The identifier to pass to the load function needed to uniquely identify + * a record set. + */ + function pathParser(path) { + var split = path.split('/'); + return { + zoneId: split[0], + recordSetId: split[1] + } + } + + resourceType.detailsViews + .prepend({ + id: 'recordsetDetailsOverview', + name: gettext('Overview'), + template: basePath + 'details/overview.html', + }, 0); + + // Append a record set view to the zones resource view + var zoneResourceType = registry.getResourceType("OS::Designate::Zone"); + zoneResourceType.detailsViews + .append({ + id: 'zoneRecordSets', + name: gettext('Record Sets'), + template: basePath + 'details/zone-recordsets.html', + }); + } + +})(); diff --git a/designatedashboard/static/designatedashboard/resources/os-designate-recordset/details/drawer.html b/designatedashboard/static/designatedashboard/resources/os-designate-recordset/details/drawer.html new file mode 100644 index 0000000..28762b4 --- /dev/null +++ b/designatedashboard/static/designatedashboard/resources/os-designate-recordset/details/drawer.html @@ -0,0 +1,10 @@ + + \ No newline at end of file diff --git a/designatedashboard/static/designatedashboard/resources/os-designate-recordset/details/overview.controller.js b/designatedashboard/static/designatedashboard/resources/os-designate-recordset/details/overview.controller.js new file mode 100644 index 0000000..0340d2e --- /dev/null +++ b/designatedashboard/static/designatedashboard/resources/os-designate-recordset/details/overview.controller.js @@ -0,0 +1,46 @@ +/** + * (c) Copyright 2016 Hewlett Packard Enterprise Development LP + * + * Licensed under the Apache License, Version 2.0 (the 'License'); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an 'AS IS' BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +(function() { + "use strict"; + + angular + .module('designatedashboard.resources.os-designate-recordset') + .controller('designatedashboard.resources.os-designate-recordset.detailController', controller); + + controller.$inject = [ + 'designatedashboard.resources.os-designate-recordset.resourceType', + 'horizon.framework.conf.resource-type-registry.service', + '$scope' + ]; + + function controller( + resourceTypeCode, + registry, + $scope + ) { + var ctrl = this; + + ctrl.item = {}; + ctrl.resourceType = registry.getResourceType(resourceTypeCode); + + $scope.context.loadPromise.then(onGetResponse); + + function onGetResponse(response) { + ctrl.item = response.data; + } + } + +})(); diff --git a/designatedashboard/static/designatedashboard/resources/os-designate-recordset/details/overview.html b/designatedashboard/static/designatedashboard/resources/os-designate-recordset/details/overview.html new file mode 100644 index 0000000..5d48893 --- /dev/null +++ b/designatedashboard/static/designatedashboard/resources/os-designate-recordset/details/overview.html @@ -0,0 +1,45 @@ +
+
+
+

Details

+
+ + +
+
+

Associations

+
+ + +
+
+
+
+

Modification Times

+
+ + +
+
+
diff --git a/designatedashboard/static/designatedashboard/resources/os-designate-recordset/details/zone-recordsets.controller.js b/designatedashboard/static/designatedashboard/resources/os-designate-recordset/details/zone-recordsets.controller.js new file mode 100644 index 0000000..44184d5 --- /dev/null +++ b/designatedashboard/static/designatedashboard/resources/os-designate-recordset/details/zone-recordsets.controller.js @@ -0,0 +1,39 @@ +/** + * (c) Copyright 2016 Hewlett Packard Enterprise Development LP + * + * Licensed under the Apache License, Version 2.0 (the 'License'); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an 'AS IS' BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +(function() { + "use strict"; + + angular + .module('designatedashboard.resources.os-designate-recordset') + .controller('designatedashboard.resources.os-designate-recordset.zoneRecordSetsController', controller); + + controller.$inject = [ + '$scope', + 'designatedashboard.resources.os-designate-recordset.resourceType' + ]; + + function controller( + $scope, + resourceTypeName + ) { + var ctrl = this; + + ctrl.item = {}; + ctrl.resourceTypeName = resourceTypeName; + ctrl.extraListParams = { zoneId: $scope.context.identifier }; + } + +})(); diff --git a/designatedashboard/static/designatedashboard/resources/os-designate-recordset/details/zone-recordsets.html b/designatedashboard/static/designatedashboard/resources/os-designate-recordset/details/zone-recordsets.html new file mode 100644 index 0000000..d499fe4 --- /dev/null +++ b/designatedashboard/static/designatedashboard/resources/os-designate-recordset/details/zone-recordsets.html @@ -0,0 +1,8 @@ +
+ + +
\ No newline at end of file diff --git a/designatedashboard/static/designatedashboard/resources/os-designate-recordset/os-designate-recordset.module.js b/designatedashboard/static/designatedashboard/resources/os-designate-recordset/os-designate-recordset.module.js new file mode 100644 index 0000000..7feff60 --- /dev/null +++ b/designatedashboard/static/designatedashboard/resources/os-designate-recordset/os-designate-recordset.module.js @@ -0,0 +1,240 @@ +/** + * (c) Copyright 2016 Hewlett Packard Enterprise Development LP + * + * Licensed under the Apache License, Version 2.0 (the "License"); you may + * not use this file except in compliance with the License. You may obtain + * a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT + * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the + * License for the specific language governing permissions and limitations + * under the License. + */ + +(function () { + 'use strict'; + + /** + * @ngdoc overview + * @ngname designatedashboard.resources.os-designate-recordset + * + * @description + * Provides all of the services and widgets required + * to support and display DNS (designate) record set related content. + */ + angular + .module('designatedashboard.resources.os-designate-recordset', [ + 'ngRoute', + 'designatedashboard.resources.os-designate-recordset.actions', + 'designatedashboard.resources.os-designate-recordset.details' + ]) + .constant( + 'designatedashboard.resources.os-designate-recordset.resourceType', + 'OS::Designate::RecordSet') + .constant( + 'designatedashboard.resources.os-designate-recordset.typeMap', + { + 'A': gettext('A - Address record'), + 'AAAA': gettext('AAAA - IPv6 address record'), + 'CNAME': gettext('CNAME - Canonical name record'), + 'MX': gettext('MX - Mail exchange record'), + 'PTR': gettext('PTR - Pointer record'), + 'SPF': gettext('SPR - Sender Policy Framework'), + 'SRV': gettext('SRV - Service locator'), + 'SSHFP': gettext('SSHFP - SSH Public Key Fingerprint'), + 'TXT': gettext('TXT - Text record'), + 'SOA': gettext('SOA - Start of authority record'), + 'NS': gettext('NS - Name server') + }) + .constant( + 'designatedashboard.resources.os-designate-recordset.editableTypes', + [ + "A", + "AAAA", + "CNAME", + "MX", + "PTR", + "SPF", + "SRV", + "SSHFP", + "TXT" + ]) + .config(config) + .run(run); + + config.$inject = ['$provide', '$windowProvider']; + + function config($provide, $windowProvider) { + var path = $windowProvider.$get().STATIC_URL + 'designatedashboard/resources/os-designate-recordset/'; + $provide.constant('designatedashboard.resources.os-designate-recordset.basePath', path); + } + + run.$inject = [ + 'horizon.app.core.detailRoute', + 'horizon.framework.conf.resource-type-registry.service', + 'designatedashboard.resources.os-designate-recordset.api', + 'designatedashboard.resources.os-designate-recordset.resourceType', + 'designatedashboard.resources.os-designate-recordset.typeMap', + 'designatedashboard.resources.util' + ]; + + function run(detailRoute, + registry, + recordSetApi, + resourceTypeString, + typeMap, + util) { + var resourceType = registry.getResourceType(resourceTypeString); + resourceType + .setNames(gettext('DNS Record Set'), gettext('DNS Record Sets')) + .setListFunction(list) + .setProperty('id', { + label: gettext('ID') + }) + .setProperty('zone_id', { + label: gettext('Zone ID'), + filters: ['noValue'] + }) + .setProperty('zone_name', { + label: gettext('Zone Name'), + filters: ['noName'] + }) + .setProperty('project_id', { + label: gettext('Project ID'), + filters: ['noValue'] + }) + .setProperty('name', { + label: gettext('Name'), + filters: ['noName'] + }) + .setProperty('description', { + label: gettext('Description'), + filters: ['noName'] + }) + .setProperty('type', { + label: gettext('Type'), + filters: ['uppercase'], + values: typeMap + }) + .setProperty('ttl', { + label: gettext('Time To Live'), + filters: ['noValue'] + }) + .setProperty('records', { + label: gettext('Records'), + filters: ['noValue'] + }) + .setProperty('notes', { + label: gettext('Notes'), + filters: ['noName'] + }) + .setProperty('status', { + label: gettext('Status'), + filters: ['lowercase', 'noName'], + values: util.statusMap() + }) + .setProperty('action', { + label: gettext('Action'), + filters: ['lowercase', 'noName'], + values: util.actionMap() + }) + .setProperty('version', { + label: gettext('Version'), + filters: ['noValue'] + }) + .setProperty('created_at', { + label: gettext('Created At'), + filters: ['noValue'] + }) + .setProperty('updated_at', { + label: gettext('Updated At'), + filters: ['noValue'] + }); + + resourceType + .tableColumns + .append({ + id: 'name', + priority: 1, + sortDefault: true, + filters: ['noName'], + // For link format, see pathGenerator in details.module.js + template: '{$ item.name $}' + }) + .append({ + id: 'type', + priority: 2, + filters: ['uppercase'], + values: typeMap + }) + .append({ + id: 'records', + priority: 2, + filters: ['noValue'] + }) + .append({ + id: 'status', + filters: ['lowercase'], + values: util.statusMap(), + priority: 2 + }); + + resourceType + .filterFacets + .append({ + label: gettext('Type'), + name: 'type', + isServer: false, + singleton: true, + persistent: false, + options: Object.keys(typeMap).map(function toOptionLabel(key) { + return { + label: typeMap[key], + key: key + } + }) + }) + .append({ + label: gettext('Name'), + name: 'name', + isServer: false, + singleton: true, + persistent: false + }) + .append({ + label: gettext('Status'), + name: 'status', + isServer: false, + singleton: true, + persistent: false, + options: [ + {label: gettext('Active'), key: 'active'}, + {label: gettext('Pending'), key: 'pending'} + ] + }); + + /** + * list all recordsets within a zone. Requires "zoneId" in the params. All other + * params will be passed unmodified as URL params to the API. + * + * @param params + * zoneId (required) list recordsets within the zone + * + * @returns {*|Object} + */ + function list(params) { + return recordSetApi.list(params.zoneId, params).then(function onList(response) { + // listFunctions are expected to return data in "items" + response.data.items = response.data.recordsets; + + util.addTimestampIds(response.data.items); + + return response; + }); + } + } + +})(); diff --git a/designatedashboard/static/designatedashboard/resources/os-designate-zone/actions/actions.module.js b/designatedashboard/static/designatedashboard/resources/os-designate-zone/actions/actions.module.js new file mode 100644 index 0000000..9ccdd04 --- /dev/null +++ b/designatedashboard/static/designatedashboard/resources/os-designate-zone/actions/actions.module.js @@ -0,0 +1,76 @@ +/** + * (c) Copyright 2016 Hewlett Packard Enterprise Development LP + * + * Licensed under the Apache License, Version 2.0 (the "License"); you may + * not use this file except in compliance with the License. You may obtain + * a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT + * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the + * License for the specific language governing permissions and limitations + * under the License. + */ + +(function () { + 'use strict'; + + /** + * @ngdoc overview + * @ngname designatedashboard.resources.os-designate-zone.actions + * + * @description + * Provides all of the actions for DNS Zones. + */ + angular.module('designatedashboard.resources.os-designate-zone.actions', [ + 'horizon.framework.conf', + 'horizon.app.core' + ]) + .run(run); + + run.$inject = [ + 'horizon.framework.conf.resource-type-registry.service', + 'designatedashboard.resources.os-designate-zone.resourceType', + 'designatedashboard.resources.os-designate-zone.actions.create', + 'designatedashboard.resources.os-designate-zone.actions.delete', + 'designatedashboard.resources.os-designate-zone.actions.update' + ]; + + function run(registry, + resourceTypeString, + createAction, + deleteAction, + updateAction) { + var resourceType = registry.getResourceType(resourceTypeString); + resourceType + .globalActions + .append({ + id: 'create', + service: createAction, + template: { + text: gettext('Create Zone') + } + }); + + resourceType + .itemActions + .append({ + id: 'update', + service: updateAction, + template: { + text: gettext('Update') + } + }) + .append({ + id: 'delete', + service: deleteAction, + template: { + text: gettext('Delete'), + type: 'delete' + } + }); + } + +})(); diff --git a/designatedashboard/static/designatedashboard/resources/os-designate-zone/actions/actions.module.spec.js b/designatedashboard/static/designatedashboard/resources/os-designate-zone/actions/actions.module.spec.js new file mode 100644 index 0000000..f4bcda2 --- /dev/null +++ b/designatedashboard/static/designatedashboard/resources/os-designate-zone/actions/actions.module.spec.js @@ -0,0 +1,49 @@ +/** + * + * Licensed under the Apache License, Version 2.0 (the "License"); you may + * not use this file except in compliance with the License. You may obtain + * a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT + * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the + * License for the specific language governing permissions and limitations + * under the License. + */ + +(function() { + 'use strict'; + + describe('zone actions module', function() { + var registry; + beforeEach(module('designatedashboard')); + + beforeEach(inject(function($injector) { + registry = $injector.get('horizon.framework.conf.resource-type-registry.service'); + })); + + it('registers Create Zone as a global action', function() { + var actions = registry.getResourceType('OS::Designate::Zone').globalActions; + expect(actionHasId(actions, 'create')).toBe(true); + }); + + it('registers Update Zone as a item action', function() { + var actions = registry.getResourceType('OS::Designate::Zone').itemActions; + expect(actionHasId(actions, 'update')).toBe(true); + }); + + function actionHasId(list, value) { + return list.filter(matchesId).length === 1; + + function matchesId(action) { + if (action.id === value) { + return true; + } + } + } + + }); + +})(); \ No newline at end of file diff --git a/designatedashboard/static/designatedashboard/resources/os-designate-zone/actions/common-forms.service.js b/designatedashboard/static/designatedashboard/resources/os-designate-zone/actions/common-forms.service.js new file mode 100644 index 0000000..55df08f --- /dev/null +++ b/designatedashboard/static/designatedashboard/resources/os-designate-zone/actions/common-forms.service.js @@ -0,0 +1,185 @@ +/** + * + * (c) Copyright 2016 Hewlett Packard Enterprise Development Company LP + * + * Licensed under the Apache License, Version 2.0 (the "License"); you may + * not use self file except in compliance with the License. You may obtain + * a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT + * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the + * License for the specific language governing permissions and limitations + * under the License. + */ + +(function () { + 'use strict'; + + angular + .module('designatedashboard.resources.os-designate-zone.actions') + .factory('designatedashboard.resources.os-designate-zone.actions.common-forms', service); + + service.$inject = [ + ]; + + /** + * Service to return a schema form config for action forms. Especially useful for forms + * like create and update that differ only in the readonly state of certain form fields. + * + * @returns {object} A schema form config + */ + function service() { + var service = { + getCreateFormConfig: getCreateFormConfig, + getUpdateFormConfig: getUpdateFormConfig + }; + + return service; + + ///////////////// + + /** + * Returns the create zone form config + * @returns {{schema, form, model}|*} + */ + function getCreateFormConfig() { + return getCreateUpdateFormConfig(false); + } + + /** + * Return the update zone form config + * @returns {{schema, form, model}|*} + */ + function getUpdateFormConfig() { + return getCreateUpdateFormConfig(true); + } + + /** + * Return the create/update zone form. The two forms are identical except for + * during update, some fields are read-only. + * + * @param readonly - sets readonly value on form fields that change between update and create + * @returns {object} a schema form config, including default model + */ + function getCreateUpdateFormConfig(readonly) { + return { + "schema": { + "type": "object", + "properties": { + "name": { + "type": "string", + "pattern": /^.+\.$/ + }, + "description": { + "type": "string" + }, + "email": { + "type": "string", + "format": "email", + "pattern": /^[^@]+@[^@]+$/ + }, + "type": { + "type": "string", + "enum": [ + "PRIMARY", + "SECONDARY" + ] + }, + "ttl": { + "type": "integer", + "minimum": 1, + "maximum": 2147483647 + }, + "masters": { + "type": "array", + "items": { + "type": "object", + "properties": { + "address": { + "type": "string" + } + } + }, + "minItems": 1, + "uniqueItems": true + } + } + }, + "form": [ + { + "key": "name", + "readonly": readonly, + "title": gettext("Name"), + "description": gettext("Zone name ending in '.'"), + "validationMessage": gettext("Zone must end with '.'"), + "placeholder": "example.com.", + "type": "text", + "required": true + }, + { + "key": "description", + "type": "textarea", + "title": gettext("Description"), + "description": gettext("Details about the zone.") + }, + { + "key": "email", + "title": gettext("Email Address"), + "description": gettext("Email address to contact the zone owner."), + "validationMessage": gettext("Email address must contain a single '@' character"), + "type": "text", + "condition": "model.type == 'PRIMARY'", + "required": true + }, + { + "key": "ttl", + "title": gettext("TTL"), + "description": gettext("Time To Live in seconds."), + "type": "number", + "condition": "model.type == 'PRIMARY'", + "required": true + }, + { + "key": "type", + "readonly": readonly, + "title": gettext("Type"), + "description": gettext("Select the type of zone"), + "type": "select", + "titleMap": [ + { + "value": "PRIMARY", + "name": gettext("Primary") + }, + { + "value": "SECONDARY", + "name": gettext("Secondary") + } + ] + }, + { + "key": "masters", + "readonly": readonly, + "title": gettext("Masters"), + "type": "array", + "description": gettext("DNS master(s) for the Secondary zone."), + "condition": "model.type == 'SECONDARY'", + "add": gettext("Add Master"), + "items": [ + { + "key": "masters[].address", + "title": gettext("IP Address") + } + ] + } + ], + "model": { + "type": "PRIMARY", + "ttl": 3600 + } + }; + } + } +})(); diff --git a/designatedashboard/static/designatedashboard/resources/os-designate-zone/actions/create.html b/designatedashboard/static/designatedashboard/resources/os-designate-zone/actions/create.html new file mode 100644 index 0000000..663a695 --- /dev/null +++ b/designatedashboard/static/designatedashboard/resources/os-designate-zone/actions/create.html @@ -0,0 +1 @@ +

CREATE TEMPLATE

\ No newline at end of file diff --git a/designatedashboard/static/designatedashboard/resources/os-designate-zone/actions/create.service.js b/designatedashboard/static/designatedashboard/resources/os-designate-zone/actions/create.service.js new file mode 100644 index 0000000..ed53aa3 --- /dev/null +++ b/designatedashboard/static/designatedashboard/resources/os-designate-zone/actions/create.service.js @@ -0,0 +1,126 @@ +/** + * + * (c) Copyright 2016 Hewlett Packard Enterprise Development Company LP + * + * Licensed under the Apache License, Version 2.0 (the "License"); you may + * not use self file except in compliance with the License. You may obtain + * a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT + * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the + * License for the specific language governing permissions and limitations + * under the License. + */ + +(function () { + 'use strict'; + + angular + .module('designatedashboard.resources.os-designate-zone.actions') + .factory('designatedashboard.resources.os-designate-zone.actions.create', action); + + action.$inject = [ + '$q', + 'designatedashboard.resources.os-designate-zone.actions.common-forms', + 'designatedashboard.resources.os-designate-zone.api', + 'designatedashboard.resources.os-designate-zone.resourceType', + 'horizon.app.core.openstack-service-api.policy', + 'horizon.app.core.openstack-service-api.serviceCatalog', + 'horizon.framework.widgets.form.ModalFormService', + 'horizon.framework.widgets.toast.service', + 'horizon.framework.widgets.modal-wait-spinner.service' + ]; + + /** + * @ngDoc factory + * @name designatedashboard.resources.os-designate-zone.actions.create + * + * @Description + * Brings up the Create Zone modal. + */ + function action($q, + forms, + api, + resourceTypeName, + policy, + serviceCatalog, + schemaFormModalService, + toast, + waitSpinner) { + var createZonePolicy, dnsServiceEnabled; + var title = gettext("Create Zone"); + var message = { + success: gettext('Zone %s was successfully created.') + }; + + var service = { + initScope: initScope, + allowed: allowed, + perform: perform + }; + + return service; + + ///////////////// + + function initScope() { + createZonePolicy = policy.ifAllowed({rules: [['dns', 'create_zone']]}); + dnsServiceEnabled = serviceCatalog.ifTypeEnabled('dns'); + } + + function allowed() { + return $q.all([ + createZonePolicy, + dnsServiceEnabled + ]); + } + + function perform() { + var formConfig = forms.getCreateFormConfig(); + formConfig.title = title; + return schemaFormModalService.open(formConfig).then(onSubmit, onCancel); + } + + function onSubmit(context) { + var zoneModel = angular.copy(context.model); + // schema form doesn't appear to support populating the masters array directly + // Map the masters objects to simple array of addresses + if (context.model.hasOwnProperty("masters")) { + var masters = context.model.masters.map(function (item) { + return item.address; + }); + zoneModel.masters = masters; + } + + waitSpinner.showModalSpinner(gettext('Creating Zone')); + + return api.create(zoneModel).then(onSuccess, onFailure); + } + + function onCancel() { + waitSpinner.hideModalSpinner(); + } + + function onSuccess(response) { + waitSpinner.hideModalSpinner(); + var zone = response.data; + toast.add('success', interpolate(message.success, [zone.name])); + + // To make the result of this action generically useful, reformat the return + // from the deleteModal into a standard form + return { + created: [{type: resourceTypeName, id: zone.id}], + updated: [], + deleted: [], + failed: [] + }; + } + + function onFailure() { + waitSpinner.hideModalSpinner(); + } + } +})(); diff --git a/designatedashboard/static/designatedashboard/resources/os-designate-zone/actions/delete.service.js b/designatedashboard/static/designatedashboard/resources/os-designate-zone/actions/delete.service.js new file mode 100644 index 0000000..5882dd6 --- /dev/null +++ b/designatedashboard/static/designatedashboard/resources/os-designate-zone/actions/delete.service.js @@ -0,0 +1,169 @@ +/** + * (c) Copyright 2016 Hewlett Packard Enterprise Development LP + * + * Licensed under the Apache License, Version 2.0 (the "License"); you may + * not use self file except in compliance with the License. You may obtain + * a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT + * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the + * License for the specific language governing permissions and limitations + * under the License. + */ + +(function() { + 'use strict'; + + angular + .module('designatedashboard.resources.os-designate-zone.actions') + .factory('designatedashboard.resources.os-designate-zone.actions.delete', action); + + action.$inject = [ + '$q', + 'designatedashboard.resources.os-designate-zone.api', + 'designatedashboard.resources.util', + 'horizon.app.core.openstack-service-api.policy', + 'horizon.framework.util.actions.action-result.service', + 'horizon.framework.util.i18n.gettext', + 'horizon.framework.util.q.extensions', + 'horizon.framework.widgets.modal.deleteModalService', + 'horizon.framework.widgets.toast.service', + 'designatedashboard.resources.os-designate-zone.resourceType' + ]; + + /* + * @ngdoc factory + * @name designatedashboard.resources.os-designate-zone.actions.delete + * + * @Description + * Brings up the delete zone confirmation modal dialog. + + * On submit, delete given zone. + * On cancel, do nothing. + */ + function action( + $q, + zoneApi, + util, + policy, + actionResultService, + gettext, + $qExtensions, + deleteModal, + toast, + resourceType + ) { + var scope, context, deleteZonePromise; + var notAllowedMessage = gettext("You are not allowed to delete zones: %s"); + + var service = { + initScope: initScope, + allowed: allowed, + perform: perform + }; + + return service; + + ////////////// + + function initScope(newScope) { + scope = newScope; + context = { }; + deleteZonePromise = policy.ifAllowed({rules: [['dns', 'delete_zone']]}); + } + + function perform(items) { + var zones = angular.isArray(items) ? items : [items]; + context.labels = labelize(zones.length); + context.deleteEntity = deleteZone; + return $qExtensions.allSettled(zones.map(checkPermission)).then(afterCheck); + } + + function allowed(zone) { + // only row actions pass in zone + // otherwise, assume it is a batch action + if (zone) { + return $q.all([ + deleteZonePromise, + util.notDeleted(zone), + util.notPending(zone) + ]); + } else { + return policy.ifAllowed({ rules: [['dns', 'delete_zone']] }); + } + } + + function checkPermission(zone) { + return {promise: allowed(zone), context: zone}; + } + + function afterCheck(result) { + var outcome = $q.reject(); // Reject the promise by default + if (result.fail.length > 0) { + toast.add('error', getMessage(notAllowedMessage, result.fail)); + outcome = $q.reject(result.fail); + } + if (result.pass.length > 0) { + outcome = deleteModal.open(scope, result.pass.map(getEntity), context).then(createResult); + } + return outcome; + } + + function createResult(deleteModalResult) { + // To make the result of this action generically useful, reformat the return + // from the deleteModal into a standard form + var actionResult = actionResultService.getActionResult(); + deleteModalResult.pass.forEach(function markDeleted(item) { + actionResult.deleted(resourceType, getEntity(item).id); + }); + deleteModalResult.fail.forEach(function markFailed(item) { + actionResult.failed(resourceType, getEntity(item).id); + }); + return actionResult.result; + } + + function labelize(count) { + return { + + title: ngettext( + 'Confirm Delete Zone', + 'Confirm Delete Zones', count), + + message: ngettext( + 'You have selected "%s". Deleted zone is not recoverable.', + 'You have selected "%s". Deleted zones are not recoverable.', count), + + submit: ngettext( + 'Delete Zone', + 'Delete Zones', count), + + success: ngettext( + 'Deleted Zone: %s.', + 'Deleted Zones: %s.', count), + + error: ngettext( + 'Unable to delete Zone: %s.', + 'Unable to delete Zones: %s.', count) + }; + } + + function deleteZone(zone) { + return zoneApi.deleteZone(zone, true); + } + + function getMessage(message, entities) { + return interpolate(message, [entities.map(getName).join(", ")]); + } + + function getName(result) { + return getEntity(result).name; + } + + function getEntity(result) { + return result.context; + } + } +})(); diff --git a/designatedashboard/static/designatedashboard/resources/os-designate-zone/actions/update.service.js b/designatedashboard/static/designatedashboard/resources/os-designate-zone/actions/update.service.js new file mode 100644 index 0000000..2610570 --- /dev/null +++ b/designatedashboard/static/designatedashboard/resources/os-designate-zone/actions/update.service.js @@ -0,0 +1,147 @@ +/** + * + * (c) Copyright 2016 Hewlett Packard Enterprise Development Company LP + * + * Licensed under the Apache License, Version 2.0 (the "License"); you may + * not use self file except in compliance with the License. You may obtain + * a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT + * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the + * License for the specific language governing permissions and limitations + * under the License. + */ + +(function () { + 'use strict'; + + angular + .module('designatedashboard.resources.os-designate-zone.actions') + .factory('designatedashboard.resources.os-designate-zone.actions.update', action); + + action.$inject = [ + '$q', + 'designatedashboard.resources.os-designate-zone.actions.common-forms', + 'designatedashboard.resources.os-designate-zone.api', + 'designatedashboard.resources.os-designate-zone.resourceType', + 'designatedashboard.resources.util', + 'horizon.app.core.openstack-service-api.policy', + 'horizon.app.core.openstack-service-api.serviceCatalog', + 'horizon.framework.widgets.form.ModalFormService', + 'horizon.framework.widgets.toast.service', + 'horizon.framework.widgets.modal-wait-spinner.service' + ]; + + /** + * @ngDoc factory + * @name designatedashboard.resources.os-designate-zone.actions.update + * + * @Description + * Brings up the Update Zone modal. + */ + function action($q, + forms, + api, + resourceTypeName, + util, + policy, + serviceCatalog, + schemaFormModalService, + toast, + waitSpinner) { + var updateZonePolicy, dnsServiceEnabled; + var title = gettext("Update Zone"); + var message = { + success: gettext('Zone %s was successfully updated.') + }; + + var service = { + initScope: initScope, + allowed: allowed, + perform: perform + }; + + return service; + + ///////////////// + + function initScope() { + updateZonePolicy = policy.ifAllowed({rules: [['dns', 'update_zone']]}); + dnsServiceEnabled = serviceCatalog.ifTypeEnabled('dns'); + } + + function allowed(zone) { + // only supports row action (exactly 1 zone) + if (zone) { + return $q.all([ + updateZonePolicy, + util.notDeleted(zone), + util.notPending(zone) + ]); + } else { + return false; + } + } + + function perform(item) { + var formConfig = forms.getUpdateFormConfig(); + formConfig.title = title; + formConfig.model = util.getModel(formConfig.form, item); + + // Append the id so it can be used on submit + formConfig.model.id = item.id; + + // schema form doesn't appear to support populating the masters array directly + // Map the masters objects to address objects + if (item.hasOwnProperty("masters")) { + var masters = item.masters.map(function (item) { + return { "address": item } + }) + formConfig.masters = masters; + } + return schemaFormModalService.open(formConfig).then(onSubmit, onCancel); + } + + function onSubmit(context) { + var zoneModel = angular.copy(context.model); + // schema form doesn't appear to support populating the masters array directly + // Map the masters objects to simple array of addresses + if (context.model.hasOwnProperty("masters")) { + var masters = context.model.masters.map(function (item) { + return item.address; + }) + zoneModel.masters = masters; + } + + waitSpinner.showModalSpinner(gettext('Updating Zone')); + + return api.update(zoneModel.id, zoneModel).then(onSuccess, onFailure); + } + + function onCancel() { + waitSpinner.hideModalSpinner(); + } + + function onSuccess(response) { + waitSpinner.hideModalSpinner(); + var zone = response.data; + toast.add('success', interpolate(message.success, [zone.name])); + + // To make the result of this action generically useful, reformat the return + // from the deleteModal into a standard form + return { + created: [], + updated: [{type: resourceTypeName, id: zone.id}], + deleted: [], + failed: [] + }; + } + + function onFailure() { + waitSpinner.hideModalSpinner(); + } + } +})(); diff --git a/designatedashboard/static/designatedashboard/resources/os-designate-zone/api.service.js b/designatedashboard/static/designatedashboard/resources/os-designate-zone/api.service.js new file mode 100644 index 0000000..44638cb --- /dev/null +++ b/designatedashboard/static/designatedashboard/resources/os-designate-zone/api.service.js @@ -0,0 +1,152 @@ +/** + * (c) Copyright 2016 Hewlett Packard Enterprise Development LP + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +(function () { + 'use strict'; + + angular + .module('designatedashboard.resources.os-designate-zone') + .factory('designatedashboard.resources.os-designate-zone.api', apiService); + + apiService.$inject = [ + 'designatedashboard.apiPassthroughUrl', + 'horizon.framework.util.http.service', + 'horizon.framework.widgets.toast.service' + ]; + + /** + * @ngdoc service + * @param {Object} httpService + * @param {Object} toastService + * @name apiService + * @description Provides direct access to Designate Zone APIs. + * @returns {Object} The service + */ + function apiService(apiPassthroughUrl, httpService, toastService) { + var service = { + get: get, + list: list, + deleteZone: deleteZone, + create: create, + update: update + }; + + return service; + + /////////////// + + /** + * @name list + * @description + * Get a list of zones. + * + * The listing result is an object with property "items." Each item is + * a zone. + * + * @param {Object} params + * Query parameters. Optional. + * + * @returns {Object} The result of the API call + */ + /* + function list(params) { + var config = params ? {'params': params} : {}; + return httpService.get('/api/designate/zones/', config) + .error(function () { + toastService.add('error', gettext('Unable to retrieve the zone.')); + }); + }*/ + function list(params) { + var config = params ? {'params': params} : {}; + return httpService.get(apiPassthroughUrl + 'v2/zones/', config) + .error(function () { + toastService.add('error', gettext('Unable to retrieve the zone.')); + }); + } + + /** + * @name get + * @description + * Get a single zone by ID. + * + * @param {string} id + * Specifies the id of the zone to request. + * + * @returns {Object} The result of the API call + */ + function get(id) { + return httpService.get(apiPassthroughUrl + 'v2/zones/' + id + '/') + .error(function () { + toastService.add('error', gettext('Unable to retrieve the zone.')); + }); + } + + /** + * @name deleteZone + * @description + * Delete a single zone by ID + * @param id + * @returns {*} + */ + function deleteZone(id) { + return httpService.delete(apiPassthroughUrl + 'v2/zones/' + id + '/') + .error(function () { + toastService.add('error', gettext('Unable to delete the zone.')); + }); + } + + /** + * @name create + * @description + * Create a zone + * + * @param {Object} data + * Specifies the zone information to create + * + * @returns {Object} The created zone object + */ + function create(data) { + return httpService.post(apiPassthroughUrl + 'v2/zones/', data) + .error(function() { + toastService.add('error', gettext('Unable to create the zone.')); + }) + } + + /** + * @name create + * @description + * Update a zone + * + * @param {Object} id - zone id + * @param {Object} data to pass directly to zone update API + * Specifies the zone information to update + * + * @returns {Object} The updated zone object + */ + function update(id, data) { + // The update API will not accept extra data. Restrict the input to only the allowed + // fields + var apiData = { + email: data.email, + ttl: data.ttl, + description: data.description + }; + return httpService.patch(apiPassthroughUrl + 'v2/zones/' + id + '/', apiData ) + .error(function() { + toastService.add('error', gettext('Unable to update the zone.')); + }) + } + } +}()); diff --git a/designatedashboard/static/designatedashboard/resources/os-designate-zone/details/details.module.js b/designatedashboard/static/designatedashboard/resources/os-designate-zone/details/details.module.js new file mode 100644 index 0000000..96a12b6 --- /dev/null +++ b/designatedashboard/static/designatedashboard/resources/os-designate-zone/details/details.module.js @@ -0,0 +1,61 @@ +/** + * (c) Copyright 2016 Hewlett Packard Enterprise Development LP + * + * Licensed under the Apache License, Version 2.0 (the "License"); you may + * not use this file except in compliance with the License. You may obtain + * a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT + * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the + * License for the specific language governing permissions and limitations + * under the License. + */ + +(function() { + 'use strict'; + + /** + * @ngdoc overview + * @ngname designatedashboard.resources.os-designate-zone.details + * + * @description + * Provides details features for zones. + */ + angular.module('designatedashboard.resources.os-designate-zone.details', + ['horizon.framework.conf', 'horizon.app.core']) + .run(run); + + run.$inject = [ + 'designatedashboard.resources.os-designate-zone.resourceType', + 'designatedashboard.resources.os-designate-zone.api', + 'designatedashboard.resources.os-designate-zone.basePath', + 'horizon.framework.conf.resource-type-registry.service' + ]; + + function run( + zoneResourceType, + zoneApi, + basePath, + registry + ) { + var resourceType = registry.getResourceType(zoneResourceType); + resourceType + .setLoadFunction(loadFunction) + .setSummaryTemplateUrl(basePath + 'details/drawer.html'); + + resourceType.detailsViews + .prepend({ + id: 'zoneDetailsOverview', + name: gettext('Overview'), + template: basePath + 'details/overview.html', + }, 0); + + function loadFunction(identifier) { + return zoneApi.get(identifier); + } + } + +})(); diff --git a/designatedashboard/static/designatedashboard/resources/os-designate-zone/details/drawer.html b/designatedashboard/static/designatedashboard/resources/os-designate-zone/details/drawer.html new file mode 100644 index 0000000..0f425b2 --- /dev/null +++ b/designatedashboard/static/designatedashboard/resources/os-designate-zone/details/drawer.html @@ -0,0 +1,10 @@ + + \ No newline at end of file diff --git a/designatedashboard/static/designatedashboard/resources/os-designate-zone/details/overview.controller.js b/designatedashboard/static/designatedashboard/resources/os-designate-zone/details/overview.controller.js new file mode 100644 index 0000000..58c5601 --- /dev/null +++ b/designatedashboard/static/designatedashboard/resources/os-designate-zone/details/overview.controller.js @@ -0,0 +1,55 @@ +/** + * (c) Copyright 2016 Hewlett Packard Enterprise Development LP + * + * Licensed under the Apache License, Version 2.0 (the 'License'); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an 'AS IS' BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +(function() { + "use strict"; + + angular + .module('designatedashboard.resources.os-designate-zone') + .controller('designatedashboard.resources.os-designate-zone.detailController', controller); + + controller.$inject = [ + 'designatedashboard.resources.os-designate-zone.resourceType', + 'horizon.framework.conf.resource-type-registry.service', + '$scope' + ]; + + function controller( + resourceTypeCode, + registry, + $scope + ) { + var ctrl = this; + + ctrl.item; + ctrl.resourceType = registry.getResourceType(resourceTypeCode); + + $scope.context.loadPromise.then(onGetResponse); + + function onGetResponse(response) { + ctrl.item = response.data; + + var attr = ''; + var keys = Object.keys(response.data.attributes); + + for (var i = keys.length - 1; i >= 0; i--) { + attr += keys[i] + ':' + response.data.attributes[keys[i]] + ', '; + } + + ctrl.item.attributes = attr; + } + } + +})(); diff --git a/designatedashboard/static/designatedashboard/resources/os-designate-zone/details/overview.html b/designatedashboard/static/designatedashboard/resources/os-designate-zone/details/overview.html new file mode 100644 index 0000000..6a0c4f9 --- /dev/null +++ b/designatedashboard/static/designatedashboard/resources/os-designate-zone/details/overview.html @@ -0,0 +1,58 @@ +
+
+
+

Details

+
+ + +
+
+

Attributes

+
+ + +
+
+
+
+

Modification Times

+
+ + +
+
+

Associations

+
+ + +
+
+
diff --git a/designatedashboard/static/designatedashboard/resources/os-designate-zone/os-designate-zone.module.js b/designatedashboard/static/designatedashboard/resources/os-designate-zone/os-designate-zone.module.js new file mode 100644 index 0000000..3c8bf61 --- /dev/null +++ b/designatedashboard/static/designatedashboard/resources/os-designate-zone/os-designate-zone.module.js @@ -0,0 +1,204 @@ +/** + * (c) Copyright 2016 Hewlett Packard Enterprise Development LP + * + * Licensed under the Apache License, Version 2.0 (the "License"); you may + * not use this file except in compliance with the License. You may obtain + * a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT + * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the + * License for the specific language governing permissions and limitations + * under the License. + */ + +(function () { + 'use strict'; + + /** + * @ngdoc overview + * @ngname designatedashboard.resources.os-designate-zone + * + * @description + * Provides all of the services and widgets required + * to support and display DNS (designate) zone related content. + */ + angular + .module('designatedashboard.resources.os-designate-zone', [ + 'ngRoute', + 'designatedashboard.resources.os-designate-zone.actions', + 'designatedashboard.resources.os-designate-zone.details' + ]) + .constant( + 'designatedashboard.resources.os-designate-zone.resourceType', + 'OS::Designate::Zone') + .config(config) + .run(run); + + config.$inject = ['$provide', '$windowProvider']; + + function config($provide, $windowProvider) { + var path = $windowProvider.$get().STATIC_URL + 'designatedashboard/resources/os-designate-zone/'; + $provide.constant('designatedashboard.resources.os-designate-zone.basePath', path); + } + + run.$inject = [ + 'horizon.app.core.detailRoute', + 'horizon.framework.conf.resource-type-registry.service', + 'designatedashboard.resources.os-designate-zone.api', + 'designatedashboard.resources.os-designate-zone.resourceType', + 'designatedashboard.resources.util' + ]; + + function run(detailRoute, + registry, + zoneApi, + resourceTypeString, + util) { + var resourceType = registry.getResourceType(resourceTypeString); + resourceType + .setNames(gettext('DNS Zone'), gettext('DNS Zones')) + .setListFunction(listZones) + .setProperty('action', { + label: gettext('Action'), + filters: ['lowercase', 'noName'], + values: util.actionMap() + }) + .setProperty('attributes', { + label: gettext('Attributes') + }) + .setProperty('created_at', { + label: gettext('Created At'), + filters: ['noValue'] + }) + .setProperty('description', { + label: gettext('Description'), + filters: ['noName'] + }) + .setProperty('email', { + label: gettext('Email'), + filters: ['noName'] + }) + .setProperty('id', { + label: gettext('ID') + }) + .setProperty('masters', { + label: gettext('Masters'), + filters: ['noValue'] + }) + .setProperty('name', { + label: gettext('Name'), + filters: ['noName'] + }) + .setProperty('pool_id', { + label: gettext('Pool ID') + }) + .setProperty('project_id', { + label: gettext('Project ID') + }) + .setProperty('serial', { + label: gettext('Serial'), + filters: ['noValue'] + }) + .setProperty('status', { + label: gettext('Status'), + filters: ['lowercase', 'noName'], + values: util.statusMap() + }) + .setProperty('transferred_at', { + label: gettext('Transferred At'), + filters: ['noValue'] + }) + .setProperty('ttl', { + label: gettext('Time To Live'), + filters: ['noValue'] + }) + .setProperty('type', { + label: gettext('Type'), + filters: ['lowercase', 'noName'], + values: typeMap() + }) + .setProperty('updated_at', { + label: gettext('Updated At'), + filters: ['noValue'] + }) + .setProperty('version', { + label: gettext('Version'), + filters: ['noValue'] + }); + + resourceType + .tableColumns + .append({ + id: 'name', + priority: 1, + sortDefault: true, + template: '{$ item.name $}' + }) + .append({ + id: 'type', + filters: ['lowercase'], + values: typeMap(), + priority: 2 + }) + .append({ + id: 'status', + filters: ['lowercase'], + values: util.statusMap(), + priority: 2 + }); + + resourceType + .filterFacets + .append({ + label: gettext('Name'), + name: 'name', + isServer: false, + singleton: true, + persistent: false + }) + .append({ + label: gettext('Type'), + name: 'type', + isServer: false, + singleton: true, + persistent: false, + options: [ + {label: gettext('Primary'), key: 'primary'}, + {label: gettext('Secondary'), key: 'secondary'} + ] + }) + .append({ + label: gettext('Status'), + name: 'status', + isServer: false, + singleton: true, + persistent: false, + options: [ + {label: gettext('Active'), key: 'active'}, + {label: gettext('Pending'), key: 'pending'} + ] + }); + + function typeMap() { + return { + 'primary': gettext('Primary'), + 'secondary': gettext('Secondary') + } + } + + function listZones() { + return zoneApi.list().then(function onList(response) { + // listFunctions are expected to return data in "items" + response.data.items = response.data.zones; + + util.addTimestampIds(response.data.items, 'updated_at'); + + return response; + }); + } + } + +})(); diff --git a/designatedashboard/static/designatedashboard/resources/resources.module.js b/designatedashboard/static/designatedashboard/resources/resources.module.js new file mode 100644 index 0000000..57febbe --- /dev/null +++ b/designatedashboard/static/designatedashboard/resources/resources.module.js @@ -0,0 +1,36 @@ +/* + * (c) Copyright 2016 Hewlett-Packard Development Company, L.P. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +(function () { + 'use strict'; + + /** + * @ngdoc overview + * @name designatedashboard.resources + * @description + * + * # designatedashboard.resources + * + * This module hosts registered resource types. This module file may + * contain individual registrations, or may have sub-modules that + * more fully contain registrations. + */ + angular + .module('designatedashboard.resources', [ + 'designatedashboard.resources.os-designate-recordset', + 'designatedashboard.resources.os-designate-zone', + 'designatedashboard.resources.os-designate-floatingip' + ]); +})(); diff --git a/designatedashboard/static/designatedashboard/resources/util.service.js b/designatedashboard/static/designatedashboard/resources/util.service.js new file mode 100644 index 0000000..cfaa787 --- /dev/null +++ b/designatedashboard/static/designatedashboard/resources/util.service.js @@ -0,0 +1,107 @@ +/** + * (c) Copyright 2016 Hewlett Packard Enterprise Development LP + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +(function () { + 'use strict'; + + angular + .module('designatedashboard.resources') + .factory('designatedashboard.resources.util', utilService); + + utilService.$inject = [ + 'horizon.framework.util.q.extensions', + ]; + + + function utilService($qExtensions) { + var service = { + notDeleted: notDeleted, + notPending: notPending, + getModel: getModel, + actionMap: actionMap, + statusMap: statusMap, + addTimestampIds: addTimestampIds + }; + + return service; + + /////////////// + + function notDeleted(resource) { + return $qExtensions.booleanAsPromise(resource.status !== 'DELETED'); + } + + function notPending(resource) { + return $qExtensions.booleanAsPromise(resource.status !== 'PENDING'); + } + + /** + * Build a model object based on the given item, using only the fields present in the form config 'key's. + * Only 'truthy' values are copied. + * + * @param form - an array of objects describing the form. Must have a 'key' attribute. + * @param item - the data to copy into the model + */ + function getModel(form, item) { + var result = {}; + var value; + form.forEach(function iterateForm(formItem) { + value = item[formItem.key]; + if (value) { + result[formItem.key] = item[formItem.key]; + } + }); + return result; + } + + function actionMap() { + return { + 'none': gettext('None'), + 'create': gettext('Create') + } + } + + function statusMap() { + return { + 'active': gettext('Active'), + 'pending': gettext('Pending') + } + } + + /** + * hz-resource-table tracks by 'id' which doesn't change when an individual item is updated. + * Create a synthetic '_timestampId' using the item id plus the specified timestamp field. + * When this field is used as a track-by in hz-resource-table, items in the table to update + * after row actions. + * + * If a timestamp field is not specified, the current time is used. + * + * @param items {object} - The items to add a _timestampId. + * @param idField {string} - (Optional) A field on the item to use as the id. Defaults to 'id' + * @param timestampField {string} - (Optional) A field on item to use as a timestamp. Defaults + * to current time. + */ + function addTimestampIds(items, idField, timestampField) { + var _idField = idField || 'id'; + var timestamp = Date.now(); + items.map(function annotateFloatingIp(item) { + if ( angular.isDefined(timestampField) ) { + timestamp = item[timestampField]; + } + item._timestampId = item[_idField] + timestamp; + }); + } + } +}()); diff --git a/designatedashboard/static/designatedashboard/reverse_dns.html b/designatedashboard/static/designatedashboard/reverse_dns.html new file mode 100644 index 0000000..c739336 --- /dev/null +++ b/designatedashboard/static/designatedashboard/reverse_dns.html @@ -0,0 +1,4 @@ + + + diff --git a/designatedashboard/static/designatedashboard/zones.html b/designatedashboard/static/designatedashboard/zones.html new file mode 100644 index 0000000..cb977d0 --- /dev/null +++ b/designatedashboard/static/designatedashboard/zones.html @@ -0,0 +1,4 @@ + + + diff --git a/designatedashboard/tests/.secret_key_store b/designatedashboard/tests/.secret_key_store new file mode 100644 index 0000000..6e8db9b --- /dev/null +++ b/designatedashboard/tests/.secret_key_store @@ -0,0 +1 @@ +5NMb6ZfXYBLClFGFYf6VkbiJ9TRNyU3w8NQHPG8LXJltU8EZFeB7I632vQ8MF5m6 \ No newline at end of file diff --git a/designatedashboard/tests/__init__.py b/designatedashboard/tests/__init__.py new file mode 100644 index 0000000..e69de29 diff --git a/designatedashboard/tests/base.py b/designatedashboard/tests/base.py new file mode 100644 index 0000000..9062a3a --- /dev/null +++ b/designatedashboard/tests/base.py @@ -0,0 +1,100 @@ +# -*- coding: utf-8 -*- + +# Copyright 2010-2011 OpenStack Foundation +# Copyright (c) 2013 Hewlett-Packard Development Company, L.P. +# +# Licensed under the Apache License, Version 2.0 (the "License"); you may +# not use this file except in compliance with the License. You may obtain +# a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT +# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the +# License for the specific language governing permissions and limitations +# under the License. + +import os + +import fixtures +import testtools + +from openstack_dashboard.test import helpers as test + +from designatedashboard.dashboards.project.dns_domains import forms + + +_TRUE_VALUES = ('True', 'true', '1', 'yes') + + +class TestCase(testtools.TestCase): + + """Test case base class for all unit tests.""" + + def setUp(self): + """Run before each test method to initialize test environment.""" + + super(TestCase, self).setUp() + test_timeout = os.environ.get('OS_TEST_TIMEOUT', 0) + try: + test_timeout = int(test_timeout) + except ValueError: + # If timeout value is invalid do not set a timeout. + test_timeout = 0 + if test_timeout > 0: + self.useFixture(fixtures.Timeout(test_timeout, gentle=True)) + + self.useFixture(fixtures.NestedTempfile()) + self.useFixture(fixtures.TempHomeDir()) + + if os.environ.get('OS_STDOUT_CAPTURE') in _TRUE_VALUES: + stdout = self.useFixture(fixtures.StringStream('stdout')).stream + self.useFixture(fixtures.MonkeyPatch('sys.stdout', stdout)) + if os.environ.get('OS_STDERR_CAPTURE') in _TRUE_VALUES: + stderr = self.useFixture(fixtures.StringStream('stderr')).stream + self.useFixture(fixtures.MonkeyPatch('sys.stderr', stderr)) + + self.log_fixture = self.useFixture(fixtures.FakeLogger()) + + +class BaseRecordFormCleanTests(test.TestCase): + + DOMAIN_NAME = 'foo.com.' + HOSTNAME = 'www' + + MSG_FIELD_REQUIRED = 'This field is required' + MSG_INVALID_HOSTNAME = 'Enter a valid hostname. The '\ + 'hostname should contain letters '\ + 'and numbers, and be no more than '\ + '63 characters.' + MSG_INVALID_HOSTNAME_SHORT = 'Enter a valid hostname' + + def setUp(self): + super(BaseRecordFormCleanTests, self).setUp() + + # Request object with messages support + self.request = self.factory.get('', {}) + + # Set-up form instance + kwargs = {} + kwargs['initial'] = {'domain_name': self.DOMAIN_NAME} + self.form = forms.RecordCreate(self.request, **kwargs) + self.form._errors = {} + self.form.cleaned_data = { + 'domain_name': self.DOMAIN_NAME, + 'name': '', + 'data': '', + 'txt': '', + 'priority': None, + 'ttl': None, + } + + def assert_no_errors(self): + self.assertEqual(self.form._errors, {}) + + def assert_error(self, field, msg): + self.assertIn(msg, self.form._errors[field]) + + def assert_required_error(self, field): + self.assert_error(field, self.MSG_FIELD_REQUIRED) diff --git a/designatedashboard/tests/settings.py b/designatedashboard/tests/settings.py new file mode 100644 index 0000000..afc2b87 --- /dev/null +++ b/designatedashboard/tests/settings.py @@ -0,0 +1,83 @@ +# Copyright 2015 Hewlett-Packard Development Company, L.P. +# +# Licensed under the Apache License, Version 2.0 (the "License"); you may +# not use this file except in compliance with the License. You may obtain +# a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT +# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the +# License for the specific language governing permissions and limitations +# under the License. +import socket + +SECRET_KEY = 'HELLA_SECRET!' + +DATABASES = {'default': {'ENGINE': 'django.db.backends.sqlite3', + 'NAME': 'test'}} + +from horizon.test.settings import * # noqa + +socket.setdefaulttimeout(1) + +DEBUG = False +TEMPLATE_DEBUG = DEBUG + +TESTSERVER = 'http://testserver' + + +MESSAGE_STORAGE = 'django.contrib.messages.storage.cookie.CookieStorage' + +TEST_RUNNER = 'django_nose.NoseTestSuiteRunner' +NOSE_ARGS = ['--nocapture', + '--nologcapture', + '--cover-package=windc'] + +EMAIL_BACKEND = 'django.core.mail.backends.locmem.EmailBackend' +SESSION_ENGINE = 'django.contrib.sessions.backends.cache' + +OPENSTACK_ADDRESS = "localhost" +OPENSTACK_ADMIN_TOKEN = "openstack" +OPENSTACK_KEYSTONE_URL = "http://%s:5000/v2.0" % OPENSTACK_ADDRESS +OPENSTACK_KEYSTONE_ADMIN_URL = "http://%s:35357/v2.0" % OPENSTACK_ADDRESS +OPENSTACK_KEYSTONE_DEFAULT_ROLE = "Member" + +# Silence logging output during tests. +LOGGING = { + 'version': 1, + 'disable_existing_loggers': False, + 'handlers': { + 'null': { + 'level': 'DEBUG', + 'class': 'logging.NullHandler' + }, + }, + 'loggers': { + 'django.db.backends': { + 'handlers': ['null'], + 'propagate': False, + }, + 'horizon': { + 'handlers': ['null'], + 'propagate': False, + }, + 'novaclient': { + 'handlers': ['null'], + 'propagate': False, + }, + 'keystoneclient': { + 'handlers': ['null'], + 'propagate': False, + }, + 'quantum': { + 'handlers': ['null'], + 'propagate': False, + }, + 'nose.plugins.manager': { + 'handlers': ['null'], + 'propagate': False, + } + } +} diff --git a/designatedashboard/tests/test_designatedashboard.py b/designatedashboard/tests/test_designatedashboard.py new file mode 100644 index 0000000..12d2310 --- /dev/null +++ b/designatedashboard/tests/test_designatedashboard.py @@ -0,0 +1,406 @@ +# Copyright 2012 United States Government as represented by the +# Administrator of the National Aeronautics and Space Administration. +# All Rights Reserved. +# +# Copyright 2012 Nebula, Inc. +# +# Licensed under the Apache License, Version 2.0 (the "License"); you may +# not use this file except in compliance with the License. You may obtain +# a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT +# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the +# License for the specific language governing permissions and limitations +# under the License. +from __future__ import unicode_literals + +from django.core.urlresolvers import reverse # noqa +# from django import http + +from designatedashboard.tests import base + +DOMAIN_ID = '123' +# INDEX_URL = reverse('horizon:project:dns_domains:index') +# RECORDS_URL = reverse('horizon:project:dns_domains:records', +# args=[DOMAIN_ID]) + + +# class DNSDomainsTests(test.TestCase): + +# def setUp(self): +# super(DNSDomainsTests, self).setUp() + +# @test.create_stubs( +# {api.designate: ('domain_list',)}) +# def test_index(self): +# domains = self.dns_domains.list() +# api.designate.domain_list( +# IsA(http.HttpRequest)).AndReturn(domains) +# self.mox.ReplayAll() + +# res = self.client.get(INDEX_URL) + +# self.assertTemplateUsed(res, 'project/dns_domains/index.html') +# self.assertEqual(len(res.context['table'].data), len(domains)) + +# @test.create_stubs( +# {api.designate: ('domain_get', 'server_list', 'record_list')}) +# def test_records(self): +# domain_id = '123' +# domain = self.dns_domains.first() +# servers = self.dns_servers.list() +# records = self.dns_records.list() + +# api.designate.domain_get( +# IsA(http.HttpRequest), +# domain_id).AndReturn(domain) + +# api.designate.server_list( +# IsA(http.HttpRequest), +# domain_id).AndReturn(servers) + +# api.designate.record_list( +# IsA(http.HttpRequest), +# domain_id).AndReturn(records) + +# self.mox.ReplayAll() + +# res = self.client.get(RECORDS_URL) + +# self.assertTemplateUsed(res, 'project/dns_domains/records.html') +# self.assertEqual(len(res.context['table'].data), len(records)) + + +class ARecordFormTests(base.BaseRecordFormCleanTests): + + IPV4 = '1.1.1.1' + + MSG_INVALID_IPV4 = 'Enter a valid IPv4 address' + + def setUp(self): + super(ARecordFormTests, self).setUp() + self.form.cleaned_data['type'] = 'A' + self.form.cleaned_data['name'] = self.HOSTNAME + self.form.cleaned_data['data'] = self.IPV4 + + def test_valid_field_values(self): + self.form.clean() + self.assert_no_errors() + + def test_valid_name_field_wild_card(self): + self.form.cleaned_data['name'] = '*' + self.form.clean() + self.assert_no_errors() + + def test_missing_name_field(self): + self.form.cleaned_data['name'] = '' + self.form.clean() + self.assert_no_errors() + self.assertIsNotNone(self.form.cleaned_data['name']) + + def test_missing_data_field(self): + self.form.cleaned_data['data'] = '' + self.form.clean() + self.assert_required_error('data') + + def test_invalid_name_field(self): + self.form.cleaned_data['name'] = '$#%foo!!' + self.form.clean() + self.assert_error('name', self.MSG_INVALID_HOSTNAME) + + def test_invalid_name_field_starting_dash(self): + self.form.cleaned_data['name'] = '-ww' + self.form.clean() + self.assert_error('name', self.MSG_INVALID_HOSTNAME) + + def test_invalid_name_field_trailing_dash(self): + self.form.cleaned_data['name'] = 'co-' + self.form.clean() + self.assert_error('name', self.MSG_INVALID_HOSTNAME) + + def test_invalid_name_field_bad_wild_card(self): + self.form.cleaned_data['name'] = 'derp.*.' + self.DOMAIN_NAME + self.form.clean() + self.assert_error('name', self.MSG_INVALID_HOSTNAME) + + def test_outside_of_domain_name_field(self): + self.form.cleaned_data['name'] = 'www.bar.com.' + self.form.clean() + self.assert_error('name', self.MSG_INVALID_HOSTNAME) + + def test_invalid_data_field(self): + self.form.cleaned_data['data'] = 'foo' + self.form.clean() + self.assert_error('data', self.MSG_INVALID_IPV4) + + +class AAAARecordFormTests(base.BaseRecordFormCleanTests): + + IPV6 = '1111:1111:1111:11::1' + + MSG_INVALID_IPV6 = 'Enter a valid IPv6 address' + + def setUp(self): + super(AAAARecordFormTests, self).setUp() + self.form.cleaned_data['type'] = 'AAAA' + self.form.cleaned_data['name'] = self.HOSTNAME + self.form.cleaned_data['data'] = self.IPV6 + + def test_valid_field_values(self): + self.form.clean() + self.assert_no_errors() + + def test_valid_name_field_wild_card(self): + self.form.cleaned_data['name'] = '*' + self.form.clean() + self.assert_no_errors() + + def test_missing_name_field(self): + self.form.cleaned_data['name'] = '' + self.form.clean() + self.assert_no_errors() + self.assertIsNotNone(self.form.cleaned_data['name']) + + def test_missing_data_field(self): + self.form.cleaned_data['data'] = '' + self.form.clean() + self.assert_required_error('data') + + def test_invalid_name_field(self): + self.form.cleaned_data['name'] = '#@$foo!!' + self.form.clean() + self.assert_error('name', self.MSG_INVALID_HOSTNAME) + + def test_invalid_name_field_starting_dash(self): + self.form.cleaned_data['name'] = '-ww.foo.com' + self.form.clean() + self.assert_error('name', self.MSG_INVALID_HOSTNAME) + + def test_invalid_name_field_trailing_dash(self): + self.form.cleaned_data['name'] = 'www.foo.co-' + self.form.clean() + self.assert_error('name', self.MSG_INVALID_HOSTNAME) + + def test_invalid_name_field_bad_wild_card(self): + self.form.cleaned_data['name'] = 'derp.*.' + self.DOMAIN_NAME + self.form.clean() + self.assert_error('name', self.MSG_INVALID_HOSTNAME) + + def test_outside_of_domain_name_field(self): + self.form.cleaned_data['name'] = 'www.bar.com.' + self.form.clean() + self.assert_error('name', self.MSG_INVALID_HOSTNAME) + + def test_invalid_data_field(self): + self.form.cleaned_data['data'] = 'foo' + self.form.clean() + self.assert_error('data', self.MSG_INVALID_IPV6) + + +class CNAMERecordFormTests(base.BaseRecordFormCleanTests): + + CNAME = 'bar.foo.com.' + + def setUp(self): + super(CNAMERecordFormTests, self).setUp() + self.form.cleaned_data['type'] = 'CNAME' + self.form.cleaned_data['name'] = self.HOSTNAME + self.form.cleaned_data['data'] = self.CNAME + + def test_valid_field_values(self): + self.form.clean() + self.assert_no_errors() + + def test_valid_name_field_wild_card(self): + self.form.cleaned_data['name'] = '*' + self.form.clean() + self.assert_no_errors() + + def test_missing_name_field(self): + self.form.cleaned_data['name'] = '' + self.form.clean() + self.assert_required_error('name') + + def test_missing_data_field(self): + self.form.cleaned_data['data'] = '' + self.form.clean() + self.assert_required_error('data') + + def test_invalid_name_field(self): + self.form.cleaned_data['name'] = '$#%#$foo!!!' + self.form.clean() + self.assert_error('name', self.MSG_INVALID_HOSTNAME) + + def test_invalid_name_field_starting_dash(self): + self.form.cleaned_data['name'] = '-ww.foo.com' + self.form.clean() + self.assert_error('name', self.MSG_INVALID_HOSTNAME) + + def test_invalid_name_field_trailing_dash(self): + self.form.cleaned_data['name'] = 'www.foo.co-' + self.form.clean() + self.assert_error('name', self.MSG_INVALID_HOSTNAME) + + def test_invalid_name_field_bad_wild_card(self): + self.form.cleaned_data['name'] = 'derp.*.' + self.DOMAIN_NAME + self.form.clean() + self.assert_error('name', self.MSG_INVALID_HOSTNAME) + + def test_outside_of_domain_name_field(self): + self.form.cleaned_data['name'] = 'www.bar.com.' + self.form.clean() + self.assert_error('name', self.MSG_INVALID_HOSTNAME) + + def test_invalid_data_field(self): + self.form.cleaned_data['data'] = 'foo' + self.form.clean() + self.assert_error('data', self.MSG_INVALID_HOSTNAME_SHORT) + + +class MXRecordFormTests(base.BaseRecordFormCleanTests): + + MAIL_SERVER = 'mail.foo.com.' + PRIORITY = 10 + + def setUp(self): + super(MXRecordFormTests, self).setUp() + self.form.cleaned_data['type'] = 'MX' + self.form.cleaned_data['data'] = self.MAIL_SERVER + self.form.cleaned_data['priority'] = self.PRIORITY + + def test_valid_field_values(self): + self.form.clean() + self.assert_no_errors() + + def test_missing_data_field(self): + self.form.cleaned_data['data'] = '' + self.form.clean() + self.assert_required_error('data') + + def test_missing_priority_field(self): + self.form.cleaned_data['priority'] = None + self.form.clean() + self.assert_required_error('priority') + + def test_invalid_data_field(self): + self.form.cleaned_data['data'] = 'foo' + self.form.clean() + self.assert_error('data', self.MSG_INVALID_HOSTNAME_SHORT) + + def test_default_assignment_name_field(self): + self.form.clean() + self.assertEqual(self.DOMAIN_NAME, self.form.cleaned_data['name']) + + +class TXTRecordFormTests(base.BaseRecordFormCleanTests): + + TEXT = 'Lorem ipsum' + + def setUp(self): + super(TXTRecordFormTests, self).setUp() + self.form.cleaned_data['type'] = 'TXT' + self.form.cleaned_data['name'] = self.HOSTNAME + self.form.cleaned_data['txt'] = self.TEXT + + def test_valid_field_values(self): + self.form.clean() + self.assert_no_errors() + + def test_valid_name_field_wild_card(self): + self.form.cleaned_data['name'] = '*' + self.form.clean() + self.assert_no_errors() + + def test_missing_name_field(self): + self.form.cleaned_data['name'] = '' + self.form.clean() + self.assert_no_errors() + self.assertIsNotNone(self.form.cleaned_data['name']) + + def test_missing_txt_field(self): + self.form.cleaned_data['txt'] = '' + self.form.clean() + self.assert_required_error('txt') + + def test_invalid_name_field(self): + self.form.cleaned_data['name'] = 'foo-' + self.form.clean() + self.assert_error('name', self.MSG_INVALID_HOSTNAME) + + def test_invalid_name_field_starting_dash(self): + self.form.cleaned_data['name'] = '-ww.foo.com' + self.form.clean() + self.assert_error('name', self.MSG_INVALID_HOSTNAME) + + def test_invalid_name_field_trailing_dash(self): + self.form.cleaned_data['name'] = 'www.foo.co-' + self.form.clean() + self.assert_error('name', self.MSG_INVALID_HOSTNAME) + + def test_invalid_name_field_bad_wild_card(self): + self.form.cleaned_data['name'] = 'derp.*' + self.form.clean() + self.assert_error('name', self.MSG_INVALID_HOSTNAME) + + def test_outside_of_domain_name_field(self): + self.form.cleaned_data['name'] = 'www.bar.com.' + self.form.clean() + self.assert_error('name', self.MSG_INVALID_HOSTNAME) + + def test_default_assignment_data_field(self): + self.form.clean() + self.assertEqual(self.TEXT, self.form.cleaned_data['data']) + + +class SRVRecordFormTests(base.BaseRecordFormCleanTests): + + SRV_NAME = '_foo._tcp.' + SRV_DATA = '1 1 srv.foo.com.' + PRIORITY = 10 + + MSG_INVALID_SRV_NAME = 'Enter a valid SRV name' + MSG_INVALID_SRV_DATA = 'Enter a valid SRV record' + + def setUp(self): + super(SRVRecordFormTests, self).setUp() + self.form.cleaned_data['type'] = 'SRV' + self.form.cleaned_data['name'] = self.SRV_NAME + self.form.cleaned_data['data'] = self.SRV_DATA + self.form.cleaned_data['priority'] = self.PRIORITY + + def test_valid_field_values(self): + self.form.clean() + self.assert_no_errors() + + def test_missing_name_field(self): + self.form.cleaned_data['name'] = '' + self.form.clean() + self.assert_required_error('name') + + def test_missing_data_field(self): + self.form.cleaned_data['data'] = '' + self.form.clean() + self.assert_required_error('data') + + def test_missing_priority_field(self): + self.form.cleaned_data['priority'] = None + self.form.clean() + self.assert_required_error('priority') + + def test_invalid_name_field(self): + self.form.cleaned_data['name'] = 'foo' + self.form.clean() + self.assert_error('name', self.MSG_INVALID_SRV_NAME) + + def test_invalid_data_field(self): + self.form.cleaned_data['data'] = 'foo' + self.form.clean() + self.assert_error('data', self.MSG_INVALID_SRV_DATA) + + def test_default_assignment_name_field(self): + self.form.clean() + self.assertEqual(self.SRV_NAME + self.DOMAIN_NAME, + self.form.cleaned_data['name']) diff --git a/designatedashboard/tests/test_ptr_record_form.py b/designatedashboard/tests/test_ptr_record_form.py new file mode 100644 index 0000000..feafc98 --- /dev/null +++ b/designatedashboard/tests/test_ptr_record_form.py @@ -0,0 +1,85 @@ +# Copyright 2015 NEC Corporation. All rights reserved. +# +# Licensed under the Apache License, Version 2.0 (the "License"); you may +# not use this file except in compliance with the License. You may obtain +# a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT +# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the +# License for the specific language governing permissions and limitations +# under the License. + +from designatedashboard.tests import base + + +class PTRRecordFormTests(base.BaseRecordFormCleanTests): + + PTR = "6.0.0.10.in-addr.arpa." + + def setUp(self): + super(PTRRecordFormTests, self).setUp() + self.form.cleaned_data['type'] = 'PTR' + self.form.cleaned_data['name'] = self.HOSTNAME + self.form.cleaned_data['data'] = self.PTR + + def test_valid_field_values(self): + self.form.clean() + self.assert_no_errors() + + def test_valid_name_field_wild_card(self): + self.form.cleaned_data['name'] = '*' + self.form.clean() + self.assert_no_errors() + + def test_missing_name_field(self): + self.form.cleaned_data['name'] = '' + self.form.clean() + self.assert_no_errors() + self.assertIsNotNone(self.form.cleaned_data['name']) + + def test_invalid_name_field(self): + self.form.cleaned_data['name'] = '#@$foo!!' + self.form.clean() + self.assert_error('name', self.MSG_INVALID_HOSTNAME) + + def test_invalid_name_field_starting_dash(self): + self.form.cleaned_data['name'] = '-ww.foo.com' + self.form.clean() + self.assert_error('name', self.MSG_INVALID_HOSTNAME) + + def test_invalid_name_field_trailing_dash(self): + self.form.cleaned_data['name'] = 'www.foo.co-' + self.form.clean() + self.assert_error('name', self.MSG_INVALID_HOSTNAME) + + def test_invalid_name_field_bad_wild_card(self): + self.form.cleaned_data['name'] = 'derp.*.' + self.DOMAIN_NAME + self.form.clean() + self.assert_error('name', self.MSG_INVALID_HOSTNAME) + + def test_outside_of_domain_name_field(self): + self.form.cleaned_data['name'] = 'www.bar.com.' + self.form.clean() + + def test_invalid_data_field(self): + self.form.cleaned_data['data'] = '#@$' + self.PTR + '!!' + self.form.clean() + self.assert_error('data', self.MSG_INVALID_HOSTNAME_SHORT) + + def test_invalid_data_field_starting_dash(self): + self.form.cleaned_data['data'] = '-' + self.PTR + self.form.clean() + self.assert_error('data', self.MSG_INVALID_HOSTNAME_SHORT) + + def test_invalid_data_field_trailing_dash(self): + self.form.cleaned_data['data'] = self.PTR + '-' + self.form.clean() + self.assert_error('data', self.MSG_INVALID_HOSTNAME_SHORT) + + def test_invalid_data_field_bad_wild_card(self): + self.form.cleaned_data['data'] = 'derp.*.' + self.PTR + self.form.clean() + self.assert_error('data', self.MSG_INVALID_HOSTNAME_SHORT) diff --git a/designatedashboard/tests/test_spf_record_form.py b/designatedashboard/tests/test_spf_record_form.py new file mode 100644 index 0000000..7885c3f --- /dev/null +++ b/designatedashboard/tests/test_spf_record_form.py @@ -0,0 +1,70 @@ +# Copyright 2015 NEC Corporation. All rights reserved. +# +# Licensed under the Apache License, Version 2.0 (the "License"); you may +# not use this file except in compliance with the License. You may obtain +# a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT +# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the +# License for the specific language governing permissions and limitations +# under the License. + +from designatedashboard.tests import base + + +class SPFRecordFormTests(base.BaseRecordFormCleanTests): + + TEXT = 'v=spf1 +all' + + def setUp(self): + super(SPFRecordFormTests, self).setUp() + self.form.cleaned_data['type'] = 'SPF' + self.form.cleaned_data['name'] = self.HOSTNAME + self.form.cleaned_data['txt'] = self.TEXT + + def test_valid_field_values(self): + self.form.clean() + self.assert_no_errors() + + def test_valid_name_field_wild_card(self): + self.form.cleaned_data['name'] = '*' + self.form.clean() + self.assert_no_errors() + + def test_missing_name_field(self): + self.form.cleaned_data['name'] = '' + self.form.clean() + self.assert_no_errors() + + def test_missing_txt_field(self): + self.form.cleaned_data['txt'] = '' + self.form.clean() + self.assert_no_errors() + + def test_invalid_name_field(self): + self.form.cleaned_data['name'] = 'foo-' + self.form.clean() + self.assert_error('name', self.MSG_INVALID_HOSTNAME) + + def test_invalid_name_field_starting_dash(self): + self.form.cleaned_data['name'] = '-ww.foo.com' + self.form.clean() + self.assert_error('name', self.MSG_INVALID_HOSTNAME) + + def test_invalid_name_field_trailing_dash(self): + self.form.cleaned_data['name'] = 'www.foo.co-' + self.form.clean() + self.assert_error('name', self.MSG_INVALID_HOSTNAME) + + def test_invalid_name_field_bad_wild_card(self): + self.form.cleaned_data['name'] = 'derp.*' + self.form.clean() + self.assert_error('name', self.MSG_INVALID_HOSTNAME) + + def test_outside_of_domain_name_field(self): + self.form.cleaned_data['name'] = 'www.bar.com.' + self.form.clean() + self.assert_error('name', self.MSG_INVALID_HOSTNAME) diff --git a/designatedashboard/tests/test_sshfp_record_form.py b/designatedashboard/tests/test_sshfp_record_form.py new file mode 100644 index 0000000..32d7799 --- /dev/null +++ b/designatedashboard/tests/test_sshfp_record_form.py @@ -0,0 +1,92 @@ +# Copyright 2015 NEC Corporation. All rights reserved. +# +# Licensed under the Apache License, Version 2.0 (the "License"); you may +# not use this file except in compliance with the License. You may obtain +# a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT +# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the +# License for the specific language governing permissions and limitations +# under the License. + +from designatedashboard.tests import base + + +class SSHFPRecordFormTests(base.BaseRecordFormCleanTests): + + TEXT = '2 1 d1eb0d876ec69d18bcefc4263ae43ec33ae14f4c' + MSG_INVALID_RECORD = "Enter a valid SSHFP record" + + def setUp(self): + super(SSHFPRecordFormTests, self).setUp() + self.form.cleaned_data['type'] = 'SSHFP' + self.form.cleaned_data['name'] = self.HOSTNAME + self.form.cleaned_data['txt'] = self.TEXT + + def test_valid_field_values(self): + self.form.clean() + self.assert_no_errors() + + def test_valid_name_field_wild_card(self): + self.form.cleaned_data['name'] = '*' + self.form.clean() + self.assert_no_errors() + + def test_missing_txt_field(self): + self.form.cleaned_data['txt'] = '' + self.form.clean() + self.assert_error('txt', self.MSG_INVALID_RECORD) + + def test_invalid_txt_field(self): + self.form.cleaned_data['txt'] = 'foo' + self.form.clean() + self.assert_error('txt', self.MSG_INVALID_RECORD) + + def test_invalid_text_field_starting_dash(self): + self.form.cleaned_data['txt'] = '-2 1 d1eb0d876ec69d18bcef\ + c4263ae43ec33ae14f4c' + self.form.clean() + self.assert_error('txt', self.MSG_INVALID_RECORD) + + def test_invalid_text_field_trailing_dash(self): + self.form.cleaned_data['txt'] = '2 1 d1eb0d876ec69d18bcef\ + c4263ae43ec33ae14f4c-' + self.form.clean() + self.assert_error('txt', self.MSG_INVALID_RECORD) + + def test_invalid_text_field_bad_wild_card(self): + self.form.cleaned_data['txt'] = '1 2 e0d5320e7e36dea8e369b*' + self.form.clean() + self.assert_error('txt', self.MSG_INVALID_RECORD) + + def test_default_assignment_data_field(self): + self.form.clean() + self.assertEqual(self.TEXT, self.form.cleaned_data['data']) + + def test_invalid_name_field(self): + self.form.cleaned_data['name'] = 'foo-' + self.form.clean() + self.assert_error('name', self.MSG_INVALID_HOSTNAME) + + def test_invalid_name_field_starting_dash(self): + self.form.cleaned_data['name'] = '-ww.foo.com' + self.form.clean() + self.assert_error('name', self.MSG_INVALID_HOSTNAME) + + def test_invalid_name_field_trailing_dash(self): + self.form.cleaned_data['name'] = 'www.foo.co-' + self.form.clean() + self.assert_error('name', self.MSG_INVALID_HOSTNAME) + + def test_invalid_name_field_bad_wild_card(self): + self.form.cleaned_data['name'] = 'derp.*' + self.form.clean() + self.assert_error('name', self.MSG_INVALID_HOSTNAME) + + def test_outside_of_domain_name_field(self): + self.form.cleaned_data['name'] = 'www.bar.com.' + self.form.clean() + self.assert_error('name', self.MSG_INVALID_HOSTNAME) diff --git a/doc/source/conf.py b/doc/source/conf.py new file mode 100755 index 0000000..67602e2 --- /dev/null +++ b/doc/source/conf.py @@ -0,0 +1,82 @@ +# -*- coding: utf-8 -*- +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless reqdashboardred by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or +# implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +import os +import sys + +sys.path.insert(0, os.path.abspath('../..')) +# -- General configuration ---------------------------------------------------- + +# Add any Sphinx extension module names here, as strings. They can be +# extensions coming with Sphinx (named 'sphinx.ext.*') or your custom ones. +extensions = [ + 'sphinx.ext.autodoc', + #'sphinx.ext.intersphinx', + 'openstackdocstheme' +] + +# openstackdocstheme options +repository_name = 'openstack/designate-dashboard' +bug_project = 'designate-dashboard' +bug_tag = '' +html_last_updated_fmt = '%Y-%m-%d %H:%M' +html_theme = 'openstackdocs' + +# autodoc generation is a bit aggressive and a ndashboardsance when doing heavy +# text edit cycles. +# execute "export SPHINX_DEBUG=1" in your terminal to disable + +# The suffix of source filenames. +source_suffix = '.rst' + +# The master toctree document. +master_doc = 'index' + +# General information about the project. +project = u'designate_dashboard' +copyright = u'2013, OpenStack Foundation' + +# If true, '()' will be appended to :func: etc. cross-reference text. +add_function_parentheses = True + +# If true, the current module name will be prepended to all description +# unit titles (such as .. function::). +add_module_names = True + +# The name of the Pygments (syntax highlighting) style to use. +pygments_style = 'sphinx' + +# -- Options for HTML output -------------------------------------------------- + +# The theme to use for HTML and HTML Help pages. Major themes that come with +# Sphinx are currently 'default' and 'sphinxdoc'. +# html_theme_path = ["."] +# html_theme = '_theme' +# html_static_path = ['static'] + +# Output file base name for HTML help bdashboardlder. +htmlhelp_basename = '%sdoc' % project + +# Grouping the document tree into LaTeX files. List of tuples +# (source start file, target name, title, author, documentclass +# [howto/manual]). +latex_documents = [ + ('index', + '%s.tex' % project, + u'%s Documentation' % project, + u'OpenStack Foundation', 'manual'), +] + +# Example configuration for intersphinx: refer to the Python standard library. +#intersphinx_mapping = {'http://docs.python.org/': None} diff --git a/doc/source/contributor/index.rst b/doc/source/contributor/index.rst new file mode 100644 index 0000000..3d4ceb4 --- /dev/null +++ b/doc/source/contributor/index.rst @@ -0,0 +1,5 @@ +============ +Contributing +============ + +.. include:: ../../../CONTRIBUTING.rst diff --git a/doc/source/index.rst b/doc/source/index.rst new file mode 100644 index 0000000..2bced7e --- /dev/null +++ b/doc/source/index.rst @@ -0,0 +1,21 @@ +.. designate_ui documentation master file, created by + sphinx-quickstart on Tue Jul 9 22:26:36 2013. + You can adapt this file completely to your liking, but it should at least + contain the root `toctree` directive. + +================================================ + Welcome to designatedashboard's documentation! +================================================ + +.. toctree:: + :maxdepth: 2 + + readme + install/index + user/index + contributor/index + +.. rubric:: Indices and tables + +* :ref:`genindex` +* :ref:`search` diff --git a/doc/source/install/index.rst b/doc/source/install/index.rst new file mode 100644 index 0000000..af26cf0 --- /dev/null +++ b/doc/source/install/index.rst @@ -0,0 +1,12 @@ +============ +Installation +============ + +At the command line:: + + $ pip install designate_dashboard + +Or, if you have virtualenvwrapper installed:: + + $ mkvirtualenv designate_dashboard + $ pip install designate_dashboard \ No newline at end of file diff --git a/doc/source/readme.rst b/doc/source/readme.rst new file mode 100644 index 0000000..38ba804 --- /dev/null +++ b/doc/source/readme.rst @@ -0,0 +1 @@ +.. include:: ../../README.rst \ No newline at end of file diff --git a/doc/source/user/index.rst b/doc/source/user/index.rst new file mode 100644 index 0000000..8ca3de4 --- /dev/null +++ b/doc/source/user/index.rst @@ -0,0 +1,7 @@ +======== +Usage +======== + +To use designate_dashboard in a project:: + + import designate_dashboard \ No newline at end of file diff --git a/karma.conf.js b/karma.conf.js new file mode 100644 index 0000000..f437037 --- /dev/null +++ b/karma.conf.js @@ -0,0 +1,141 @@ +/* + * + * Licensed under the Apache License, Version 2.0 (the 'License'); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an 'AS IS' BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +'use strict'; + +var fs = require('fs'); +var path = require('path'); + +module.exports = function (config) { + // This tox venv is setup in the post-install npm step + var toxPath = '.tox/py27/lib/python2.7/site-packages/'; + var xstaticPath = toxPath + 'xstatic/pkg/'; + + config.set({ + preprocessors: { + // Used to collect templates for preprocessing. + // NOTE: the templates must also be listed in the files section below. + './static/**/*.html': ['ng-html2js'] + }, + + // Sets up module to process templates. + ngHtml2JsPreprocessor: { + prependPrefix: '/', + moduleName: 'templates' + }, + + basePath: './', + + // Contains both source and test files. + files: [ + /* + * shim, partly stolen from /i18n/js/horizon/ + * Contains expected items not provided elsewhere (dynamically by + * Django or via jasmine template. + */ + './test-shim.js', + + // from jasmine.html + xstaticPath + 'jquery/data/jquery.js', + xstaticPath + 'angular/data/angular.js', + xstaticPath + 'angular/data/angular-route.js', + xstaticPath + 'angular/data/angular-mocks.js', + xstaticPath + 'angular/data/angular-cookies.js', + xstaticPath + 'angular_bootstrap/data/angular-bootstrap.js', + xstaticPath + 'angular_gettext/data/angular-gettext.js', + xstaticPath + 'angular_fileupload/data/ng-file-upload-all.js', + xstaticPath + 'angular/data/angular-sanitize.js', + xstaticPath + 'd3/data/d3.js', + xstaticPath + 'rickshaw/data/rickshaw.js', + xstaticPath + 'angular_smart_table/data/smart-table.js', + xstaticPath + 'angular_lrdragndrop/data/lrdragndrop.js', + xstaticPath + 'spin/data/spin.js', + xstaticPath + 'spin/data/spin.jquery.js', + xstaticPath + 'tv4/data/tv4.js', + xstaticPath + 'objectpath/data/ObjectPath.js', + xstaticPath + 'angular_schema_form/data/schema-form.js', + + // TODO: These should be mocked. + toxPath + '/horizon/static/horizon/js/horizon.js', + + /** + * Include framework source code from horizon that we need. + * Otherwise, karma will not be able to find them when testing. + * These files should be mocked in the foreseeable future. + */ + toxPath + 'horizon/static/framework/**/*.module.js', + toxPath + 'horizon/static/framework/**/!(*.spec|*.mock).js', + toxPath + 'openstack_dashboard/static/**/*.module.js', + toxPath + 'openstack_dashboard/static/**/!(*.spec|*.mock).js', + toxPath + 'openstack_dashboard/dashboards/**/static/**/*.module.js', + toxPath + 'openstack_dashboard/dashboards/**/static/**/!(*.spec|*.mock).js', + + /** + * First, list all the files that defines application's angular modules. + * Those files have extension of `.module.js`. The order among them is + * not significant. + */ + './designatedashboard/static/**/*.module.js', + + /** + * Followed by other JavaScript files that defines angular providers + * on the modules defined in files listed above. And they are not mock + * files or spec files defined below. The order among them is not + * significant. + */ + './designatedashboard/static/**/!(*.spec|*.mock).js', + + /** + * Then, list files for mocks with `mock.js` extension. The order + * among them should not be significant. + */ + toxPath + 'openstack_dashboard/static/**/*.mock.js', + //'./static/**/*.mock.js', + + /** + * Finally, list files for spec with `spec.js` extension. The order + * among them should not be significant. + */ + './designatedashboard/static/**/*.spec.js', + + /** + * Angular external templates + */ + './designatedashboard/static/**/*.html', + + ], + + autoWatch: true, + + frameworks: ['jasmine'], + + browsers: ['Chrome'], + + phantomjsLauncher: { + // Have phantomjs exit if a ResourceError is encountered + // (useful if karma exits without killing phantom) + exitOnResourceError: true + }, + + reporters: ['progress'], + + plugins: [ + 'karma-chrome-launcher', + 'karma-jasmine', + 'karma-ng-html2js-preprocessor' + ] + + }); +}; diff --git a/manage.py b/manage.py new file mode 100755 index 0000000..3cb4f27 --- /dev/null +++ b/manage.py @@ -0,0 +1,23 @@ +#!/usr/bin/env python +# Copyright 2014 Hewlett-Packard Development Company, L.P. +# +# Licensed under the Apache License, Version 2.0 (the "License"); you may +# not use this file except in compliance with the License. You may obtain +# a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT +# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the +# License for the specific language governing permissions and limitations +# under the License. +import os +import sys + + +if __name__ == "__main__": + os.environ.setdefault( + "DJANGO_SETTINGS_MODULE", "designatedashboard.settings") + from django.core.management import execute_from_command_line # noqa + execute_from_command_line(sys.argv) diff --git a/openstack-common.conf b/openstack-common.conf new file mode 100644 index 0000000..1d8dcca --- /dev/null +++ b/openstack-common.conf @@ -0,0 +1,6 @@ +[DEFAULT] + +# The list of modules to copy from oslo-incubator.git + +# The base module to hold the copy of openstack.common +base=designatedashboard diff --git a/package.json b/package.json new file mode 100644 index 0000000..6ee7760 --- /dev/null +++ b/package.json @@ -0,0 +1,26 @@ +{ + "version": "0.0.0", + "private": true, + "name": "designate-dashboard", + "description": "Designate Dashboard", + "repository": "none", + "license": "Apache 2.0", + "devDependencies": { + "eslint": "1.10.3", + "eslint-config-openstack": "1.2.4", + "jasmine-core": "2.4.1", + "karma": "1.1.2", + "karma-chrome-launcher": "1.0.1", + "karma-cli": "1.0.1", + "karma-jasmine": "1.0.2", + "karma-ng-html2js-preprocessor": "1.0.0" + }, + "scripts": { + "postinstall": "if [ ! -d .venv ]; then tox -epy27 --notest; fi", + "lint": "eslint --no-color designatedashboard/static", + "lintq": "eslint --quiet designatedashboard/static", + "test": "karma start karma.conf.js --single-run" + }, + "dependencies": {} +} + diff --git a/releasenotes/notes/.placeholder b/releasenotes/notes/.placeholder new file mode 100644 index 0000000..e69de29 diff --git a/releasenotes/source/conf.py b/releasenotes/source/conf.py new file mode 100644 index 0000000..fd620eb --- /dev/null +++ b/releasenotes/source/conf.py @@ -0,0 +1,281 @@ +# -*- coding: utf-8 -*- +# +# Designate dashboard Release Notes documentation build configuration file. +# +# This file is execfile()d with the current directory set to its +# containing dir. +# +# Note that not all possible configuration values are present in this +# autogenerated file. +# +# All configuration values have a default; values that are commented out +# serve to show the default. + +# If extensions (or modules to document with autodoc) are in another directory, +# add these directories to sys.path here. If the directory is relative to the +# documentation root, use os.path.abspath to make it absolute, like shown here. +# sys.path.insert(0, os.path.abspath('.')) + +# -- General configuration ------------------------------------------------ + +# If your documentation needs a minimal Sphinx version, state it here. +# needs_sphinx = '1.0' + +# Add any Sphinx extension module names here, as strings. They can be +# extensions coming with Sphinx (named 'sphinx.ext.*') or your custom +# ones. + +extensions = [ + 'openstackdocstheme', + 'reno.sphinxext', +] + +# openstackdocstheme options +repository_name = 'openstack/designate-dashboard' +bug_project = 'designate-dashboard' +bug_tag = '' +html_last_updated_fmt = '%Y-%m-%d %H:%M' +html_theme = 'openstackdocs' + +# Add any paths that contain templates here, relative to this directory. + +templates_path = ['_templates'] + +# The suffix of source filenames. + +source_suffix = '.rst' + +# The encoding of source files. +# source_encoding = 'utf-8-sig' + +# The master toctree document. + +master_doc = 'index' + +# General information about the project. + +project = u'Designate dashboard Client Release Notes' +copyright = u'2016, Designate dashboard developers' + +# Release notes are version independent +# The short X.Y version. +version = '' +# The full version, including alpha/beta/rc tags. +release = '' + +# The language for content autogenerated by Sphinx. Refer to documentation +# for a list of supported languages. +# language = None + +# There are two options for replacing |today|: either, you set today to some +# non-false value, then it is used: +# today = '' +# Else, today_fmt is used as the format for a strftime call. +# today_fmt = '%B %d, %Y' + +# List of patterns, relative to source directory, that match files and +# directories to ignore when looking for source files. + +exclude_patterns = [] + +# The reST default role (used for this markup: `text`) to use for all +# documents. +# default_role = None + +# If true, '()' will be appended to :func: etc. cross-reference text. +# add_function_parentheses = True + +# If true, the current module name will be prepended to all description +# unit titles (such as .. function::). +# add_module_names = True + +# If true, sectionauthor and moduleauthor directives will be shown in the +# output. They are ignored by default. +# show_authors = False + +# The name of the Pygments (syntax highlighting) style to use. + +pygments_style = 'sphinx' + +# A list of ignored prefixes for module index sorting. +# modindex_common_prefix = [] + +# If true, keep warnings as "system message" paragraphs in the built documents. +# keep_warnings = False + + +# -- Options for HTML output ---------------------------------------------- + +# The theme to use for HTML and HTML Help pages. See the documentation for +# a list of builtin themes. + +# html_theme = 'default' + +# Theme options are theme-specific and customize the look and feel of a theme +# further. For a list of options available for each theme, see the +# documentation. +# html_theme_options = {} + +# Add any paths that contain custom themes here, relative to this directory. +# html_theme_path = [] + +# The name for this set of Sphinx documents. If None, it defaults to +# " v documentation". +# html_title = None + +# A shorter title for the navigation bar. Default is the same as html_title. +# html_short_title = None + +# The name of an image file (relative to this directory) to place at the top +# of the sidebar. +# html_logo = None + +# The name of an image file (within the static path) to use as favicon of the +# docs. This file should be a Windows icon file (.ico) being 16x16 or 32x32 +# pixels large. +# html_favicon = None + +# Add any paths that contain custom static files (such as style sheets) here, +# relative to this directory. They are copied after the builtin static files, +# so a file named "default.css" will overwrite the builtin "default.css". + +html_static_path = ['_static'] + +# Add any extra paths that contain custom files (such as robots.txt or +# .htaccess) here, relative to this directory. These files are copied +# directly to the root of the documentation. +# html_extra_path = [] + +# If not '', a 'Last updated on:' timestamp is inserted at every page bottom, +# using the given strftime format. +# html_last_updated_fmt = '%b %d, %Y' + +# If true, SmartyPants will be used to convert quotes and dashes to +# typographically correct entities. +# html_use_smartypants = True + +# Custom sidebar templates, maps document names to template names. +# html_sidebars = {} + +# Additional templates that should be rendered to pages, maps page names to +# template names. +# html_additional_pages = {} + +# If false, no module index is generated. +# html_domain_indices = True + +# If false, no index is generated. +# html_use_index = True + +# If true, the index is split into individual pages for each letter. +# html_split_index = False + +# If true, links to the reST sources are added to the pages. +# tml_show_sourcelink = True + +# If true, "Created using Sphinx" is shown in the HTML footer. Default is True. +# tml_show_sphinx = True + +# If true, "(C) Copyright ..." is shown in the HTML footer. Default is True. +# tml_show_copyright = True + +# If true, an OpenSearch description file will be output, and all pages will +# contain a tag referring to it. The value of this option must be the +# base URL from which the finished HTML is served. +# tml_use_opensearch = '' + +# This is the file name suffix for HTML files (e.g. ".xhtml"). +# tml_file_suffix = None + +# Output file base name for HTML help builder. + +htmlhelp_basename = 'DesignatedashboardReleaseNotestdoc' + + +# -- Options for LaTeX output --------------------------------------------- + +latex_elements = { + + # The paper size ('letterpaper' or 'a4paper'). + # 'papersize': 'letterpaper', + + # The font size ('10pt', '11pt' or '12pt'). + # 'pointsize': '10pt', + + # Additional stuff for the LaTeX preamble. + # 'preamble': '', +} + +# Grouping the document tree into LaTeX files. List of tuples +# (source start file, target name, title, +# author, documentclass [howto, manual, or own class]). + +latex_documents = [ + ('index', 'PythonDesignatedashboard.tex', + u'Designate dashboard Release Notes Documentation', + u'Designate dashboard developers', 'manual'), +] + +# The name of an image file (relative to this directory) to place at the top of +# the title page. +# atex_logo = None + +# For "manual" documents, if this is true, then toplevel headings are parts, +# not chapters. +# atex_use_parts = False + +# If true, show page references after internal links. +# atex_show_pagerefs = False + +# If true, show URL addresses after external links. +# atex_show_urls = False + +# Documents to append as an appendix to all manuals. +# atex_appendices = [] + +# If false, no module index is generated. +# atex_domain_indices = True + + +# -- Options for manual page output --------------------------------------- + +# One entry per manual page. List of tuples +# (source start file, name, description, authors, manual section). + +man_pages = [ + ('index', 'designatedashboard', + u'Designate dashboard Release Notes Documentation', + [u'Designate dashboard developers'], 1) +] + +# If true, show URL addresses after external links. +# an_show_urls = False + + +# -- Options for Texinfo output ------------------------------------------- + +# Grouping the document tree into Texinfo files. List of tuples +# (source start file, target name, title, author, +# dir menu entry, description, category) + +texinfo_documents = [ + ('index', 'Designate dashboard', + u'Designate dashboard Release Notes Documentation', + u'Designate dashboard developers', 'Designate dashboard', + 'One line description of project.', 'Miscellaneous'), +] + +# Documents to append as an appendix to all manuals. +# exinfo_appendices = [] + +# If false, no module index is generated. +# exinfo_domain_indices = True + +# How to display URL addresses: 'footnote', 'no', or 'inline'. +# exinfo_show_urls = 'footnote' + +# If true, do not generate a @detailmenu in the "Top" node's menu. +# exinfo_no_detailmenu = False + +# -- Options for Internationalization output ------------------------------ +locale_dirs = ['locale/'] diff --git a/releasenotes/source/index.rst b/releasenotes/source/index.rst new file mode 100644 index 0000000..5b98ad9 --- /dev/null +++ b/releasenotes/source/index.rst @@ -0,0 +1,20 @@ +Welcome to Designate dashboard Agent Release Notes documentation! +================================================================= + +Contents +======== + +.. toctree:: + :maxdepth: 2 + + unreleased + pike + ocata + newton + mitaka + +Indices and tables +================== + +* :ref:`genindex` +* :ref:`search` diff --git a/releasenotes/source/locale/cs/LC_MESSAGES/releasenotes.po b/releasenotes/source/locale/cs/LC_MESSAGES/releasenotes.po new file mode 100644 index 0000000..7144149 --- /dev/null +++ b/releasenotes/source/locale/cs/LC_MESSAGES/releasenotes.po @@ -0,0 +1,45 @@ +# Zbyněk Schwarz , 2017. #zanata +msgid "" +msgstr "" +"Project-Id-Version: Designate dashboard Client Release Notes\n" +"Report-Msgid-Bugs-To: \n" +"POT-Creation-Date: 2017-11-21 15:00+0000\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"PO-Revision-Date: 2017-11-17 09:17+0000\n" +"Last-Translator: Zbyněk Schwarz \n" +"Language-Team: Czech\n" +"Language: cs\n" +"X-Generator: Zanata 3.9.6\n" +"Plural-Forms: nplurals=3; plural=(n==1) ? 0 : (n>=2 && n<=4) ? 1 : 2\n" + +msgid ":ref:`genindex`" +msgstr ":ref:`genindex`" + +msgid ":ref:`search`" +msgstr ":ref:`search`" + +msgid "Contents" +msgstr "Obsah" + +msgid "Current Series Release Notes" +msgstr "Poznámky k vydání současné verze" + +msgid "Indices and tables" +msgstr "Rejstříky a tabulky" + +msgid "Mitaka Series Release Notes" +msgstr "Poznámky k vydání verze Mitaka" + +msgid "Newton Series Release Notes" +msgstr "Poznámky k vydání verze Newton" + +msgid "Ocata Series Release Notes" +msgstr "Poznámky k vydání verze Ocata" + +msgid "Pike Series Release Notes" +msgstr "Poznámky k vydání verze Pike" + +msgid "Welcome to Designate dashboard Agent Release Notes documentation!" +msgstr "Vítejte v dokumentaci poznámek k vydání agenta nástěnky Designate!" diff --git a/releasenotes/source/locale/de/LC_MESSAGES/releasenotes.po b/releasenotes/source/locale/de/LC_MESSAGES/releasenotes.po new file mode 100644 index 0000000..cebef79 --- /dev/null +++ b/releasenotes/source/locale/de/LC_MESSAGES/releasenotes.po @@ -0,0 +1,46 @@ +# Adriano Perri , 2017. #zanata +# Frank Kloeker , 2017. #zanata +msgid "" +msgstr "" +"Project-Id-Version: Designate dashboard Client Release Notes 5.0.0\n" +"Report-Msgid-Bugs-To: \n" +"POT-Creation-Date: 2017-08-24 13:30+0000\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"PO-Revision-Date: 2017-08-25 12:08+0000\n" +"Last-Translator: Adriano Perri \n" +"Language-Team: German\n" +"Language: de\n" +"X-Generator: Zanata 3.9.6\n" +"Plural-Forms: nplurals=2; plural=(n != 1)\n" + +msgid ":ref:`genindex`" +msgstr ":ref:`genindex`" + +msgid ":ref:`search`" +msgstr ":ref:`search`" + +msgid "Contents" +msgstr "Inhalt" + +msgid "Current Series Release Notes" +msgstr "Aktuelle Serie Releasenotes" + +msgid "Indices and tables" +msgstr "Indizes und Tabellen" + +msgid "Mitaka Series Release Notes" +msgstr "Mitaka Serie Releasenotes" + +msgid "Newton Series Release Notes" +msgstr "Newton Serie Releasenotes" + +msgid "Ocata Series Release Notes" +msgstr "Ocata Serie Releasenotes" + +msgid "Pike Series Release Notes" +msgstr "Pike Serien Versionshinweise" + +msgid "Welcome to Designate dashboard Agent Release Notes documentation!" +msgstr "Willkommen bei den Releasenotes für den Designate Dashboard Agent" diff --git a/releasenotes/source/locale/en_GB/LC_MESSAGES/releasenotes.po b/releasenotes/source/locale/en_GB/LC_MESSAGES/releasenotes.po new file mode 100644 index 0000000..e30bd4c --- /dev/null +++ b/releasenotes/source/locale/en_GB/LC_MESSAGES/releasenotes.po @@ -0,0 +1,45 @@ +# Andi Chandler , 2017. #zanata +msgid "" +msgstr "" +"Project-Id-Version: Designate dashboard Client Release Notes 5.0.1\n" +"Report-Msgid-Bugs-To: \n" +"POT-Creation-Date: 2017-09-21 16:35+0000\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"PO-Revision-Date: 2017-09-28 02:18+0000\n" +"Last-Translator: Andi Chandler \n" +"Language-Team: English (United Kingdom)\n" +"Language: en-GB\n" +"X-Generator: Zanata 3.9.6\n" +"Plural-Forms: nplurals=2; plural=(n != 1)\n" + +msgid ":ref:`genindex`" +msgstr ":ref:`genindex`" + +msgid ":ref:`search`" +msgstr ":ref:`search`" + +msgid "Contents" +msgstr "Contents" + +msgid "Current Series Release Notes" +msgstr "Current Series Release Notes" + +msgid "Indices and tables" +msgstr "Indices and tables" + +msgid "Mitaka Series Release Notes" +msgstr "Mitaka Series Release Notes" + +msgid "Newton Series Release Notes" +msgstr "Newton Series Release Notes" + +msgid "Ocata Series Release Notes" +msgstr "Ocata Series Release Notes" + +msgid "Pike Series Release Notes" +msgstr "Pike Series Release Notes" + +msgid "Welcome to Designate dashboard Agent Release Notes documentation!" +msgstr "Welcome to Designate dashboard Agent Release Notes documentation!" diff --git a/releasenotes/source/locale/fr/LC_MESSAGES/releasenotes.po b/releasenotes/source/locale/fr/LC_MESSAGES/releasenotes.po new file mode 100644 index 0000000..e732216 --- /dev/null +++ b/releasenotes/source/locale/fr/LC_MESSAGES/releasenotes.po @@ -0,0 +1,21 @@ +# Gaelle , 2017. #zanata +msgid "" +msgstr "" +"Project-Id-Version: Designate dashboard Client Release Notes\n" +"Report-Msgid-Bugs-To: \n" +"POT-Creation-Date: 2017-11-21 15:00+0000\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"PO-Revision-Date: 2017-11-14 07:50+0000\n" +"Last-Translator: Gaelle \n" +"Language-Team: French\n" +"Language: fr\n" +"X-Generator: Zanata 3.9.6\n" +"Plural-Forms: nplurals=2; plural=(n > 1)\n" + +msgid "Contents" +msgstr "Contenus" + +msgid "Current Series Release Notes" +msgstr "Notes sur la Release Actuelle" diff --git a/releasenotes/source/locale/id/LC_MESSAGES/releasenotes.po b/releasenotes/source/locale/id/LC_MESSAGES/releasenotes.po new file mode 100644 index 0000000..fe7e7fc --- /dev/null +++ b/releasenotes/source/locale/id/LC_MESSAGES/releasenotes.po @@ -0,0 +1,45 @@ +# suhartono , 2017. #zanata +msgid "" +msgstr "" +"Project-Id-Version: Designate dashboard Client Release Notes 5.0.0\n" +"Report-Msgid-Bugs-To: \n" +"POT-Creation-Date: 2017-08-23 14:30+0000\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"PO-Revision-Date: 2017-08-23 02:39+0000\n" +"Last-Translator: suhartono \n" +"Language-Team: Indonesian\n" +"Language: id\n" +"X-Generator: Zanata 3.9.6\n" +"Plural-Forms: nplurals=1; plural=0\n" + +msgid ":ref:`genindex`" +msgstr ":ref:`genindex`" + +msgid ":ref:`search`" +msgstr ":ref:`search`" + +msgid "Contents" +msgstr "Contents (Isi)" + +msgid "Current Series Release Notes" +msgstr "Series Release Notes saat ini" + +msgid "Indices and tables" +msgstr "Indices and tables (indeks dan tabel)" + +msgid "Mitaka Series Release Notes" +msgstr "Mitaka Series Release Notes" + +msgid "Newton Series Release Notes" +msgstr "Newton Series Release Notes" + +msgid "Ocata Series Release Notes" +msgstr "Ocata Series Release Notes" + +msgid "Pike Series Release Notes" +msgstr "Catatan Rilis Seri Pike" + +msgid "Welcome to Designate dashboard Agent Release Notes documentation!" +msgstr "Selamat Datang di dokumentasi Designate dashboard Agent Release Notes!" diff --git a/releasenotes/source/locale/ko_KR/LC_MESSAGES/releasenotes.po b/releasenotes/source/locale/ko_KR/LC_MESSAGES/releasenotes.po new file mode 100644 index 0000000..f8e4e36 --- /dev/null +++ b/releasenotes/source/locale/ko_KR/LC_MESSAGES/releasenotes.po @@ -0,0 +1,47 @@ +# Ian Y. Choi , 2017. #zanata +# Sungjin Kang , 2017. #zanata +# minwook-shin , 2017. #zanata +msgid "" +msgstr "" +"Project-Id-Version: Designate dashboard Client Release Notes 5.0.1\n" +"Report-Msgid-Bugs-To: \n" +"POT-Creation-Date: 2017-09-20 03:15+0000\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"PO-Revision-Date: 2017-09-20 05:25+0000\n" +"Last-Translator: minwook-shin \n" +"Language-Team: Korean (South Korea)\n" +"Language: ko-KR\n" +"X-Generator: Zanata 3.9.6\n" +"Plural-Forms: nplurals=1; plural=0\n" + +msgid ":ref:`genindex`" +msgstr ":ref:`genindex`" + +msgid ":ref:`search`" +msgstr ":ref:`search`" + +msgid "Contents" +msgstr "내용" + +msgid "Current Series Release Notes" +msgstr "최신 시리즈에 대한 릴리즈 노트" + +msgid "Indices and tables" +msgstr "인텍스 및 테이블" + +msgid "Mitaka Series Release Notes" +msgstr "Mitaka 시리즈에 대한 릴리즈 노트" + +msgid "Newton Series Release Notes" +msgstr "Newton 시리즈에 대한 릴리즈 노트" + +msgid "Ocata Series Release Notes" +msgstr "Ocata 시리즈에 대한 릴리즈 노트" + +msgid "Pike Series Release Notes" +msgstr "Pike 시리즈에 대한 릴리즈 노트" + +msgid "Welcome to Designate dashboard Agent Release Notes documentation!" +msgstr "대시 보드 에이전트 릴리스 노트 문서에 오신 것을 환영합니다!" diff --git a/releasenotes/source/locale/pt_BR/LC_MESSAGES/releasenotes.po b/releasenotes/source/locale/pt_BR/LC_MESSAGES/releasenotes.po new file mode 100644 index 0000000..073a2e5 --- /dev/null +++ b/releasenotes/source/locale/pt_BR/LC_MESSAGES/releasenotes.po @@ -0,0 +1,46 @@ +# André Franciosi , 2017. #zanata +msgid "" +msgstr "" +"Project-Id-Version: Designate dashboard Client Release Notes 5.0.0\n" +"Report-Msgid-Bugs-To: \n" +"POT-Creation-Date: 2017-08-22 13:02+0000\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"PO-Revision-Date: 2017-08-22 10:43+0000\n" +"Last-Translator: André Franciosi \n" +"Language-Team: Portuguese (Brazil)\n" +"Language: pt-BR\n" +"X-Generator: Zanata 3.9.6\n" +"Plural-Forms: nplurals=2; plural=(n != 1)\n" + +msgid ":ref:`genindex`" +msgstr ":ref:`genindex`" + +msgid ":ref:`search`" +msgstr ":ref:`search`" + +msgid "Contents" +msgstr "Conteúdo" + +msgid "Current Series Release Notes" +msgstr "Atual - Série de Notas de Versão" + +msgid "Indices and tables" +msgstr "Índices e tabelas" + +msgid "Mitaka Series Release Notes" +msgstr "Mitaka - Série de Notas de Versão" + +msgid "Newton Series Release Notes" +msgstr "Newton - Série de Notas de Versão" + +msgid "Ocata Series Release Notes" +msgstr "Ocata - Série de Notas de Versão" + +msgid "Pike Series Release Notes" +msgstr "Pike - Série de Notas de Versão" + +msgid "Welcome to Designate dashboard Agent Release Notes documentation!" +msgstr "" +"Bemvindo a documentação de Notas de Versão do Agente dashboard do Designate!" diff --git a/releasenotes/source/locale/ru/LC_MESSAGES/releasenotes.po b/releasenotes/source/locale/ru/LC_MESSAGES/releasenotes.po new file mode 100644 index 0000000..2d3e798 --- /dev/null +++ b/releasenotes/source/locale/ru/LC_MESSAGES/releasenotes.po @@ -0,0 +1,37 @@ +# Fedor Tarasenko , 2017. #zanata +msgid "" +msgstr "" +"Project-Id-Version: Designate dashboard Client Release Notes 4.0.0\n" +"Report-Msgid-Bugs-To: \n" +"POT-Creation-Date: 2017-02-03 00:10+0000\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"PO-Revision-Date: 2017-02-03 08:41+0000\n" +"Last-Translator: Fedor Tarasenko \n" +"Language-Team: Russian\n" +"Language: ru\n" +"X-Generator: Zanata 3.7.3\n" +"Plural-Forms: nplurals=3; plural=(n%10==1 && n%100!=11 ? 0 : n%10>=2 && n" +"%10<=4 && (n%100<10 || n%100>=20) ? 1 : 2)\n" + +msgid ":ref:`genindex`" +msgstr ":ref:`genindex`" + +msgid ":ref:`search`" +msgstr ":ref:`search`" + +msgid "Contents" +msgstr "Содержание" + +msgid "Current Series Release Notes" +msgstr "Примечания к текущему релизу" + +msgid "Mitaka Series Release Notes" +msgstr "Примечания к релизу Mitaka" + +msgid "Newton Series Release Notes" +msgstr "Примечания к релизу Newton" + +msgid "Ocata Series Release Notes" +msgstr "Примечания к релизу Ocata" diff --git a/releasenotes/source/locale/zh_CN/LC_MESSAGES/releasenotes.po b/releasenotes/source/locale/zh_CN/LC_MESSAGES/releasenotes.po new file mode 100644 index 0000000..7ae05c7 --- /dev/null +++ b/releasenotes/source/locale/zh_CN/LC_MESSAGES/releasenotes.po @@ -0,0 +1,42 @@ +# TigerFang , 2017. #zanata +msgid "" +msgstr "" +"Project-Id-Version: Designate dashboard Client Release Notes 5.0.0\n" +"Report-Msgid-Bugs-To: \n" +"POT-Creation-Date: 2017-05-04 14:35+0000\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"PO-Revision-Date: 2017-05-15 09:55+0000\n" +"Last-Translator: TigerFang \n" +"Language-Team: Chinese (China)\n" +"Language: zh-CN\n" +"X-Generator: Zanata 3.9.6\n" +"Plural-Forms: nplurals=1; plural=0\n" + +msgid ":ref:`genindex`" +msgstr ":ref:`genindex`" + +msgid ":ref:`search`" +msgstr ":ref:`search`" + +msgid "Contents" +msgstr "内容" + +msgid "Current Series Release Notes" +msgstr "当前版本发布说明" + +msgid "Indices and tables" +msgstr "索引和表格" + +msgid "Mitaka Series Release Notes" +msgstr "Mitaka 版本发布说明" + +msgid "Newton Series Release Notes" +msgstr "Newton版本发布说明" + +msgid "Ocata Series Release Notes" +msgstr "Ocata版本发布说明" + +msgid "Welcome to Designate dashboard Agent Release Notes documentation!" +msgstr "欢迎使用Designate控制台代理发布说明文档!" diff --git a/releasenotes/source/mitaka.rst b/releasenotes/source/mitaka.rst new file mode 100644 index 0000000..97ab8d1 --- /dev/null +++ b/releasenotes/source/mitaka.rst @@ -0,0 +1,6 @@ +============================ + Mitaka Series Release Notes +============================ + +.. release-notes:: + :branch: origin/stable/mitaka diff --git a/releasenotes/source/newton.rst b/releasenotes/source/newton.rst new file mode 100644 index 0000000..be21859 --- /dev/null +++ b/releasenotes/source/newton.rst @@ -0,0 +1,6 @@ +============================ + Newton Series Release Notes +============================ + +.. release-notes:: + :branch: origin/stable/newton diff --git a/releasenotes/source/ocata.rst b/releasenotes/source/ocata.rst new file mode 100644 index 0000000..ebe62f4 --- /dev/null +++ b/releasenotes/source/ocata.rst @@ -0,0 +1,6 @@ +=================================== + Ocata Series Release Notes +=================================== + +.. release-notes:: + :branch: origin/stable/ocata diff --git a/releasenotes/source/pike.rst b/releasenotes/source/pike.rst new file mode 100644 index 0000000..e43bfc0 --- /dev/null +++ b/releasenotes/source/pike.rst @@ -0,0 +1,6 @@ +=================================== + Pike Series Release Notes +=================================== + +.. release-notes:: + :branch: stable/pike diff --git a/releasenotes/source/unreleased.rst b/releasenotes/source/unreleased.rst new file mode 100644 index 0000000..875030f --- /dev/null +++ b/releasenotes/source/unreleased.rst @@ -0,0 +1,5 @@ +============================ +Current Series Release Notes +============================ + +.. release-notes:: diff --git a/requirements.txt b/requirements.txt new file mode 100644 index 0000000..e7b376a --- /dev/null +++ b/requirements.txt @@ -0,0 +1,7 @@ +# The order of packages is significant, because pip processes them in the order +# of appearance. Changing the order has an impact on the overall integration +# process, which may cause wedges in the gate later. +oslo.log>=3.30.0 # Apache-2.0 +pbr!=2.1.0,>=2.0.0 # Apache-2.0 +Babel!=2.4.0,>=2.3.4 # BSD +python-designateclient>=2.7.0 # Apache-2.0 diff --git a/setup.cfg b/setup.cfg new file mode 100644 index 0000000..0a48004 --- /dev/null +++ b/setup.cfg @@ -0,0 +1,46 @@ +[metadata] +name = designate-dashboard +summary = Designate Horizon UI bits +description-file = + README.rst +author = OpenStack +author-email = openstack-dev@lists.openstack.org +home-page = http://docs.openstack.org/developer/designate-dashboard/ +classifier = + Environment :: OpenStack + Intended Audience :: Information Technology + Intended Audience :: System Administrators + License :: OSI Approved :: Apache Software License + Operating System :: POSIX :: Linux + Programming Language :: Python + Programming Language :: Python :: 2 + Programming Language :: Python :: 2.7 + Programming Language :: Python :: 3 + Programming Language :: Python :: 3.5 + +[files] +packages = + designatedashboard + +[build_sphinx] +source-dir = doc/source +build-dir = doc/build +all_files = 1 +warning-is-error = 1 + +[upload_sphinx] +upload-dir = doc/build/html + +[compile_catalog] +directory = designatedashboard/locale +domain = django + +[update_catalog] +domain = django +output_dir = designatedashboard/locale +input_file = designatedashboard/locale/django.pot + +[extract_messages] +keywords = _ gettext ngettext l_ lazy_gettext +mapping_file = babel-django.cfg +output_file = designatedashboard/locale/django.pot diff --git a/setup.py b/setup.py new file mode 100644 index 0000000..566d844 --- /dev/null +++ b/setup.py @@ -0,0 +1,29 @@ +# Copyright (c) 2013 Hewlett-Packard Development Company, L.P. +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or +# implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +# THIS FILE IS MANAGED BY THE GLOBAL REQUIREMENTS REPO - DO NOT EDIT +import setuptools + +# In python < 2.7.4, a lazy loading of package `pbr` will break +# setuptools if some other modules registered functions in `atexit`. +# solution from: http://bugs.python.org/issue15881#msg170215 +try: + import multiprocessing # noqa +except ImportError: + pass + +setuptools.setup( + setup_requires=['pbr>=2.0.0'], + pbr=True) diff --git a/test b/test new file mode 100644 index 0000000000000000000000000000000000000000..dd690e30fb899c8f50d20c98f0e24369a7db770f GIT binary patch literal 2048 zcmWFz^vNtqRY=P(%1ta$FlJz3U}R))P*7lCV2}V}CMafv@>zg141mN40SN}(XH2{x lRY0B0+XyKil^+d((GVai1iXk+HL7Sd1V%$(Gz4&l0041E2rvKu literal 0 HcmV?d00001 diff --git a/test-requirements.txt b/test-requirements.txt new file mode 100644 index 0000000..3c3716d --- /dev/null +++ b/test-requirements.txt @@ -0,0 +1,26 @@ +# The order of packages is significant, because pip processes them in the order +# of appearance. Changing the order has an impact on the overall integration +# process, which may cause wedges in the gate later. +hacking!=0.13.0,<0.14,>=0.12.0 # Apache-2.0 + +coverage!=4.4,>=4.0 # Apache-2.0 +mock>=2.0.0 # BSD +mox>=0.5.3 # Apache-2.0 +mox3>=0.20.0 # Apache-2.0 +oslo.config>=5.1.0 # Apache-2.0 +pylint==1.4.5 # GPLv2 +testrepository>=0.0.18 # Apache-2.0/BSD +testtools>=2.2.0 # MIT +unittest2>=1.1.0 # BSD +sphinx>=1.6.2 # BSD +openstackdocstheme>=1.17.0 # Apache-2.0 +reno>=2.5.0 # Apache-2.0 +nose>=1.3.7 # LGPL +nosehtmloutput>=0.0.3 # Apache-2.0 +openstack.nose-plugin>=0.7 # Apache-2.0 +django-nose>=1.4.4 # BSD +nosexcover>=1.0.10 # BSD + +# Horizon requirements +Django<2.0,>=1.8 # BSD +django-compressor>=2.0 # MIT diff --git a/test-shim.js b/test-shim.js new file mode 100644 index 0000000..176b1c3 --- /dev/null +++ b/test-shim.js @@ -0,0 +1,110 @@ +/** + * Licensed under the Apache License, Version 2.0 (the "License"); you may + * not use this file except in compliance with the License. You may obtain + * a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT + * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the + * License for the specific language governing permissions and limitations + * under the License. + */ + +/* + * Shim for Javascript unit tests; supplying expected global features. + * This should be removed from the codebase once i18n services are provided. + * Taken from default i18n file provided by Django. + */ + +var horizonPlugInModules = []; + + +(function (globals) { + + var django = globals.django || (globals.django = {}); + + + django.pluralidx = function (count) { return (count == 1) ? 0 : 1; }; + + /* gettext identity library */ + + django.gettext = function (msgid) { return msgid; }; + django.ngettext = function (singular, plural, count) { return (count == 1) ? singular : plural; }; + django.gettext_noop = function (msgid) { return msgid; }; + django.pgettext = function (context, msgid) { return msgid; }; + django.npgettext = function (context, singular, plural, count) { return (count == 1) ? singular : plural; }; + + + django.interpolate = function (fmt, obj, named) { + if (named) { + return fmt.replace(/%\(\w+\)s/g, function(match){return String(obj[match.slice(2,-2)])}); + } else { + return fmt.replace(/%s/g, function(match){return String(obj.shift())}); + } + }; + + + /* formatting library */ + + django.formats = { + "DATETIME_FORMAT": "N j, Y, P", + "DATETIME_INPUT_FORMATS": [ + "%Y-%m-%d %H:%M:%S", + "%Y-%m-%d %H:%M:%S.%f", + "%Y-%m-%d %H:%M", + "%Y-%m-%d", + "%m/%d/%Y %H:%M:%S", + "%m/%d/%Y %H:%M:%S.%f", + "%m/%d/%Y %H:%M", + "%m/%d/%Y", + "%m/%d/%y %H:%M:%S", + "%m/%d/%y %H:%M:%S.%f", + "%m/%d/%y %H:%M", + "%m/%d/%y" + ], + "DATE_FORMAT": "N j, Y", + "DATE_INPUT_FORMATS": [ + "%Y-%m-%d", + "%m/%d/%Y", + "%m/%d/%y" + ], + "DECIMAL_SEPARATOR": ".", + "FIRST_DAY_OF_WEEK": "0", + "MONTH_DAY_FORMAT": "F j", + "NUMBER_GROUPING": "3", + "SHORT_DATETIME_FORMAT": "m/d/Y P", + "SHORT_DATE_FORMAT": "m/d/Y", + "THOUSAND_SEPARATOR": ",", + "TIME_FORMAT": "P", + "TIME_INPUT_FORMATS": [ + "%H:%M:%S", + "%H:%M:%S.%f", + "%H:%M" + ], + "YEAR_MONTH_FORMAT": "F Y" + }; + + django.get_format = function (format_type) { + var value = django.formats[format_type]; + if (typeof(value) == 'undefined') { + return format_type; + } else { + return value; + } + }; + + /* add to global namespace */ + globals.pluralidx = django.pluralidx; + globals.gettext = django.gettext; + globals.ngettext = django.ngettext; + globals.gettext_noop = django.gettext_noop; + globals.pgettext = django.pgettext; + globals.npgettext = django.npgettext; + globals.interpolate = django.interpolate; + globals.get_format = django.get_format; + globals.STATIC_URL = '/static/'; + globals.WEBROOT = '/'; + +}(this)); diff --git a/tox.ini b/tox.ini new file mode 100644 index 0000000..97a23de --- /dev/null +++ b/tox.ini @@ -0,0 +1,48 @@ +[tox] +minversion = 1.6 +envlist = py35,py27,pep8 +skipsdist = True + +[testenv] +usedevelop = True +install_command = pip install -c{env:UPPER_CONSTRAINTS_FILE:https://git.openstack.org/cgit/openstack/requirements/plain/upper-constraints.txt?h=master} {opts} {packages} +setenv = VIRTUAL_ENV={envdir} + NOSE_WITH_OPENSTACK=1 + NOSE_OPENSTACK_COLOR=1 + NOSE_OPENSTACK_RED=0.05 + NOSE_OPENSTACK_YELLOW=0.025 + NOSE_OPENSTACK_SHOW_ELAPSED=1 + PYTHONDONTWRITEBYTECODE=1 + DJANGO_SETTINGS_MODULE=designatedashboard.settings +whitelist_externals = find +deps = -r{toxinidir}/requirements.txt + -r{toxinidir}/test-requirements.txt + http://tarballs.openstack.org/horizon/horizon-master.tar.gz + +commands = + find . -type f -name "*.pyc" -delete + {toxinidir}/manage.py test designatedashboard --settings=designatedashboard.tests.settings +passenv = http_proxy HTTP_PROXY https_proxy HTTPS_PROXY no_proxy NO_PROXY + +[testenv:pep8] +commands = flake8 + +[testenv:venv] +commands = {posargs} + +[testenv:cover] +commands = {toxinidir}/manage.py test designatedashboard --settings=designatedashboard.tests.settings --cover-xml + +[testenv:docs] +commands = python setup.py build_sphinx + +[flake8] +# E123, E125 skipped as they are invalid PEP-8. + +show-source = True +ignore = E123,E125 +builtins = _ +exclude=.venv,.git,.tox,dist,doc,*openstack/common*,*lib/python*,*egg,build,releasenotes + +[testenv:releasenotes] +commands = sphinx-build -a -E -d releasenotes/build/doctrees -b html releasenotes/source releasenotes/build/html