rubyfox-server 2.17.3.2 → 2.19.2.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (160) hide show
  1. checksums.yaml +4 -4
  2. data/lib/rubyfox/server/data/config/admin/descriptors/config_room.txt +10 -1
  3. data/lib/rubyfox/server/data/config/admin/descriptors/config_server.txt +90 -20
  4. data/lib/rubyfox/server/data/config/admin/descriptors/config_zone.txt +9 -0
  5. data/lib/rubyfox/server/data/config/admin/descriptors/runtime_room.txt +11 -0
  6. data/lib/rubyfox/server/data/config/admin/descriptors/runtime_user.txt +3 -3
  7. data/lib/rubyfox/server/data/config/core.xml +4 -4
  8. data/lib/rubyfox/server/data/config/default.words.txt +11 -0
  9. data/lib/rubyfox/server/data/config/log4j.properties +1 -2
  10. data/lib/rubyfox/server/data/config/server.xml +1 -1
  11. data/lib/rubyfox/server/data/data/GeoLite2-Country.mmdb +0 -0
  12. data/lib/rubyfox/server/data/data/bannedusers/users.bin +0 -0
  13. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/bootstrap.jar +0 -0
  14. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/catalina-tasks.xml +39 -39
  15. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/catalina.sh +0 -0
  16. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/ciphers.sh +0 -0
  17. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/commons-daemon-native.tar.gz +0 -0
  18. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/commons-daemon.jar +0 -0
  19. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/configtest.sh +0 -0
  20. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/daemon.sh +0 -0
  21. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/digest.sh +0 -0
  22. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/makebase.sh +0 -0
  23. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/setclasspath.sh +0 -0
  24. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/shutdown.sh +0 -0
  25. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/startup.sh +0 -0
  26. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/tomcat-juli.jar +0 -0
  27. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/tomcat-native.tar.gz +0 -0
  28. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/tool-wrapper.sh +0 -0
  29. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/version.sh +0 -0
  30. data/lib/rubyfox/server/data/lib/apache-tomcat/conf/Catalina/localhost/rewrite.config +1 -1
  31. data/lib/rubyfox/server/data/lib/apache-tomcat/conf/catalina.policy +263 -263
  32. data/lib/rubyfox/server/data/lib/apache-tomcat/conf/catalina.properties +209 -207
  33. data/lib/rubyfox/server/data/lib/apache-tomcat/conf/context.xml +31 -31
  34. data/lib/rubyfox/server/data/lib/apache-tomcat/conf/jaspic-providers.xml +23 -23
  35. data/lib/rubyfox/server/data/lib/apache-tomcat/conf/jaspic-providers.xsd +52 -52
  36. data/lib/rubyfox/server/data/lib/apache-tomcat/conf/keystore.jks +0 -0
  37. data/lib/rubyfox/server/data/lib/apache-tomcat/conf/server.xml +177 -161
  38. data/lib/rubyfox/server/data/lib/apache-tomcat/conf/tomcat-users.xml +18 -7
  39. data/lib/rubyfox/server/data/lib/apache-tomcat/conf/tomcat-users.xsd +59 -59
  40. data/lib/rubyfox/server/data/lib/apache-tomcat/conf/web.xml +4740 -4737
  41. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/annotations-api.jar +0 -0
  42. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/catalina-ant.jar +0 -0
  43. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/catalina-ha.jar +0 -0
  44. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/catalina-ssi.jar +0 -0
  45. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/catalina-storeconfig.jar +0 -0
  46. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/catalina-tribes.jar +0 -0
  47. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/catalina.jar +0 -0
  48. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/el-api.jar +0 -0
  49. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/jasper-el.jar +0 -0
  50. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/jasper.jar +0 -0
  51. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/jaspic-api.jar +0 -0
  52. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/jsp-api.jar +0 -0
  53. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/servlet-api.jar +0 -0
  54. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/sfs2x-ws-helper.jar +0 -0
  55. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-api.jar +0 -0
  56. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-coyote.jar +0 -0
  57. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-dbcp.jar +0 -0
  58. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-cs.jar +0 -0
  59. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-de.jar +0 -0
  60. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-es.jar +0 -0
  61. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-fr.jar +0 -0
  62. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-ja.jar +0 -0
  63. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-ko.jar +0 -0
  64. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-pt-BR.jar +0 -0
  65. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-ru.jar +0 -0
  66. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-zh-CN.jar +0 -0
  67. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-jdbc.jar +0 -0
  68. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-jni.jar +0 -0
  69. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-util-scan.jar +0 -0
  70. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-util.jar +0 -0
  71. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-websocket.jar +0 -0
  72. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/websocket-api.jar +0 -0
  73. data/lib/rubyfox/server/data/lib/javax.activation-1.2.0.jar +0 -0
  74. data/lib/rubyfox/server/data/lib/javax.mail.jar +0 -0
  75. data/lib/rubyfox/server/data/lib/js/JSApi.js +2 -1
  76. data/lib/rubyfox/server/data/lib/js/LibApi.js +181 -48
  77. data/lib/rubyfox/server/data/lib/sfs2x-admin.jar +0 -0
  78. data/lib/rubyfox/server/data/lib/sfs2x-cluster.jar +0 -0
  79. data/lib/rubyfox/server/data/lib/sfs2x-core.jar +0 -0
  80. data/lib/rubyfox/server/data/lib/sfs2x.jar +0 -0
  81. data/lib/rubyfox/server/data/sfs2x-service +26 -30
  82. data/lib/rubyfox/server/data/www/BlueBox.war +0 -0
  83. data/lib/rubyfox/server/data/www/HelloServlet/WEB-INF/web.xml +1 -3
  84. data/lib/rubyfox/server/data/www/ROOT/_css_/default.css +14 -6
  85. data/lib/rubyfox/server/data/www/ROOT/admin/assets/css/style.css +44 -2
  86. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/application.bundle.js +98 -61
  87. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/endors~mod-0~mod-1~mod-11~mod-12~mod-17~mod-6~mod-7~mod-8~mod-9.bundle.js +17357 -0
  88. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-0.bundle.js +4 -4
  89. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-1.bundle.js +3 -3
  90. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-10.bundle.js +101 -66
  91. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-11.bundle.js +544 -8
  92. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-12.bundle.js +915 -1480
  93. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-12~module-15~module-16~module-4.bundle.js +2665 -0
  94. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-13.bundle.js +606 -3093
  95. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-13~module-16~module-17~module-4.bundle.js +2665 -0
  96. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-14.bundle.js +764 -0
  97. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-15.bundle.js +71 -0
  98. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-16.bundle.js +1787 -0
  99. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-17.bundle.js +3383 -0
  100. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-4.bundle.js +121 -1009
  101. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-5.bundle.js +1214 -1744
  102. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-6.bundle.js +398 -666
  103. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-7.bundle.js +717 -192
  104. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-8.bundle.js +2117 -665
  105. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-9.bundle.js +613 -690
  106. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/vendors~mod-0~mod-1~mod-10~mod-11~mod-16~mod-5~mod-6~mod-7~mod-8.bundle.js +17357 -0
  107. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/vendors~mod-0~mod-1~mod-11~mod-12~mod-17~mod-5~mod-6~mod-7~mod-8~mod-9.bundle.js +17357 -0
  108. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/{vendors~module-0~module-1~module-13~module-4~module-5~module-7~module-8.bundle.js → vendors~mod-0~mod-1~mod-11~mod-12~mod-17~mod-5~mod-7~mod-8~mod-9.bundle.js} +2 -2
  109. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/vendors~module-12.bundle.js +807 -0
  110. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/vendors~module-13.bundle.js +807 -0
  111. data/lib/rubyfox/server/data/www/ROOT/admin/modules/cluster-configurator.html +32 -0
  112. data/lib/rubyfox/server/data/www/ROOT/admin/modules/cluster-monitor.html +185 -0
  113. data/lib/rubyfox/server/data/www/ROOT/admin/modules/cluster-updater.html +47 -0
  114. data/lib/rubyfox/server/data/www/ROOT/admin/modules/extension-deployer.html +84 -0
  115. data/lib/rubyfox/server/data/www/ROOT/admin/modules/zone-monitor.html +15 -8
  116. data/lib/rubyfox/server/data/www/ROOT/index.html +13 -23
  117. data/lib/rubyfox/server/data/www/host-manager/META-INF/context.xml +2 -2
  118. data/lib/rubyfox/server/data/www/host-manager/WEB-INF/jsp/404.jsp +2 -2
  119. data/lib/rubyfox/server/data/www/host-manager/{manager.xml → WEB-INF/manager.xml} +5 -1
  120. data/lib/rubyfox/server/data/www/host-manager/WEB-INF/web.xml +17 -0
  121. data/lib/rubyfox/server/data/www/host-manager/css/manager.css +141 -0
  122. data/lib/rubyfox/server/data/www/host-manager/images/tomcat.svg +967 -0
  123. data/lib/rubyfox/server/data/www/manager/META-INF/context.xml +2 -0
  124. data/lib/rubyfox/server/data/www/manager/WEB-INF/jsp/connectorCerts.jsp +1 -1
  125. data/lib/rubyfox/server/data/www/manager/WEB-INF/jsp/connectorCiphers.jsp +1 -1
  126. data/lib/rubyfox/server/data/www/manager/WEB-INF/jsp/connectorTrustedCerts.jsp +1 -1
  127. data/lib/rubyfox/server/data/www/manager/WEB-INF/jsp/sessionDetail.jsp +3 -3
  128. data/lib/rubyfox/server/data/www/manager/WEB-INF/jsp/sessionsList.jsp +1 -1
  129. data/lib/rubyfox/server/data/www/manager/WEB-INF/web.xml +17 -0
  130. data/lib/rubyfox/server/data/www/manager/css/manager.css +141 -0
  131. data/lib/rubyfox/server/data/www/manager/images/tomcat.svg +967 -0
  132. data/lib/rubyfox/server/data/www/manager/xform.xsl +74 -59
  133. data/lib/rubyfox/server/version.rb +1 -1
  134. metadata +30 -31
  135. data/lib/rubyfox/server/data/config/admin/icons/Analytics.png +0 -0
  136. data/lib/rubyfox/server/data/config/admin/icons/BanManager.png +0 -0
  137. data/lib/rubyfox/server/data/config/admin/icons/BlueBoxMonitor.png +0 -0
  138. data/lib/rubyfox/server/data/config/admin/icons/Console.png +0 -0
  139. data/lib/rubyfox/server/data/config/admin/icons/Dashboard.png +0 -0
  140. data/lib/rubyfox/server/data/config/admin/icons/ExtensionManager.png +0 -0
  141. data/lib/rubyfox/server/data/config/admin/icons/LicenseManager.png +0 -0
  142. data/lib/rubyfox/server/data/config/admin/icons/LogViewer.png +0 -0
  143. data/lib/rubyfox/server/data/config/admin/icons/ServerConfigurator.png +0 -0
  144. data/lib/rubyfox/server/data/config/admin/icons/ServletManager.png +0 -0
  145. data/lib/rubyfox/server/data/config/admin/icons/ZoneConfigurator.png +0 -0
  146. data/lib/rubyfox/server/data/config/admin/icons/ZoneMonitor.png +0 -0
  147. data/lib/rubyfox/server/data/lib/BlueBox.war +0 -0
  148. data/lib/rubyfox/server/data/lib/apache-tomcat/LICENSE +0 -1061
  149. data/lib/rubyfox/server/data/lib/apache-tomcat/NOTICE +0 -68
  150. data/lib/rubyfox/server/data/lib/apache-tomcat/README.md +0 -81
  151. data/lib/rubyfox/server/data/lib/apache-tomcat/RELEASE-NOTES +0 -174
  152. data/lib/rubyfox/server/data/lib/imap.jar +0 -0
  153. data/lib/rubyfox/server/data/lib/mailapi.jar +0 -0
  154. data/lib/rubyfox/server/data/lib/pop3.jar +0 -0
  155. data/lib/rubyfox/server/data/lib/smtp.jar +0 -0
  156. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-12~module-13~module-9.bundle.js +0 -2634
  157. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/vendors~module-9.bundle.js +0 -807
  158. data/lib/rubyfox/server/data/www/host-manager/images/tomcat.gif +0 -0
  159. data/lib/rubyfox/server/data/www/manager/images/tomcat.gif +0 -0
  160. /data/lib/rubyfox/server/data/data/buddylists/{BasicExamples/.keep → .keep} +0 -0
@@ -0,0 +1,1787 @@
1
+ /*! (c) gotoAndPlay | All rights reserved */
2
+ (window["webpackJsonpapplication"] = window["webpackJsonpapplication"] || []).push([["module-16"],{
3
+
4
+ /***/ "./src/components/module-specific/words-files-manager.js":
5
+ /*!***************************************************************!*\
6
+ !*** ./src/components/module-specific/words-files-manager.js ***!
7
+ \***************************************************************/
8
+ /*! exports provided: WordsFilesManager */
9
+ /***/ (function(module, __webpack_exports__, __webpack_require__) {
10
+
11
+ "use strict";
12
+ __webpack_require__.r(__webpack_exports__);
13
+ /* WEBPACK VAR INJECTION */(function($) {/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "WordsFilesManager", function() { return WordsFilesManager; });
14
+ /* harmony import */ var _utils_utilities__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../../utils/utilities */ "./src/utils/utilities.js");
15
+
16
+
17
+ class WordsFilesManager extends HTMLElement
18
+ {
19
+ constructor()
20
+ {
21
+ super();
22
+
23
+ this.REFRESH_WORDS_FILES_CLICK_EVENT = 'refreshWordsFilesClick';
24
+ this.EDIT_WORDS_FILE_CLICK_EVENT = 'editWordsFileClick';
25
+ this.SAVE_WORDS_FILE_CLICK_EVENT = 'saveWordsFileClick';
26
+ this.REMOVE_WORDS_FILE_CLICK_EVENT = 'removeWordsFileClick';
27
+ this.ASSIGN_WORDS_FILE_CLICK_EVENT = 'assingWordsFileClick';
28
+
29
+ this.CONFIG_FOLDER = 'config/';
30
+ this.WORDS_FILE_EXT = '.words.txt';
31
+
32
+ this._modalHtml = `
33
+ <div class="modal" id="editModal" tabindex="-1" role="dialog" aria-labelledby="editModalTitle" aria-hidden="true">
34
+ <div class="modal-dialog modal-dialog-centered" role="document">
35
+ <div class="modal-content">
36
+ <div class="modal-header">
37
+ <h5 class="modal-title text-primary" id="editModalTitle">Word File Editor</h5>
38
+ <button type="button" class="close" data-dismiss="modal" aria-label="Close">
39
+ <span aria-hidden="true">&times;</span>
40
+ </button>
41
+ </div>
42
+ <div class="modal-body in-flow-invalid-msg">
43
+ <fieldset id="editFieldset">
44
+ <div class="form-group">
45
+ <div class="col-form-label form-label-container">
46
+ <label for="filename" class="form-label">Filename <i class="fas fa-question-circle text-muted help" title="Name of the words file. The file will be saved in server's <em>config</em> folder."></i></label>
47
+ </div>
48
+ <div class="inner-widget">
49
+ <div class="input-group">
50
+ <input type="text" id="filename" name="filename" class="form-control k-textbox" autocomplete="off" required data-required-msg="Required" aria-describedby="extension" />
51
+ <div class="input-group-append">
52
+ <span class="input-group-text" id="extension">.words.txt</span>
53
+ </div>
54
+ </div>
55
+ <span class="k-invalid-msg position-static" data-for="filename"></span>
56
+ </div>
57
+ </div>
58
+ <div class="form-group">
59
+ <div class="col-form-label form-label-container">
60
+ <label for="content" class="form-label">Content <i class="fas fa-question-circle text-muted help" title="Enter a word or a valid regular expression per line. See configuration's <em>Words file</em> field description for additional information."></i></label>
61
+ </div>
62
+ <div class="inner-widget">
63
+ <textarea id="content" name="content" class="form-control k-textarea w-100" rows="10"></textarea>
64
+ <span class="k-invalid-msg position-static" data-for="content"></span>
65
+ </div>
66
+ </div>
67
+ </fieldset>
68
+ </div>
69
+ <div class="modal-footer flex-column">
70
+ <div class="d-flex w-100">
71
+ <div class="flex-grow-1 text-left">
72
+ <button id="saveWordFileButton" type="button" class="k-button k-primary"><i class="fas fa-save mr-1"></i>Save word file</button>
73
+ <i id="saveSpinner" class="fas fa-circle-notch fa-spin text-primary align-middle ml-1"></i>
74
+ </div>
75
+ <div class="flex-grow-1 text-right">
76
+ <button type="button" class="k-button k-secondary" data-dismiss="modal">Cancel</button>
77
+ </div>
78
+ </div>
79
+ </div>
80
+ </div>
81
+ </div>
82
+ </div>
83
+ `;
84
+
85
+ //-------------------------------------------
86
+
87
+ $(this).append(`
88
+ <div class="col-sm-5 col-lg-4 col-form-label form-label-container">
89
+ <label class="form-label">Available words files <i class="fas fa-question-circle text-muted help" title="The list of words files found in server's <em>config</em> folder. Click on the Assign button to set the <strong>Words file</strong> field to the selected file. Configuration submission is then required to save the new value."></i></label>
90
+ </div>
91
+ <div class="inner-widget align-self-center col-sm">
92
+ <div>
93
+ <div id="wordsFiles" class="limited-height"></div>
94
+ <div id="actionButtons" class="mt-2 text-right" disabled>
95
+ <i id="actionSpinner" class="fas fa-circle-notch fa-spin text-primary align-middle"></i>
96
+ <button id="refreshButton" type="button" class="k-button k-secondary ml-2" title="Refresh"><i class="fas fa-redo-alt"></i></button>
97
+ <button id="addButton" type="button" class="k-button k-secondary ml-2" title="Add"><i class="fas fa-plus"></i></button>
98
+ <button id="editButton" type="button" class="k-button k-secondary ml-2" title="Edit" disabled><i class="fas fa-pen"></i></button>
99
+ <button id="removeButton" type="button" class="k-button k-secondary ml-2" title="Remove" disabled><i class="fas fa-times"></i></button>
100
+ <button id="assignButton" type="button" class="k-button k-secondary ml-2" title="Assign" disabled><i class="fas fa-compress-arrows-alt"></i></button>
101
+ </div>
102
+ </div>
103
+ </div>
104
+ `);
105
+
106
+ //-------------------------------------------
107
+
108
+ // Initialize grid
109
+ this._wordsFilesGrid = $('#wordsFiles', $(this)).kendoGrid({
110
+ resizable: true,
111
+ selectable: 'row',
112
+ change: $.proxy(this._onWordsFilesGridSelectionChange, this),
113
+ columns: [
114
+ {
115
+ field: 'name',
116
+ title: 'Filename',
117
+ width: 120
118
+ },
119
+ {
120
+ field: 'date',
121
+ title: 'Date',
122
+ width: 80
123
+ },
124
+ {
125
+ field: 'size',
126
+ title: 'Size',
127
+ width: 80
128
+ }
129
+ ],
130
+ noRecords: {
131
+ template: 'No files.'
132
+ }
133
+ }).data('kendoGrid');
134
+
135
+ // Add listeners to button clicks
136
+ $('#refreshButton', $(this)).on('click', $.proxy(this._onReloadClick, this));
137
+ $('#addButton', $(this)).on('click', $.proxy(this._onAddClick, this));
138
+ $('#editButton', $(this)).on('click', $.proxy(this._onEditClick, this));
139
+ $('#removeButton', $(this)).on('click', $.proxy(this._onRemoveClick, this));
140
+ $('#assignButton', $(this)).on('click', $.proxy(this._onAssignClick, this));
141
+ }
142
+
143
+ destroy()
144
+ {
145
+ // Destroy grid
146
+ this._wordsFilesGrid.destroy();
147
+
148
+ // Remove event listeners
149
+ $('#refreshButton', $(this)).off('click');
150
+ $('#addButton', $(this)).off('click');
151
+ $('#editButton', $(this)).off('click');
152
+ $('#removeButton', $(this)).off('click');
153
+ $('#assignButton', $(this)).off('click');
154
+
155
+ // Hide modal (which in turn destroys it)
156
+ let modalElement = $('#editModal', $(this));
157
+
158
+ if (modalElement)
159
+ modalElement.modal('hide');
160
+ }
161
+
162
+ get enabled()
163
+ {
164
+ return this._isEnabled;
165
+ }
166
+
167
+ set enabled(value)
168
+ {
169
+ this._isEnabled = value;
170
+
171
+ // Enable/disable buttons
172
+ $('#actionButtons', this).attr('disabled', !value);
173
+
174
+ // Hide spinner
175
+ if (value)
176
+ this.actionSpinnerVisible = false;
177
+
178
+ // Enable/disable modal
179
+ let modalElement = $('#editModal', $(this));
180
+
181
+ if (modalElement)
182
+ {
183
+ // Disable modal close buttons
184
+ $('button[data-dismiss="modal"]', modalElement).attr('disabled', !value);
185
+
186
+ // Disable save button
187
+ $('#saveWordFileButton', modalElement).attr('disabled', !value);
188
+
189
+ // Disable fieldset
190
+ $('#editFieldset', modalElement).attr('disabled', !value);
191
+
192
+ // Hide spinner
193
+ if (value)
194
+ this.saveSpinnerVisible = false;
195
+ }
196
+ }
197
+
198
+ set actionSpinnerVisible(value)
199
+ {
200
+ if (value)
201
+ $('#actionSpinner', $(this)).show();
202
+ else
203
+ $('#actionSpinner', $(this)).hide();
204
+ }
205
+
206
+ set saveSpinnerVisible(value)
207
+ {
208
+ let modalElement = $('#editModal', $(this));
209
+
210
+ if (modalElement)
211
+ {
212
+ if (value)
213
+ $('#saveSpinner', modalElement).show();
214
+ else
215
+ $('#saveSpinner', modalElement).hide();
216
+ }
217
+ }
218
+
219
+ refreshWordsFilesList(wordsFilesList, hideEditModal)
220
+ {
221
+ if (hideEditModal)
222
+ {
223
+ let modalElement = $('#editModal', $(this));
224
+
225
+ if (modalElement)
226
+ modalElement.modal('hide');
227
+ }
228
+
229
+ let files = [];
230
+ this._existingFilenames = [];
231
+
232
+ for (let f = 0; f < wordsFilesList.size(); f++)
233
+ {
234
+ const file = wordsFilesList.getSFSObject(f);
235
+
236
+ const fileObj = {};
237
+ fileObj.name = file.getUtfString('name');
238
+ fileObj.date = file.getUtfString('date') + ' ' + file.getUtfString('time');
239
+ fileObj.size = Object(_utils_utilities__WEBPACK_IMPORTED_MODULE_0__["bytesToSize"])(file.getLong('size'), 2);
240
+
241
+ // Populate files list
242
+ files.push(fileObj);
243
+
244
+ // Save ref to existing filenames, to check them when a new file is created
245
+ this._existingFilenames.push(fileObj.name);
246
+ }
247
+
248
+ // Assign data source to grid
249
+ this._setWordsFilesGridDataSource(files);
250
+ this._onWordsFilesGridSelectionChange();
251
+
252
+ // Enable
253
+ this.enabled = true;
254
+ }
255
+
256
+ getSelectedWordsFileName()
257
+ {
258
+ if (this._wordsFilesGrid.select() != null)
259
+ {
260
+ let selectedIndex = this._wordsFilesGrid.select().index();
261
+ return this._wordsFilesGrid.dataSource.at(selectedIndex).name;
262
+ }
263
+ else
264
+ return null;
265
+ }
266
+
267
+ editWordsFile(filename, content)
268
+ {
269
+ this._isNewFile = false;
270
+
271
+ this.enabled = true;
272
+
273
+ // Show modal
274
+ this._showModal();
275
+
276
+ // Remove default extension from filename
277
+ filename = filename.substring(0, filename.lastIndexOf(this.WORDS_FILE_EXT));
278
+
279
+ // Enter content filename and content in modal form
280
+ $('#editModal #filename', $(this)).val(filename);
281
+ $('#editModal #content', $(this)).val(content);
282
+
283
+ // Set filename field as not editable and hide note
284
+ $('#editModal #filename', $(this)).attr('disabled', true);
285
+ $('#editModal #filenameNote', $(this)).hide();
286
+ }
287
+
288
+ getExistingFilenames()
289
+ {
290
+ return this._existingFilenames;
291
+ }
292
+
293
+ _setWordsFilesGridDataSource(ds)
294
+ {
295
+ // Read current horizontal scroll value
296
+ const scrollLeft = $('#wordsFiles .k-grid-content', this._wordsFilesGrid.wrapper).scrollLeft();
297
+
298
+ // Assign data source to grid
299
+ this._wordsFilesGrid.setDataSource(ds);
300
+
301
+ // Set horizontal scroll
302
+ $('#wordsFiles .k-grid-content', this._wordsFilesGrid.wrapper).scrollLeft(scrollLeft);
303
+ }
304
+
305
+ _onWordsFilesGridSelectionChange()
306
+ {
307
+ // Enable/disable buttons
308
+ const selectedRows = this._wordsFilesGrid.select();
309
+ $('#editButton').attr('disabled', selectedRows.length == 0);
310
+ $('#removeButton').attr('disabled', selectedRows.length == 0);
311
+ $('#assignButton').attr('disabled', selectedRows.length == 0);
312
+ }
313
+
314
+ _onReloadClick()
315
+ {
316
+ // Fire event to request file content to server
317
+ let evt = new CustomEvent(this.REFRESH_WORDS_FILES_CLICK_EVENT, {
318
+ detail: null,
319
+ bubbles: false,
320
+ cancelable: false
321
+ });
322
+
323
+ this.dispatchEvent(evt);
324
+ }
325
+
326
+ _onAddClick()
327
+ {
328
+ this._isNewFile = true;
329
+
330
+ // Show modal
331
+ this._showModal();
332
+ }
333
+
334
+ _onEditClick()
335
+ {
336
+ // Disable buttons
337
+ this.enabled = false;
338
+ this.actionSpinnerVisible = true;
339
+
340
+ // Fire event to request file content to server
341
+ let evt = new CustomEvent(this.EDIT_WORDS_FILE_CLICK_EVENT, {
342
+ detail: this.getSelectedWordsFileName(),
343
+ bubbles: false,
344
+ cancelable: false
345
+ });
346
+
347
+ this.dispatchEvent(evt);
348
+ }
349
+
350
+ _onRemoveClick()
351
+ {
352
+ // Fire event to request file removal to server
353
+ let evt = new CustomEvent(this.REMOVE_WORDS_FILE_CLICK_EVENT, {
354
+ detail: this.getSelectedWordsFileName(),
355
+ bubbles: false,
356
+ cancelable: false
357
+ });
358
+
359
+ this.dispatchEvent(evt);
360
+ }
361
+
362
+ _onAssignClick()
363
+ {
364
+ // Fire event to substitute path in configuration
365
+ let evt = new CustomEvent(this.ASSIGN_WORDS_FILE_CLICK_EVENT, {
366
+ detail: this.CONFIG_FOLDER + this.getSelectedWordsFileName(),
367
+ bubbles: false,
368
+ cancelable: false
369
+ });
370
+
371
+ this.dispatchEvent(evt);
372
+ }
373
+
374
+ _onSaveWordFileClick()
375
+ {
376
+ if (this._validator.validate())
377
+ {
378
+ // Show spinner
379
+ $('#editModal #saveSpinner', $(this)).show();
380
+
381
+ // Fire event to request file to be saved by server
382
+ let evt = new CustomEvent(this.SAVE_WORDS_FILE_CLICK_EVENT, {
383
+ detail: {
384
+ filename: $('#editModal #filename', $(this)).val() + this.WORDS_FILE_EXT,
385
+ isNew: this._isNewFile,
386
+ content: $('#editModal #content', $(this)).val()
387
+ },
388
+ bubbles: false,
389
+ cancelable: false
390
+ });
391
+
392
+ this.dispatchEvent(evt);
393
+ }
394
+ }
395
+
396
+ _showModal()
397
+ {
398
+ // Append modal html
399
+ $(this).append(this._modalHtml);
400
+
401
+ let modalElement = $('#editModal', $(this));
402
+
403
+ // Initialize kendo validation
404
+ this._validator = modalElement.find('#editFieldset').kendoValidator({
405
+ validateOnBlur: true,
406
+ rules: {
407
+ requiredFilename: $.proxy(function(input) {
408
+ let valid = true;
409
+ if (input.is('[name=filename]'))
410
+ valid = input.val() !== '';
411
+ return valid;
412
+ }, this),
413
+ validFilename: $.proxy(function(input) {
414
+ let valid = true;
415
+ if (input.is('[name=filename]'))
416
+ valid = (!input.val().includes('\\') && !input.val().includes('/'));
417
+ return valid;
418
+ }, this)
419
+ },
420
+ messages: {
421
+ requiredFilename: 'Required',
422
+ validFilename: 'Contains invalid characters (slash, backslash)'
423
+ }
424
+ }).data('kendoValidator');
425
+
426
+ // Hide save spinner
427
+ $('#saveSpinner', modalElement).hide();
428
+
429
+ // Add listener to Save button click
430
+ $('#saveWordFileButton', modalElement).on('click', $.proxy(this._onSaveWordFileClick, this));
431
+
432
+ // Add listener to modal hide event
433
+ modalElement.on('hidden.bs.modal', $.proxy(this._destroyModal, this));
434
+
435
+ // Initialize bootstrap modal
436
+ modalElement.modal({
437
+ backdrop: 'static',
438
+ keyboard: false,
439
+ });
440
+ }
441
+
442
+ _destroyModal()
443
+ {
444
+ let modalElement = $('#editModal', $(this));
445
+
446
+ if (modalElement)
447
+ {
448
+ // Remove listeners
449
+ $('#saveWordFileButton', modalElement).off('click');
450
+ modalElement.off('hidden.bs.modal');
451
+
452
+ // Destroy everything Kendo
453
+ kendo.destroy(modalElement);
454
+
455
+ // Dispose modal
456
+ modalElement.modal('dispose');
457
+
458
+ // Remove html
459
+ modalElement.remove();
460
+ modalElement = null;
461
+ }
462
+ }
463
+ }
464
+
465
+ // DEFINE COMPONENT
466
+ if (!window.customElements.get('words-files-manager'))
467
+ window.customElements.define('words-files-manager', WordsFilesManager);
468
+
469
+ /* WEBPACK VAR INJECTION */}.call(this, __webpack_require__(/*! jquery */ "jquery")))
470
+
471
+ /***/ }),
472
+
473
+ /***/ "./src/components/sidebar-layout.js":
474
+ /*!******************************************!*\
475
+ !*** ./src/components/sidebar-layout.js ***!
476
+ \******************************************/
477
+ /*! exports provided: SidebarLayout */
478
+ /***/ (function(module, __webpack_exports__, __webpack_require__) {
479
+
480
+ "use strict";
481
+ __webpack_require__.r(__webpack_exports__);
482
+ /* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "SidebarLayout", function() { return SidebarLayout; });
483
+ class SidebarLayout extends HTMLElement
484
+ {
485
+ constructor()
486
+ {
487
+ super();
488
+
489
+ // Attach a shadow root
490
+ const shadowRoot = this.attachShadow({mode: 'open'});
491
+ shadowRoot.innerHTML = `
492
+ <style>
493
+ :host {
494
+ display: flex;
495
+ flex-direction: row;
496
+ }
497
+
498
+ @media (max-width: 575.98px) {
499
+ :host(.split-xs) ::slotted(:not([aria-selected="true"])) {
500
+ display: none !important;
501
+ }
502
+
503
+ :host(.split-xs) ::slotted([aria-selected="true"]) {
504
+ flex-grow: 1;
505
+ }
506
+ }
507
+
508
+ @media (max-width: 767.98px) {
509
+ :host(.split-sm) ::slotted(:not([aria-selected="true"])) {
510
+ display: none !important;
511
+ }
512
+
513
+ :host(.split-sm) ::slotted([aria-selected="true"]) {
514
+ flex-grow: 1;
515
+ }
516
+ }
517
+
518
+ @media (max-width: 991.98px) {
519
+ :host(.split-md) ::slotted(:not([aria-selected="true"])) {
520
+ display: none !important;
521
+ }
522
+
523
+ :host(.split-md) ::slotted([aria-selected="true"]) {
524
+ flex-grow: 1;
525
+ }
526
+ }
527
+
528
+ @media (max-width: 1199.98px) {
529
+ :host(.split-lg) ::slotted(:not([aria-selected="true"])) {
530
+ display: none !important;
531
+ }
532
+
533
+ :host(.split-lg) ::slotted([aria-selected="true"]) {
534
+ flex-grow: 1;
535
+ }
536
+ }
537
+
538
+ .side-col::slotted(*) {
539
+ }
540
+
541
+ .main-col::slotted(*) {
542
+ flex-grow: 1;
543
+ }
544
+ </style>
545
+
546
+ <slot class="side-col" name="side-column"></slot>
547
+ <slot class="main-col" name="main-column"></slot>
548
+ `;
549
+
550
+ // Set initial selection
551
+ this.selectedIndex = 0;
552
+ }
553
+
554
+ get selectedPanel()
555
+ {
556
+ return this._selectedPanel;
557
+ }
558
+
559
+ set selectedPanel(element) // 'side' or 'main'
560
+ {
561
+ if (element != null && element.parentNode == this)
562
+ {
563
+ this._selectedPanel = element;
564
+
565
+ for (let element of this.children)
566
+ {
567
+ if (element == this._selectedPanel)
568
+ element.setAttribute('aria-selected', 'true');
569
+ else
570
+ element.removeAttribute('aria-selected');
571
+ }
572
+ }
573
+ else
574
+ {
575
+ console.error('Element is not a child of SidebarLayout');
576
+ }
577
+ }
578
+
579
+ get selectedIndex()
580
+ {
581
+ return Array.from(this.children).indexOf(this._selectedPanel);
582
+ }
583
+
584
+ set selectedIndex(index)
585
+ {
586
+ if (this.children.length > 0)
587
+ {
588
+ if (this.children[index] == null)
589
+ {
590
+ console.error('Invalid SidebarLayout index');
591
+ return;
592
+ }
593
+
594
+ let element = this.children[index];
595
+ this.selectedPanel = element;
596
+ }
597
+ }
598
+ }
599
+
600
+ // DEFINE COMPONENT
601
+ if (!window.customElements.get('sidebar-layout'))
602
+ window.customElements.define('sidebar-layout', SidebarLayout);
603
+
604
+
605
+ /***/ }),
606
+
607
+ /***/ "./src/modules/zone-configurator.js":
608
+ /*!******************************************!*\
609
+ !*** ./src/modules/zone-configurator.js ***!
610
+ \******************************************/
611
+ /*! exports provided: default */
612
+ /***/ (function(module, __webpack_exports__, __webpack_require__) {
613
+
614
+ "use strict";
615
+ __webpack_require__.r(__webpack_exports__);
616
+ /* WEBPACK VAR INJECTION */(function($) {/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "default", function() { return ZoneConfigurator; });
617
+ /* harmony import */ var _base_module__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./base-module */ "./src/modules/base-module.js");
618
+ /* harmony import */ var _components_view_stack__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../components/view-stack */ "./src/components/view-stack.js");
619
+ /* harmony import */ var _components_sidebar_layout__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ../components/sidebar-layout */ "./src/components/sidebar-layout.js");
620
+ /* harmony import */ var _utils_uibuilder_config_interface_builder__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ../utils/uibuilder/config-interface-builder */ "./src/utils/uibuilder/config-interface-builder.js");
621
+ /* harmony import */ var _utils_utilities__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ../utils/utilities */ "./src/utils/utilities.js");
622
+ /* harmony import */ var _components_module_specific_words_files_manager__WEBPACK_IMPORTED_MODULE_5__ = __webpack_require__(/*! ../components/module-specific/words-files-manager */ "./src/components/module-specific/words-files-manager.js");
623
+
624
+
625
+
626
+
627
+
628
+
629
+
630
+ class ZoneConfigurator extends _base_module__WEBPACK_IMPORTED_MODULE_0__["BaseModule"]
631
+ {
632
+ constructor()
633
+ {
634
+ super('zoneConfig');
635
+
636
+ this.ITEM_TYPE_ZONE = 'zone';
637
+ this.ITEM_TYPE_ROOM = 'room';
638
+
639
+ // Outgoing requests
640
+ this.REQ_GET_ZONES = 'getZones';
641
+
642
+ this.REQ_GET_ZONE_CONFIG = 'getZoneConfig';
643
+ this.REQ_SAVE_ZONE_CONFIG = 'saveZoneConfig';
644
+ this.REQ_NEW_ZONE_CONFIG = 'newZoneConfig';
645
+ this.REQ_DELETE_ZONE_CONFIG = 'delZoneConfig';
646
+ this.REQ_ACTIVATE_ZONE = 'actZone';
647
+
648
+ this.REQ_GET_ROOM_CONFIG = 'getRoomConfig';
649
+ this.REQ_SAVE_ROOM_CONFIG = 'saveRoomConfig';
650
+ this.REQ_NEW_ROOM_CONFIG = 'newRoomConfig';
651
+ this.REQ_DELETE_ROOM_CONFIG = 'delRoomConfig';
652
+
653
+ this.REQ_REFRESH_WORDS_FILE = 'refreshWordsFiles';
654
+ this.REQ_EDIT_WORDS_FILE = 'editWordsFile';
655
+ this.REQ_SAVE_WORDS_FILE = 'saveWordsFile';
656
+ this.REQ_DELETE_WORDS_FILE = 'delWordsFile';
657
+
658
+ // Incoming responses
659
+ this.RESP_ZONES = 'zones';
660
+
661
+ this.RESP_ZONE_CONFIG = 'zoneConfig';
662
+ this.RESP_ZONE_CONFIG_UPDATE_CONFIRM = 'zoneCfgUpd';
663
+ this.RESP_ZONE_ADDED = 'zoneAdded';
664
+ this.RESP_ZONE_REFUSED = 'zoneRefused';
665
+ this.RESP_ZONE_DELETED = 'zoneDel';
666
+ this.RESP_ZONE_ACTIVATED = 'zoneAct';
667
+ this.RESP_ZONE_ACTIVATION_ERROR = 'zoneActErr';
668
+
669
+ this.RESP_ROOM_CONFIG = 'roomConfig';
670
+ this.RESP_ROOM_CONFIG_UPDATE_CONFIRM = 'roomCfgUpd';
671
+ this.RESP_ROOM_ADDED = 'roomAdded';
672
+ this.RESP_ROOM_REFUSED = 'roomRefused';
673
+ this.RESP_ROOM_DELETED = 'roomDel';
674
+
675
+ this.RESP_REFRESH_WORDS_FILES = 'refreshWordsFiles';
676
+ this.RESP_WORDS_FILE_CONTENT = 'wordsFile';
677
+ this.RESP_WORDS_FILE_ERROR = 'wordsFileErr';
678
+ }
679
+
680
+ //------------------------------------
681
+ // COMMON MODULE INTERFACE METHODS
682
+ // This members are used by the main controller
683
+ // to communicate with the module's controller.
684
+ // This methods override those in BaseModule class.
685
+ //------------------------------------
686
+
687
+ initialize(idData, shellController)
688
+ {
689
+ // Call super method
690
+ super.initialize(idData, shellController);
691
+
692
+ // Create interface builder instance
693
+ this._interfaceBuilder = new _utils_uibuilder_config_interface_builder__WEBPACK_IMPORTED_MODULE_3__["ConfigInterfaceBuilder"]();
694
+
695
+ // Set listener for custom actions triggered by configuration interface
696
+ $('#znc-tabNavigator').on('value-set', $.proxy(this._onConfigValueSet, this));
697
+
698
+ // Initialize Zones/Rooms treeview
699
+ this._treeview = $('#znc-treeView').kendoTreeView({
700
+ loadOnDemand: false,
701
+ dataTextField: 'name',
702
+ template: kendo.template('<span class="# if (!item.active) { # inactive-list-item # } #">#: item.name #</span>'),
703
+ change: $.proxy(this._onZoneRoomChange, this),
704
+ }).data('kendoTreeView');
705
+
706
+ // Listen to treeview double-click event
707
+ $('#znc-treeView').on('dblclick', $.proxy(this._onTreeItemDoubleClick, this));
708
+
709
+ // Request zones & rooms list to server instance
710
+ this.sendExtensionRequest(this.REQ_GET_ZONES);
711
+
712
+ // Initialize progress bar
713
+ $('#znc-progressBar').kendoProgressBar({
714
+ min: 0,
715
+ max: 100,
716
+ value: false,
717
+ type: 'value',
718
+ animation: {
719
+ duration: 400
720
+ }
721
+ });
722
+
723
+ // Add listeners to utility buttons
724
+ $('#znc-addZoneButton').on('click', $.proxy(this._onAddZoneClick, this));
725
+ $('#znc-addRoomButton').on('click', $.proxy(this._onAddRoomClick, this));
726
+ $('#znc-editButton').on('click', $.proxy(this._onEditClick, this));
727
+ $('#znc-removeButton').on('click', $.proxy(this._onRemoveClick, this));
728
+ $('#znc-activateButton').on('click', $.proxy(this._onActivateClick, this));
729
+
730
+ // Add listener to interface buttons
731
+ $('#znc-cancelButton').on('click', $.proxy(this._onCancelClick, this));
732
+ $('#znc-reloadButton').on('click', $.proxy(this._onReloadClick, this));
733
+ $('#znc-submitButton').on('click', $.proxy(this._onSubmitClick, this));
734
+
735
+ // Save ref to words files manager and add custom event listeners
736
+ this._wordsFilesManager = document.getElementById('znc-wordsFilesManager');
737
+ $(this._wordsFilesManager).on(this._wordsFilesManager.REFRESH_WORDS_FILES_CLICK_EVENT, $.proxy(this._onWordsFileReloadClick, this));
738
+ $(this._wordsFilesManager).on(this._wordsFilesManager.EDIT_WORDS_FILE_CLICK_EVENT, $.proxy(this._onWordsFileEditClick, this));
739
+ $(this._wordsFilesManager).on(this._wordsFilesManager.SAVE_WORDS_FILE_CLICK_EVENT, $.proxy(this._onWordsFileSaveClick, this));
740
+ $(this._wordsFilesManager).on(this._wordsFilesManager.REMOVE_WORDS_FILE_CLICK_EVENT, $.proxy(this._onWordsFileRemoveClick, this));
741
+ $(this._wordsFilesManager).on(this._wordsFilesManager.ASSIGN_WORDS_FILE_CLICK_EVENT, $.proxy(this._onWordsFileAssignClick, this));
742
+ }
743
+
744
+ destroy()
745
+ {
746
+ // Call super method
747
+ super.destroy();
748
+
749
+ // Remove tree view doubleclick listener
750
+ $('#znc-treeView').off('dblclick');
751
+
752
+ // Remove listener for custom actions triggered by configuration interface
753
+ $('#znc-tabNavigator').off('value-set');
754
+
755
+ // Remove listener for zone/room activation event
756
+ $('#znc-addZoneButton').off('click');
757
+ $('#znc-addRoomButton').off('click');
758
+ $('#znc-editButton').off('click');
759
+ $('#znc-removeButton').off('click');
760
+ $('#znc-activateButton').off('click');
761
+
762
+ // Remove interface buttons click listeners
763
+ $('#znc-cancelButton').off('click');
764
+ $('#znc-reloadButton').off('click');
765
+ $('#znc-submitButton').off('click');
766
+
767
+ // Remove listeners to words files manager
768
+ $(this._wordsFilesManager).off(this._wordsFilesManager.REFRESH_WORDS_FILES_CLICK_EVENT);
769
+ $(this._wordsFilesManager).off(this._wordsFilesManager.EDIT_WORDS_FILE_CLICK_EVENT);
770
+ $(this._wordsFilesManager).off(this._wordsFilesManager.SAVE_WORDS_FILE_CLICK_EVENT);
771
+ $(this._wordsFilesManager).off(this._wordsFilesManager.REMOVE_WORDS_FILE_CLICK_EVENT);
772
+ $(this._wordsFilesManager).off(this._wordsFilesManager.ASSIGN_WORDS_FILE_CLICK_EVENT);
773
+
774
+ // Clear tabs container
775
+ this._clearTabs();
776
+ }
777
+
778
+ onExtensionCommand(command, data)
779
+ {
780
+ const username = data.getUtfString('user');
781
+
782
+ /****** ZONES & ROOMS ******/
783
+
784
+ // Zones & rooms list received
785
+ if (command == this.RESP_ZONES)
786
+ this._populateTree(data);
787
+
788
+ // Zone or room configuration data received
789
+ else if (command == this.RESP_ZONE_CONFIG || command == this.RESP_ROOM_CONFIG)
790
+ {
791
+ // Build user interface based on received data
792
+ this._interfaceBuilder.buildInterface(data.getSFSArray('settings'), 'znc-tabNavigator', false);
793
+
794
+ // Enable scrolling tabs (if needed)
795
+ if (this._reinitTabs)
796
+ {
797
+ $('#znc-tabNavigator #tabs').scrollingTabs({
798
+ bootstrapVersion: 4,
799
+ scrollToTabEdge: true,
800
+ enableSwiping: true,
801
+ disableScrollArrowsOnFullyScrolled: true,
802
+ cssClassLeftArrow: 'fa fa-chevron-left',
803
+ cssClassRightArrow: 'fa fa-chevron-right'
804
+ });
805
+ }
806
+
807
+ // Enable interface
808
+ this._enableConfigInterface(true);
809
+ }
810
+
811
+ /****** ZONES ******/
812
+
813
+ // Zone configuration update confirmation
814
+ else if (command == this.RESP_ZONE_CONFIG_UPDATE_CONFIRM)
815
+ {
816
+ // If a 'name' parameter is received, it means the zone name changed, and we have to update the zones list
817
+ if (data.getUtfString('zName') != null)
818
+ this._updateZoneNameInList(data.getInt('zId'), data.getUtfString('zName'));
819
+
820
+ // If the current user is the updater, show a notification; otherwise, show a dialog box suggesting to reload
821
+ if (username == this.smartFox.mySelf.name)
822
+ {
823
+ // Enable interface
824
+ this._enableConfigInterface(true);
825
+
826
+ // Display notification
827
+ this.shellCtrl.showNotification('Zone modified', `Zone settings updated successfully; changes will be applied on next <strong>server restart</strong>`);
828
+
829
+ // Reset the 'modified' flag
830
+ this._interfaceBuilder.resetIsModified();
831
+ }
832
+ else
833
+ {
834
+ // An alert box is displayed if the user is currently editing the same zone
835
+ if (data.getInt('zId') == this._editedZoneId)
836
+ {
837
+ // Show alert
838
+ this.shellCtrl.showSimpleAlert(`Administrator ${username} has modified the Zone you are currently editing; please reload to update your view.`);
839
+
840
+ // Disable submit button
841
+ $('#znc-submitButton').attr('disabled', true);
842
+ }
843
+ else
844
+ {
845
+ // Display notification
846
+ if (data.getUtfString('zName') != null)
847
+ this.shellCtrl.showNotification('Zone renamed', `Administrator ${username} has changed the name on one of the Zones`);
848
+ }
849
+ }
850
+ }
851
+
852
+ // New zone added
853
+ else if (command == this.RESP_ZONE_ADDED)
854
+ {
855
+ const zoneName = data.getSFSObject('zone').getUtfString('name');
856
+
857
+ // If the current user is the updater, reset the interface; otherwise, just show a notification
858
+ if (username == this.smartFox.mySelf.name)
859
+ {
860
+ // Reset interface
861
+ this._onCancelClick();
862
+
863
+ // Display notification
864
+ this.shellCtrl.showNotification('Zone added', `Zone '${zoneName}' created successfully`);
865
+ }
866
+ else
867
+ {
868
+ // Display notification
869
+ this.shellCtrl.showNotification('Zone added', `Administrator ${username} created Zone '${zoneName}'`);
870
+ }
871
+
872
+ // Add new zone to tree
873
+ let zonesDS = this._treeview.dataSource;
874
+ zonesDS.add(this._createZoneObject(data.getSFSObject('zone')));
875
+ zonesDS.sync();
876
+ }
877
+
878
+ // New zone creation refused due to invalid zone name
879
+ else if (command == this.RESP_ZONE_REFUSED)
880
+ {
881
+ // Re-enable interface
882
+ this._enableConfigInterface(true);
883
+
884
+ // Show warning
885
+ this.shellCtrl.showSimpleAlert('Zone configuration can\'t be saved because another Zone with the same name already exists.', true);
886
+ }
887
+
888
+ // Existing zone deleted
889
+ else if (command == this.RESP_ZONE_DELETED)
890
+ {
891
+ // If the current user is the deleter, reset the interface; otherwise, just show a notification
892
+ if (username == this.smartFox.mySelf.name)
893
+ {
894
+ // Re-enable interface
895
+ this._enableListInterface(true);
896
+
897
+ // Display notification
898
+ this.shellCtrl.showNotification('Zone removed', `Zone '${data.getUtfString('zName')}' deleted successfully`);
899
+ }
900
+ else
901
+ {
902
+ // An alert box is displayed if the user is currently editing the same zone
903
+ if (data.getInt('zId') == this._editedZoneId)
904
+ {
905
+ // Show alert
906
+ this.shellCtrl.showSimpleAlert(`Administrator ${username} has deleted the Zone you are currently modifying; you have to cancel your editing.`);
907
+
908
+ // Disable submit and reload buttons
909
+ $('#znc-reloadButton').attr('disabled', true);
910
+ $('#znc-submitButton').attr('disabled', true);
911
+ }
912
+ else
913
+ {
914
+ // Display notification
915
+ this.shellCtrl.showNotification('Zone removed', `Administrator ${username} deleted Zone '${data.getUtfString('zName')}'`);
916
+ }
917
+ }
918
+
919
+ // Reset selection if the currently selected item or its parent is being removed
920
+ let selectedNode = this._treeview.select();
921
+ let selectedDataItem = this._treeview.dataItem(selectedNode);
922
+ if (selectedDataItem)
923
+ {
924
+ if (selectedDataItem.type == this.ITEM_TYPE_ZONE && selectedDataItem.id == data.getInt('zId'))
925
+ this._deselectTreeItem();
926
+
927
+ if (selectedDataItem.type == this.ITEM_TYPE_ROOM)
928
+ {
929
+ let parentDataItem = this._treeview.dataItem(this._treeview.parent(selectedNode));
930
+
931
+ if (parentDataItem.id == data.getInt('zId'))
932
+ this._deselectTreeItem();
933
+ }
934
+ }
935
+
936
+ // Remove zone from tree
937
+ let dataItem = this._getZoneDataItemById(data.getInt('zId'));
938
+ let zonesDS = this._treeview.dataSource;
939
+ zonesDS.remove(dataItem);
940
+ zonesDS.sync();
941
+ }
942
+
943
+ // Zone activated
944
+ else if (command == this.RESP_ZONE_ACTIVATED)
945
+ {
946
+ // Set zone activation status
947
+ const zoneName = this._setZoneActivationStatus(data.getInt('zId'), data.getUtfString('actRooms'), true);
948
+
949
+ // Display notification
950
+ if (username == this.smartFox.mySelf.name)
951
+ this.shellCtrl.showNotification('Zone activated', `Zone '${zoneName}' activated successfully`);
952
+ else
953
+ this.shellCtrl.showNotification('Zone activated', `Administrator ${username} activated Zone '${zoneName}'`);
954
+ }
955
+
956
+ // Zone activation error
957
+ else if (command == this.RESP_ZONE_ACTIVATION_ERROR)
958
+ {
959
+ // Set zone activation status
960
+ this._setZoneActivationStatus(data.getInt('zId'), '', false);
961
+
962
+ // Show alert
963
+ this.shellCtrl.showSimpleAlert(data.getUtfString('error'), true);
964
+ }
965
+
966
+ /****** ROOMS ******/
967
+
968
+ // Room configuration update confirmation
969
+ else if (command == this.RESP_ROOM_CONFIG_UPDATE_CONFIRM)
970
+ {
971
+ if (data.getUtfString('rName') != null)
972
+ this._updateRoomNameInList(data.getInt('zId'), data.getInt('rId'), data.getUtfString('rName'));
973
+
974
+ // If the current user is the updater, show a notification; otherwise, show a dialog box suggesting to reload
975
+ if (username == this.smartFox.mySelf.name)
976
+ {
977
+ // Enable interface
978
+ this._enableConfigInterface(true);
979
+
980
+ // Display notification
981
+ this.shellCtrl.showNotification('Room modified', `Room settings updated successfully; changes will be applied on next <strong>server restart</strong>`);
982
+
983
+ // Reset the 'modified' flag
984
+ this._interfaceBuilder.resetIsModified();
985
+ }
986
+ else
987
+ {
988
+ // An alert box is displayed if the user is currently editing the same room
989
+ if (data.getInt('rId') == this._editedRoomId)
990
+ {
991
+ // Show alert
992
+ this.shellCtrl.showSimpleAlert(`Administrator ${username} has modified the Room you are currently editing; please reload to update your view.`);
993
+
994
+ // Disable submit button
995
+ $('#znc-submitButton').attr('disabled', true);
996
+ }
997
+ else
998
+ {
999
+ // Display notification
1000
+ if (data.getUtfString('rName') != null)
1001
+ this.shellCtrl.showNotification('Room renamed', `Administrator ${username} has changed the name on one of the Rooms`);
1002
+ }
1003
+ }
1004
+ }
1005
+
1006
+ // New room added
1007
+ else if (command == this.RESP_ROOM_ADDED)
1008
+ {
1009
+ const roomData = data.getSFSObject('room');
1010
+ const zoneId = data.getInt('zId');
1011
+
1012
+ let zonesDS = this._treeview.dataSource;
1013
+ let zoneItem = zonesDS.get(zoneId);
1014
+
1015
+ // If the current user is the updater, reset the interface; otherwise, just show a notification
1016
+ if (username == this.smartFox.mySelf.name)
1017
+ {
1018
+ // Reset interface
1019
+ this._onCancelClick();
1020
+
1021
+ // Display notification
1022
+ this.shellCtrl.showNotification('Room added', `Room '${roomData.getUtfString('name')}' created successfully`);
1023
+ }
1024
+ else
1025
+ {
1026
+ // Display notification
1027
+ this.shellCtrl.showNotification('Room added', `Administrator ${username} created Room '${roomData.getUtfString('name')}' in Zone '${zoneItem.name}'`);
1028
+ }
1029
+
1030
+ // Add new room to tree
1031
+ zoneItem.append(this._createRoomObject(roomData, zoneId));
1032
+ zonesDS.sync();
1033
+
1034
+ // Expand zone node where room was added
1035
+ this._treeview.expand(this._treeview.select());
1036
+ }
1037
+
1038
+ // New room creation refused due to invalid room name
1039
+ else if (command == this.RESP_ROOM_REFUSED)
1040
+ {
1041
+ // Re-enable interface
1042
+ this._enableConfigInterface(true);
1043
+
1044
+ // Show warning
1045
+ this.shellCtrl.showSimpleAlert('Room configuration can\'t be saved because another Room with the same name already exists.', true);
1046
+ }
1047
+
1048
+ // Existing room deleted
1049
+ else if (command == this.RESP_ROOM_DELETED)
1050
+ {
1051
+ let zoneItem = this._getZoneDataItemById(data.getInt('zId'));
1052
+ let roomItem = this._getRoomDataItemById(data.getInt('zId'), data.getInt('rId'));
1053
+
1054
+ // If the current user is the deleter, reset the interface; otherwise, just show a notification
1055
+ if (username == this.smartFox.mySelf.name)
1056
+ {
1057
+ // Re-enable interface
1058
+ this._enableListInterface(true);
1059
+
1060
+ // Display notification
1061
+ this.shellCtrl.showNotification('Room removed', `Room '${roomItem.name}' deleted successfully`);
1062
+ }
1063
+ else
1064
+ {
1065
+ // An alert box is displayed if the user is currently editing the same room
1066
+ if (data.getInt('rId') == this._editedRoomId)
1067
+ {
1068
+ // Show alert
1069
+ this.shellCtrl.showSimpleAlert(`Administrator ${username} has deleted the Room you are currently modifying; you have to cancel your editing.`);
1070
+
1071
+ // Disable submit and reload buttons
1072
+ $('#znc-reloadButton').attr('disabled', true);
1073
+ $('#znc-submitButton').attr('disabled', true);
1074
+ }
1075
+ else
1076
+ {
1077
+ // Display notification
1078
+ this.shellCtrl.showNotification('Room removed', `Administrator ${username} deleted Room '${roomItem.name}' from Zone '${zoneItem.name}'`);
1079
+ }
1080
+ }
1081
+
1082
+ // Reset selection if the currently selected item or its parent is being removed
1083
+ let selectedNode = this._treeview.select();
1084
+ let selectedDataItem = this._treeview.dataItem(selectedNode);
1085
+ if (selectedDataItem)
1086
+ {
1087
+ if (selectedDataItem.type == this.ITEM_TYPE_ROOM && selectedDataItem.id == data.getInt('rId'))
1088
+ this._deselectTreeItem();
1089
+ }
1090
+
1091
+ // Remove room from tree
1092
+ zoneItem.children.remove(roomItem);
1093
+ this._treeview.dataSource.sync();
1094
+ }
1095
+
1096
+ /****** WORDS FILES ******/
1097
+
1098
+ // Words files list received
1099
+ else if (command == this.RESP_REFRESH_WORDS_FILES)
1100
+ {
1101
+ this._wordsFilesManager.refreshWordsFilesList(data.getSFSArray('wf'), username == this.smartFox.mySelf.name);
1102
+
1103
+ // If another user caused a refresh (for example deleting a file, or adding a new one) show a notification
1104
+ if (username != null && username != this.smartFox.mySelf.name && this._editedZoneId > -1)
1105
+ this.shellCtrl.showNotification('Words files modified', `Administrator ${username} has added, modified or deleted a words file.`);
1106
+ }
1107
+
1108
+ // Words file content received
1109
+ else if (command == this.RESP_WORDS_FILE_CONTENT)
1110
+ {
1111
+ this._wordsFilesManager.editWordsFile(data.getUtfString('filename'), data.getText('content'));
1112
+ }
1113
+
1114
+ // Words file error (edit/save)
1115
+ else if (command == this.RESP_WORDS_FILE_ERROR)
1116
+ {
1117
+ // Enable buttons
1118
+ this._wordsFilesManager.enabled = true;
1119
+
1120
+ // Show alert
1121
+ this.shellCtrl.showSimpleAlert(data.getUtfString('error'), true);
1122
+ }
1123
+
1124
+ // else if ()
1125
+ }
1126
+
1127
+ //---------------------------------
1128
+ // UI EVENT LISTENERS
1129
+ //---------------------------------
1130
+
1131
+ _onTreeItemDoubleClick(e)
1132
+ {
1133
+ // Get event target's closest tree node
1134
+ let treeNode = $(e.target).closest('.k-item[role=treeitem]');
1135
+
1136
+ // Get associated data item
1137
+ let dataItem = this._treeview.dataItem(treeNode);
1138
+
1139
+ // Load configuration
1140
+ this._loadConfiguration(dataItem.type);
1141
+ }
1142
+
1143
+ _onZoneRoomChange()
1144
+ {
1145
+ // Reset utility buttons
1146
+ this._setUtilityButtonsState(this._selectedItem);
1147
+ }
1148
+
1149
+ // Utility buttons listeners
1150
+
1151
+ _onAddZoneClick()
1152
+ {
1153
+ // Deselect list item
1154
+ this._deselectTreeItem();
1155
+
1156
+ // Load configuration
1157
+ this._loadConfiguration(this.ITEM_TYPE_ZONE);
1158
+ }
1159
+
1160
+ _onAddRoomClick()
1161
+ {
1162
+ // Select parent list item
1163
+ this._selectParentTreeItem();
1164
+
1165
+ // Load configuration
1166
+ this._loadConfiguration(this.ITEM_TYPE_ROOM);
1167
+ }
1168
+
1169
+ _onEditClick()
1170
+ {
1171
+ // Load configuration
1172
+ this._loadConfiguration(this._selectedItem.type);
1173
+ }
1174
+
1175
+ _onRemoveClick()
1176
+ {
1177
+ this.shellCtrl.showConfirmWarning(`Are you sure you want to delete the selected ${this._selectedItem.type == this.ITEM_TYPE_ZONE ? 'Zone' : 'Room'} configuration?`, $.proxy(this._onRemoveConfirm, this));
1178
+ }
1179
+
1180
+ _onRemoveConfirm()
1181
+ {
1182
+ // Disable zone/room selection list
1183
+ this._enableListInterface(false);
1184
+
1185
+ let params = new SFS2X.SFSObject();
1186
+
1187
+ // Request zone removal
1188
+ if (this._selectedItem.type == this.ITEM_TYPE_ZONE)
1189
+ {
1190
+ params.putInt('zId', this._selectedItem.id);
1191
+ this.sendExtensionRequest(this.REQ_DELETE_ZONE_CONFIG, params);
1192
+ }
1193
+ else
1194
+ {
1195
+ params.putInt('zId', this._selectedItemParent.id);
1196
+ params.putInt('rId', this._selectedItem.id);
1197
+ this.sendExtensionRequest(this.REQ_DELETE_ROOM_CONFIG, params);
1198
+ }
1199
+ }
1200
+
1201
+ _onActivateClick()
1202
+ {
1203
+ // Get selected data item
1204
+ if (this._selectedItem.type == this.ITEM_TYPE_ZONE)
1205
+ {
1206
+ let params = new SFS2X.SFSObject();
1207
+ params.putInt('zId', this._selectedItem.id);
1208
+
1209
+ this.sendExtensionRequest(this.REQ_ACTIVATE_ZONE, params);
1210
+ }
1211
+ }
1212
+
1213
+ // Configuration buttons listeners
1214
+
1215
+ _onCancelClick()
1216
+ {
1217
+ // Enable zone/room selection lists
1218
+ this._enableListInterface(true);
1219
+
1220
+ // Disable configuration interface
1221
+ this._enableConfigInterface(false);
1222
+
1223
+ // Clear main container
1224
+ this._resetTabsContainer(false, true);
1225
+
1226
+ // Set isEditing flag
1227
+ this._isEditing = false;
1228
+ this._editedItemType = '';
1229
+
1230
+ // Switch panel
1231
+ this._switchPanel('znc-sidebarPanel');
1232
+ }
1233
+
1234
+ _onReloadClick()
1235
+ {
1236
+ // Hide validation messages
1237
+ this._interfaceBuilder.resetValidation();
1238
+
1239
+ // Reload configuration
1240
+ this._loadConfiguration(this._editedItemType, false);
1241
+ }
1242
+
1243
+ _onSubmitClick()
1244
+ {
1245
+ // Check validity
1246
+ if (this._interfaceBuilder.checkIsValid())
1247
+ {
1248
+ let changes = this._interfaceBuilder.getChangedData();
1249
+
1250
+ if (changes.size() > 0)
1251
+ {
1252
+ //console.log(changes.getDump())
1253
+ //return;
1254
+
1255
+ // In case the zone/room name changed, check it against the list (duplicate names not allowed!)
1256
+ if (this._validateName(changes))
1257
+ {
1258
+ // Disable configuration interface
1259
+ this._enableConfigInterface(false);
1260
+
1261
+ // Send settings to server instance
1262
+ let params = new SFS2X.SFSObject();
1263
+ params.putSFSArray('settings', changes);
1264
+ params.putBool('backup', $('#znc-backupCheck').prop('checked'));
1265
+ params.putInt('zId', this._editedZoneId);
1266
+ params.putInt('rId', this._editedRoomId);
1267
+
1268
+ if (this._editedItemType == this.ITEM_TYPE_ZONE)
1269
+ {
1270
+ // Submit zone settings
1271
+ if (this._editedZoneId > -1)
1272
+ this.sendExtensionRequest(this.REQ_SAVE_ZONE_CONFIG, params);
1273
+ else
1274
+ this.sendExtensionRequest(this.REQ_NEW_ZONE_CONFIG, params);
1275
+ }
1276
+ else
1277
+ {
1278
+ // Submit room settings
1279
+ if (this._editedRoomId > -1)
1280
+ this.sendExtensionRequest(this.REQ_SAVE_ROOM_CONFIG, params);
1281
+ else
1282
+ this.sendExtensionRequest(this.REQ_NEW_ROOM_CONFIG, params);
1283
+ }
1284
+ }
1285
+ else
1286
+ {
1287
+ // Show alert
1288
+ this.shellCtrl.showSimpleAlert(`Unable to submit configuration because the ${Object(_utils_utilities__WEBPACK_IMPORTED_MODULE_4__["capitalizeFirst"])(this._editedItemType)} name already exists; duplicate names are not allowed.`, true);
1289
+ }
1290
+ }
1291
+ }
1292
+ else
1293
+ {
1294
+ // Show alert
1295
+ this.shellCtrl.showSimpleAlert('Unable to submit configuration changes due to an invalid value; please verify the highlighted form fields in all tabs.', true);
1296
+ }
1297
+ }
1298
+
1299
+ _onConfigValueSet(e) // SAME METHOD DUPLICATED IN zone-monitor.js
1300
+ {
1301
+ const configParam = e.target.data;
1302
+
1303
+ // Handle extension name/type dropdowns update and update the main class dropdown datasource accordingly
1304
+ if (configParam.name == 'extension.name' || configParam.name == 'extension.type' || configParam.name == 'extension.filterClass')
1305
+ {
1306
+ // All involved ConfigFormItems must be available and initialized to proceed
1307
+ const nameFormItem = this._interfaceBuilder.getConfigFormItem('extension.name');
1308
+ const typeFormItem = this._interfaceBuilder.getConfigFormItem('extension.type');
1309
+ const classFormItem = this._interfaceBuilder.getConfigFormItem('extension.file');
1310
+ const filterFormItem = this._interfaceBuilder.getConfigFormItem('extension.filterClass');
1311
+
1312
+ if (nameFormItem != null && typeFormItem != null && classFormItem != null && filterFormItem != null)
1313
+ {
1314
+ const source = nameFormItem.data;
1315
+ let classesList = [];
1316
+
1317
+ let data = source.triggerData;
1318
+ for (let i = 0; i < data.size(); i++)
1319
+ {
1320
+ let ext = data.getSFSObject(i);
1321
+
1322
+ if (ext.getUtfString('name') == nameFormItem.data.value && ext.getUtfString('type') == typeFormItem.data.value)
1323
+ {
1324
+ let classes = ext.get('classesString').split(','); // We don't use "getUTfString" because the type is Text in case a very large list of classes is returned
1325
+
1326
+ if (filterFormItem.data.value == true)
1327
+ {
1328
+ let filteredClasses = classes.filter(_utils_utilities__WEBPACK_IMPORTED_MODULE_4__["filterClassName"]);
1329
+ classes = filteredClasses;
1330
+ }
1331
+
1332
+ classesList = classesList.concat(classes);
1333
+ }
1334
+ }
1335
+
1336
+ let currentClass = classFormItem.data.value;
1337
+
1338
+ // If the classes list doesn't contain the current value, create an empty entry and reset the value
1339
+ if (classesList.indexOf(currentClass) < 0)
1340
+ {
1341
+ if (classesList.length == 0)
1342
+ {
1343
+ classesList.push('');
1344
+ currentClass = '';
1345
+ }
1346
+ else
1347
+ currentClass = classesList[0];
1348
+ }
1349
+
1350
+ let mainClassDropDown = classFormItem._innerWidget;
1351
+ mainClassDropDown.setDataSource(classesList);
1352
+
1353
+ classFormItem.data.value = currentClass;
1354
+ classFormItem._setWidgetValue();
1355
+ }
1356
+ }
1357
+ }
1358
+
1359
+ _onWordsFileReloadClick(evt)
1360
+ {
1361
+ // Send request to server
1362
+ this.sendExtensionRequest(this.REQ_REFRESH_WORDS_FILE);
1363
+ }
1364
+
1365
+ _onWordsFileEditClick(evt)
1366
+ {
1367
+ // Send request to server
1368
+ let params = new SFS2X.SFSObject();
1369
+ params.putUtfString('filename', evt.detail);
1370
+ this.sendExtensionRequest(this.REQ_EDIT_WORDS_FILE, params);
1371
+ }
1372
+
1373
+ _onWordsFileSaveClick(evt)
1374
+ {
1375
+ this._tempWordsFileData = evt.detail;
1376
+
1377
+ // Check if a new file is being created
1378
+ if (this._tempWordsFileData.isNew)
1379
+ {
1380
+ // If yes, check if name already exists
1381
+ if (this._wordsFilesManager.getExistingFilenames().includes(this._tempWordsFileData.filename))
1382
+ {
1383
+ // Show confirm dialog
1384
+ this.shellCtrl.showConfirmWarning('A words file with the entered name already exists; do you want to proceed anyway? The existing file will be overwritten.', $.proxy(this._onWordsFileSaveConfirm, this));
1385
+ return;
1386
+ }
1387
+ }
1388
+
1389
+ // Proceed
1390
+ this._onWordsFileSaveConfirm();
1391
+ }
1392
+
1393
+ _onWordsFileSaveConfirm()
1394
+ {
1395
+ // Disable words files manager buttons
1396
+ this._wordsFilesManager.enabled = false;
1397
+ this._wordsFilesManager.saveSpinnerVisible = true;
1398
+
1399
+ // Send request to server
1400
+ let params = new SFS2X.SFSObject();
1401
+ params.putUtfString('filename', this._tempWordsFileData.filename);
1402
+ params.putText('content', this._tempWordsFileData.content);
1403
+ this.sendExtensionRequest(this.REQ_SAVE_WORDS_FILE, params);
1404
+ }
1405
+
1406
+ _onWordsFileRemoveClick(evt)
1407
+ {
1408
+ this.shellCtrl.showConfirmWarning('Are you sure you want to delete the selected words file?', $.proxy(this._onWordsFileRemoveConfirm, this));
1409
+ }
1410
+
1411
+ _onWordsFileRemoveConfirm()
1412
+ {
1413
+ let wordsFile = this._wordsFilesManager.getSelectedWordsFileName();
1414
+
1415
+ if (wordsFile != null)
1416
+ {
1417
+ // Disable words files manager buttons
1418
+ this._wordsFilesManager.enabled = false;
1419
+ this._wordsFilesManager.actionSpinnerVisible = true;
1420
+
1421
+ // Send request to server
1422
+ let params = new SFS2X.SFSObject();
1423
+ params.putUtfString('filename', wordsFile);
1424
+ this.sendExtensionRequest(this.REQ_DELETE_WORDS_FILE, params);
1425
+ }
1426
+ }
1427
+
1428
+ _onWordsFileAssignClick(evt)
1429
+ {
1430
+ let path = evt.detail;
1431
+
1432
+ // Write path of the selected words file in "wordsFilter.wordsFile" dynamically created field
1433
+ const wordsFileFormItem = this._interfaceBuilder.getConfigFormItem('wordsFilter.wordsFile');
1434
+ wordsFileFormItem.data.value = path;
1435
+ wordsFileFormItem._setWidgetValue();
1436
+ }
1437
+
1438
+ //---------------------------------
1439
+ // PRIVATE METHODS
1440
+ //---------------------------------
1441
+
1442
+ _enableListInterface(enabled)
1443
+ {
1444
+ $('#znc-utilButtons').attr('disabled', !enabled);
1445
+ $('#znc-treeView').attr('disabled', !enabled);
1446
+ }
1447
+
1448
+ _setUtilityButtonsState(dataItem = null)
1449
+ {
1450
+ let disable = true;
1451
+
1452
+ if (dataItem)
1453
+ {
1454
+ // Enable 'activate zone' button if zone is inactive
1455
+ $('#znc-activateButton').attr('disabled', (dataItem.type != this.ITEM_TYPE_ZONE || dataItem.active));
1456
+
1457
+ disable = false;
1458
+ }
1459
+ else
1460
+ {
1461
+ // Disable 'activate zone' button
1462
+ $('#znc-activateButton').attr('disabled', true);
1463
+ }
1464
+
1465
+ // Enable/disable other utility buttons
1466
+ $('#znc-addZoneButton').attr('disabled', false); // Always enabled
1467
+ $('#znc-addRoomButton').attr('disabled', disable);
1468
+ $('#znc-editButton').attr('disabled', disable);
1469
+ $('#znc-removeButton').attr('disabled', disable);
1470
+ }
1471
+
1472
+ _enableConfigInterface(enabled)
1473
+ {
1474
+ $('#znc-cancelButton').attr('disabled', !enabled);
1475
+ $('#znc-reloadButton').attr('disabled', !enabled);
1476
+ $('#znc-submitButton').attr('disabled', !enabled);
1477
+ $('#znc-backupCheck').attr('disabled', !enabled);
1478
+
1479
+ this._interfaceBuilder.disableInterface(!enabled);
1480
+
1481
+ // Also switch view when enabled
1482
+ if (enabled)
1483
+ this._switchView('znc-main');
1484
+ }
1485
+
1486
+ _switchView(viewId)
1487
+ {
1488
+ document.getElementById('znc-viewstack').selectedElement = document.getElementById(viewId);
1489
+ }
1490
+
1491
+ _clearTabs()
1492
+ {
1493
+ // Destroy scrolling tabs
1494
+ $('#znc-tabNavigator #tabs').scrollingTabs('destroy');
1495
+
1496
+ // Remove all tab navigator content
1497
+ this._interfaceBuilder.destroyInterface();
1498
+
1499
+ // Set flag to re-initialize tabs if needed
1500
+ this._reinitTabs = true;
1501
+ }
1502
+
1503
+ _populateTree(data)
1504
+ {
1505
+ let zData = data.getSFSArray('zones');
1506
+
1507
+ let zonesArr = [];
1508
+ for (let z = 0; z < zData.size(); z++)
1509
+ zonesArr.push( this._createZoneObject(zData.getSFSObject(z)) );
1510
+
1511
+ // Create datasource
1512
+ let zones = new kendo.data.HierarchicalDataSource({
1513
+ data: zonesArr,
1514
+ sort: {
1515
+ field: 'name',
1516
+ dir: 'asc'
1517
+ },
1518
+ schema: {
1519
+ model: {
1520
+ id: 'id',
1521
+ children: {
1522
+ schema: {
1523
+ data: 'rooms',
1524
+ sort: {
1525
+ field: 'name',
1526
+ dir: 'asc'
1527
+ }
1528
+ }
1529
+ }
1530
+ }
1531
+ }
1532
+ });
1533
+
1534
+ // Set tree view dataprovider
1535
+ this._treeview.setDataSource(zones);
1536
+
1537
+ // Set utility buttons state (add, remove, edit, etc)
1538
+ this._setUtilityButtonsState();
1539
+ }
1540
+
1541
+ _createZoneObject(zoneData)
1542
+ {
1543
+ let zone = {
1544
+ type: this.ITEM_TYPE_ZONE,
1545
+ name: zoneData.getUtfString('name'),
1546
+ id: zoneData.getInt('id'),
1547
+ active: zoneData.getBool('act')
1548
+ }
1549
+
1550
+ // Create rooms list dataprovider
1551
+ let rData = zoneData.getSFSArray('rooms');
1552
+
1553
+ let roomsArr = [];
1554
+ for (let r = 0; r < rData.size(); r++)
1555
+ roomsArr.push( this._createRoomObject(rData.getSFSObject(r), zoneData.getInt('id')) );
1556
+
1557
+ zone.rooms = roomsArr;
1558
+
1559
+ return zone;
1560
+ }
1561
+
1562
+ _createRoomObject(roomData, zoneId)
1563
+ {
1564
+ let room = {
1565
+ type: this.ITEM_TYPE_ROOM,
1566
+ name: roomData.getUtfString('name'),
1567
+ id: roomData.getInt('id'),
1568
+ active: roomData.getBool('act'),
1569
+ parentId: zoneId,
1570
+ spriteCssClass: this._getRoomListIconCssClass(roomData.getBool('act'))
1571
+ };
1572
+
1573
+ return room;
1574
+ }
1575
+
1576
+ _getRoomListIconCssClass(isActive)
1577
+ {
1578
+ return isActive ? 'fas fa-door-open' : 'fas fa-door-closed inactive-list-item';
1579
+ }
1580
+
1581
+ _setZoneActivationStatus(zoneId, activeRooms, isActive)
1582
+ {
1583
+ let zoneDI = this._getZoneDataItemById(zoneId);
1584
+
1585
+ zoneDI.active = isActive;
1586
+
1587
+ let activeRoomsArr = activeRooms.split(',');
1588
+
1589
+ if (zoneDI.hasChildren)
1590
+ {
1591
+ for (let i = 0; i < zoneDI.children.data().length; i++)
1592
+ {
1593
+ let room = zoneDI.children.data()[i];
1594
+ room.active = (isActive && activeRoomsArr.indexOf(room.name) > -1);
1595
+ room.spriteCssClass = this._getRoomListIconCssClass(room.active)
1596
+ }
1597
+ }
1598
+
1599
+ // Refresh list
1600
+ this._treeview.dataSource.sync();
1601
+
1602
+ // Return zone name
1603
+ return zoneDI.name;
1604
+ }
1605
+
1606
+ _deselectTreeItem()
1607
+ {
1608
+ this._treeview.select($());
1609
+ }
1610
+
1611
+ _selectParentTreeItem()
1612
+ {
1613
+ let selectedNode = this._treeview.select();
1614
+ let selectedDataItem = this._treeview.dataItem(selectedNode);
1615
+
1616
+ if (selectedDataItem.type == this.ITEM_TYPE_ROOM)
1617
+ {
1618
+ let parentNode = this._treeview.parent(selectedNode);
1619
+ this._treeview.select(parentNode);
1620
+ }
1621
+ }
1622
+
1623
+ _loadConfiguration(type, resetTabs = true)
1624
+ {
1625
+ // Disable zone/room selection list
1626
+ this._enableListInterface(false);
1627
+
1628
+ // Disable configuration interface
1629
+ this._enableConfigInterface(false);
1630
+
1631
+ // Clear main container
1632
+ this._resetTabsContainer(true, resetTabs);
1633
+
1634
+ // Set isEditing flag
1635
+ this._isEditing = true;
1636
+ this._editedItemType = type;
1637
+
1638
+ // Request zone or room configuration data to server instance
1639
+ let params = new SFS2X.SFSObject();
1640
+ params.putInt('zId', this._editedZoneId);
1641
+ params.putInt('rId', this._editedRoomId);
1642
+
1643
+ // If no room is selected, then we are editing a zone
1644
+ if (this._editedItemType == this.ITEM_TYPE_ZONE)
1645
+ this.sendExtensionRequest(this.REQ_GET_ZONE_CONFIG, params);
1646
+ else
1647
+ this.sendExtensionRequest(this.REQ_GET_ROOM_CONFIG, params);
1648
+
1649
+ // Switch panel
1650
+ this._switchPanel('znc-mainPanel');
1651
+ }
1652
+
1653
+ _resetTabsContainer(isLoading, resetTabs)
1654
+ {
1655
+ if (resetTabs)
1656
+ this._clearTabs();
1657
+ else
1658
+ this._reinitTabs = false;
1659
+
1660
+ if (!isLoading)
1661
+ this._switchView('znc-select');
1662
+ else
1663
+ this._switchView('znc-loading');
1664
+ }
1665
+
1666
+ _switchPanel(panelId)
1667
+ {
1668
+ document.getElementById('znc-view').selectedPanel = document.getElementById(panelId);
1669
+ }
1670
+
1671
+ _getZoneDataItemById(zoneId)
1672
+ {
1673
+ let zonesDS = this._treeview.dataSource;
1674
+ return zonesDS.get(zoneId);
1675
+ }
1676
+
1677
+ _getRoomDataItemById(zoneId, roomId)
1678
+ {
1679
+ let zoneDI = this._getZoneDataItemById(zoneId);
1680
+
1681
+ if (zoneDI.hasChildren)
1682
+ return zoneDI.children.get(roomId);
1683
+
1684
+ return null;
1685
+ }
1686
+
1687
+ _updateZoneNameInList(zoneId, zoneName)
1688
+ {
1689
+ this._getZoneDataItemById(zoneId).name = zoneName;
1690
+ this._treeview.dataSource.sync();
1691
+ }
1692
+
1693
+ _updateRoomNameInList(zoneId, roomId, roomName)
1694
+ {
1695
+ this._getRoomDataItemById(zoneId, roomId).name = roomName;
1696
+ this._treeview.dataSource.sync();
1697
+ }
1698
+
1699
+ _validateName(changes)
1700
+ {
1701
+ const zoneId = this._editedZoneId;
1702
+
1703
+ for (let i = 0; i < changes.size(); i++)
1704
+ {
1705
+ const setting = changes.getSFSObject(i);
1706
+
1707
+ if (setting.containsKey('name') && setting.getUtfString('name') == 'name')
1708
+ {
1709
+ // Get name value
1710
+ const name = setting.getText('value');
1711
+
1712
+ // Get data source
1713
+ let ds = [];
1714
+
1715
+ if (this._editedItemType == this.ITEM_TYPE_ZONE)
1716
+ ds = this._treeview.dataSource.data();
1717
+ else
1718
+ {
1719
+ if (this._getZoneDataItemById(zoneId).hasChildren)
1720
+ ds = this._getZoneDataItemById(zoneId).children.data();
1721
+ }
1722
+
1723
+
1724
+ // Check if name exists in data source
1725
+ for (let j = 0; j < ds.length; j++)
1726
+ {
1727
+ if (ds[j].name == name)
1728
+ {
1729
+ return false;
1730
+ }
1731
+ }
1732
+
1733
+ break;
1734
+ }
1735
+ }
1736
+
1737
+ return true;
1738
+ }
1739
+
1740
+ //---------------------------------
1741
+ // PRIVATE GETTERS
1742
+ //---------------------------------
1743
+
1744
+ get _selectedItem()
1745
+ {
1746
+ return this._treeview.dataItem(this._treeview.select());
1747
+ }
1748
+
1749
+ get _selectedItemParent()
1750
+ {
1751
+ let selectedNode = this._treeview.select();
1752
+ let parentNode = this._treeview.parent(selectedNode);
1753
+
1754
+ return this._treeview.dataItem(parentNode);
1755
+ }
1756
+
1757
+ get _editedZoneId()
1758
+ {
1759
+ if (this._isEditing && this._selectedItem)
1760
+ {
1761
+ if (this._selectedItem.type == this.ITEM_TYPE_ZONE)
1762
+ return this._selectedItem.id;
1763
+ else
1764
+ return this._selectedItemParent.id;
1765
+ }
1766
+
1767
+ return -1;
1768
+ }
1769
+
1770
+ get _editedRoomId()
1771
+ {
1772
+ if (this._isEditing && this._selectedItem)
1773
+ {
1774
+ if (this._selectedItem.type == this.ITEM_TYPE_ROOM)
1775
+ return this._selectedItem.id;
1776
+ }
1777
+
1778
+ return -1;
1779
+ }
1780
+ }
1781
+
1782
+ /* WEBPACK VAR INJECTION */}.call(this, __webpack_require__(/*! jquery */ "jquery")))
1783
+
1784
+ /***/ })
1785
+
1786
+ }]);
1787
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXNzZXRzL2pzL2NvcmUvbW9kdWxlcy9tb2R1bGUtMTYuYnVuZGxlLmpzIiwic291cmNlcyI6WyJ3ZWJwYWNrOi8vYXBwbGljYXRpb24vLi9zcmMvY29tcG9uZW50cy9tb2R1bGUtc3BlY2lmaWMvd29yZHMtZmlsZXMtbWFuYWdlci5qcyIsIndlYnBhY2s6Ly9hcHBsaWNhdGlvbi8uL3NyYy9jb21wb25lbnRzL3NpZGViYXItbGF5b3V0LmpzIiwid2VicGFjazovL2FwcGxpY2F0aW9uLy4vc3JjL21vZHVsZXMvem9uZS1jb25maWd1cmF0b3IuanMiXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtieXRlc1RvU2l6ZX0gZnJvbSAnLi4vLi4vdXRpbHMvdXRpbGl0aWVzJztcblxuZXhwb3J0IGNsYXNzIFdvcmRzRmlsZXNNYW5hZ2VyIGV4dGVuZHMgSFRNTEVsZW1lbnRcbntcblx0Y29uc3RydWN0b3IoKVxuXHR7XG5cdCAgICBzdXBlcigpO1xuXG5cdFx0dGhpcy5SRUZSRVNIX1dPUkRTX0ZJTEVTX0NMSUNLX0VWRU5UID0gJ3JlZnJlc2hXb3Jkc0ZpbGVzQ2xpY2snO1xuXHRcdHRoaXMuRURJVF9XT1JEU19GSUxFX0NMSUNLX0VWRU5UID0gJ2VkaXRXb3Jkc0ZpbGVDbGljayc7XG5cdFx0dGhpcy5TQVZFX1dPUkRTX0ZJTEVfQ0xJQ0tfRVZFTlQgPSAnc2F2ZVdvcmRzRmlsZUNsaWNrJztcblx0XHR0aGlzLlJFTU9WRV9XT1JEU19GSUxFX0NMSUNLX0VWRU5UID0gJ3JlbW92ZVdvcmRzRmlsZUNsaWNrJztcblx0XHR0aGlzLkFTU0lHTl9XT1JEU19GSUxFX0NMSUNLX0VWRU5UID0gJ2Fzc2luZ1dvcmRzRmlsZUNsaWNrJztcblxuXHRcdHRoaXMuQ09ORklHX0ZPTERFUiA9ICdjb25maWcvJztcblx0XHR0aGlzLldPUkRTX0ZJTEVfRVhUID0gJy53b3Jkcy50eHQnO1xuXG5cdFx0dGhpcy5fbW9kYWxIdG1sID0gYFxuXHRcdFx0PGRpdiBjbGFzcz1cIm1vZGFsXCIgaWQ9XCJlZGl0TW9kYWxcIiB0YWJpbmRleD1cIi0xXCIgcm9sZT1cImRpYWxvZ1wiIGFyaWEtbGFiZWxsZWRieT1cImVkaXRNb2RhbFRpdGxlXCIgYXJpYS1oaWRkZW49XCJ0cnVlXCI+XG5cdFx0XHRcdDxkaXYgY2xhc3M9XCJtb2RhbC1kaWFsb2cgbW9kYWwtZGlhbG9nLWNlbnRlcmVkXCIgcm9sZT1cImRvY3VtZW50XCI+XG5cdFx0XHRcdFx0PGRpdiBjbGFzcz1cIm1vZGFsLWNvbnRlbnRcIj5cblx0XHRcdFx0XHRcdDxkaXYgY2xhc3M9XCJtb2RhbC1oZWFkZXJcIj5cblx0XHRcdFx0XHRcdFx0PGg1IGNsYXNzPVwibW9kYWwtdGl0bGUgdGV4dC1wcmltYXJ5XCIgaWQ9XCJlZGl0TW9kYWxUaXRsZVwiPldvcmQgRmlsZSBFZGl0b3I8L2g1PlxuXHRcdFx0XHRcdFx0XHQ8YnV0dG9uIHR5cGU9XCJidXR0b25cIiBjbGFzcz1cImNsb3NlXCIgZGF0YS1kaXNtaXNzPVwibW9kYWxcIiBhcmlhLWxhYmVsPVwiQ2xvc2VcIj5cblx0XHRcdFx0XHRcdFx0XHQ8c3BhbiBhcmlhLWhpZGRlbj1cInRydWVcIj4mdGltZXM7PC9zcGFuPlxuXHRcdFx0XHRcdFx0XHQ8L2J1dHRvbj5cblx0XHRcdFx0XHRcdDwvZGl2PlxuXHRcdFx0XHRcdFx0PGRpdiBjbGFzcz1cIm1vZGFsLWJvZHkgaW4tZmxvdy1pbnZhbGlkLW1zZ1wiPlxuXHRcdFx0XHRcdFx0XHQ8ZmllbGRzZXQgaWQ9XCJlZGl0RmllbGRzZXRcIj5cblx0XHRcdFx0XHRcdFx0XHQ8ZGl2IGNsYXNzPVwiZm9ybS1ncm91cFwiPlxuXHRcdFx0XHRcdFx0XHRcdFx0PGRpdiBjbGFzcz1cImNvbC1mb3JtLWxhYmVsIGZvcm0tbGFiZWwtY29udGFpbmVyXCI+XG5cdFx0XHRcdFx0XHRcdFx0XHRcdDxsYWJlbCBmb3I9XCJmaWxlbmFtZVwiIGNsYXNzPVwiZm9ybS1sYWJlbFwiPkZpbGVuYW1lIDxpIGNsYXNzPVwiZmFzIGZhLXF1ZXN0aW9uLWNpcmNsZSB0ZXh0LW11dGVkIGhlbHBcIiB0aXRsZT1cIk5hbWUgb2YgdGhlIHdvcmRzIGZpbGUuIFRoZSBmaWxlIHdpbGwgYmUgc2F2ZWQgaW4gc2VydmVyJ3MgPGVtPmNvbmZpZzwvZW0+IGZvbGRlci5cIj48L2k+PC9sYWJlbD5cblx0XHRcdFx0XHRcdFx0XHRcdDwvZGl2PlxuXHRcdFx0XHRcdFx0XHRcdFx0PGRpdiBjbGFzcz1cImlubmVyLXdpZGdldFwiPlxuXHRcdFx0XHRcdFx0XHRcdFx0XHQ8ZGl2IGNsYXNzPVwiaW5wdXQtZ3JvdXBcIj5cblx0XHRcdFx0XHRcdFx0XHRcdFx0PGlucHV0IHR5cGU9XCJ0ZXh0XCIgaWQ9XCJmaWxlbmFtZVwiIG5hbWU9XCJmaWxlbmFtZVwiIGNsYXNzPVwiZm9ybS1jb250cm9sIGstdGV4dGJveFwiIGF1dG9jb21wbGV0ZT1cIm9mZlwiIHJlcXVpcmVkIGRhdGEtcmVxdWlyZWQtbXNnPVwiUmVxdWlyZWRcIiBhcmlhLWRlc2NyaWJlZGJ5PVwiZXh0ZW5zaW9uXCIgLz5cblx0XHRcdFx0XHRcdFx0XHRcdFx0XHQ8ZGl2IGNsYXNzPVwiaW5wdXQtZ3JvdXAtYXBwZW5kXCI+XG5cdFx0XHRcdFx0XHRcdFx0XHRcdFx0XHQ8c3BhbiBjbGFzcz1cImlucHV0LWdyb3VwLXRleHRcIiBpZD1cImV4dGVuc2lvblwiPi53b3Jkcy50eHQ8L3NwYW4+XG5cdFx0XHRcdFx0XHRcdFx0XHRcdFx0PC9kaXY+XG5cdFx0XHRcdFx0XHRcdFx0XHRcdDwvZGl2PlxuXHRcdFx0XHRcdFx0XHRcdFx0XHQ8c3BhbiBjbGFzcz1cImstaW52YWxpZC1tc2cgcG9zaXRpb24tc3RhdGljXCIgZGF0YS1mb3I9XCJmaWxlbmFtZVwiPjwvc3Bhbj5cblx0XHRcdFx0XHRcdFx0XHRcdDwvZGl2PlxuXHRcdFx0XHRcdFx0XHRcdDwvZGl2PlxuXHRcdFx0XHRcdFx0XHRcdDxkaXYgY2xhc3M9XCJmb3JtLWdyb3VwXCI+XG5cdFx0XHRcdFx0XHRcdFx0XHQ8ZGl2IGNsYXNzPVwiY29sLWZvcm0tbGFiZWwgZm9ybS1sYWJlbC1jb250YWluZXJcIj5cblx0XHRcdFx0XHRcdFx0XHRcdFx0PGxhYmVsIGZvcj1cImNvbnRlbnRcIiBjbGFzcz1cImZvcm0tbGFiZWxcIj5Db250ZW50IDxpIGNsYXNzPVwiZmFzIGZhLXF1ZXN0aW9uLWNpcmNsZSB0ZXh0LW11dGVkIGhlbHBcIiB0aXRsZT1cIkVudGVyIGEgd29yZCBvciBhIHZhbGlkIHJlZ3VsYXIgZXhwcmVzc2lvbiBwZXIgbGluZS4gU2VlIGNvbmZpZ3VyYXRpb24ncyA8ZW0+V29yZHMgZmlsZTwvZW0+IGZpZWxkIGRlc2NyaXB0aW9uIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uLlwiPjwvaT48L2xhYmVsPlxuXHRcdFx0XHRcdFx0XHRcdFx0PC9kaXY+XG5cdFx0XHRcdFx0XHRcdFx0XHQ8ZGl2IGNsYXNzPVwiaW5uZXItd2lkZ2V0XCI+XG5cdFx0XHRcdFx0XHRcdFx0XHRcdDx0ZXh0YXJlYSBpZD1cImNvbnRlbnRcIiBuYW1lPVwiY29udGVudFwiIGNsYXNzPVwiZm9ybS1jb250cm9sIGstdGV4dGFyZWEgdy0xMDBcIiByb3dzPVwiMTBcIj48L3RleHRhcmVhPlxuXHRcdFx0XHRcdFx0XHRcdFx0XHQ8c3BhbiBjbGFzcz1cImstaW52YWxpZC1tc2cgcG9zaXRpb24tc3RhdGljXCIgZGF0YS1mb3I9XCJjb250ZW50XCI+PC9zcGFuPlxuXHRcdFx0XHRcdFx0XHRcdFx0PC9kaXY+XG5cdFx0XHRcdFx0XHRcdFx0PC9kaXY+XG5cdFx0XHRcdFx0XHRcdDwvZmllbGRzZXQ+XG5cdFx0XHRcdFx0XHQ8L2Rpdj5cblx0XHRcdFx0XHRcdDxkaXYgY2xhc3M9XCJtb2RhbC1mb290ZXIgZmxleC1jb2x1bW5cIj5cblx0XHRcdFx0XHRcdFx0PGRpdiBjbGFzcz1cImQtZmxleCB3LTEwMFwiPlxuXHRcdFx0XHRcdFx0XHRcdDxkaXYgY2xhc3M9XCJmbGV4LWdyb3ctMSB0ZXh0LWxlZnRcIj5cblx0XHRcdFx0XHRcdFx0XHRcdDxidXR0b24gaWQ9XCJzYXZlV29yZEZpbGVCdXR0b25cIiB0eXBlPVwiYnV0dG9uXCIgY2xhc3M9XCJrLWJ1dHRvbiBrLXByaW1hcnlcIj48aSBjbGFzcz1cImZhcyBmYS1zYXZlIG1yLTFcIj48L2k+U2F2ZSB3b3JkIGZpbGU8L2J1dHRvbj5cblx0XHRcdFx0XHRcdFx0XHRcdDxpIGlkPVwic2F2ZVNwaW5uZXJcIiBjbGFzcz1cImZhcyBmYS1jaXJjbGUtbm90Y2ggZmEtc3BpbiB0ZXh0LXByaW1hcnkgYWxpZ24tbWlkZGxlIG1sLTFcIj48L2k+XG5cdFx0XHRcdFx0XHRcdFx0PC9kaXY+XG5cdFx0XHRcdFx0XHRcdFx0PGRpdiBjbGFzcz1cImZsZXgtZ3Jvdy0xIHRleHQtcmlnaHRcIj5cblx0XHRcdFx0XHRcdFx0XHRcdDxidXR0b24gdHlwZT1cImJ1dHRvblwiIGNsYXNzPVwiay1idXR0b24gay1zZWNvbmRhcnlcIiBkYXRhLWRpc21pc3M9XCJtb2RhbFwiPkNhbmNlbDwvYnV0dG9uPlxuXHRcdFx0XHRcdFx0XHRcdDwvZGl2PlxuXHRcdFx0XHRcdFx0XHQ8L2Rpdj5cblx0XHRcdFx0XHRcdDwvZGl2PlxuXHRcdFx0XHRcdDwvZGl2PlxuXHRcdFx0XHQ8L2Rpdj5cblx0XHRcdDwvZGl2PlxuXHRcdGA7XG5cblx0XHQvLy0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS1cblxuXHRcdCQodGhpcykuYXBwZW5kKGBcblx0XHRcdDxkaXYgY2xhc3M9XCJjb2wtc20tNSBjb2wtbGctNCBjb2wtZm9ybS1sYWJlbCBmb3JtLWxhYmVsLWNvbnRhaW5lclwiPlxuXHRcdFx0XHQ8bGFiZWwgY2xhc3M9XCJmb3JtLWxhYmVsXCI+QXZhaWxhYmxlIHdvcmRzIGZpbGVzIDxpIGNsYXNzPVwiZmFzIGZhLXF1ZXN0aW9uLWNpcmNsZSB0ZXh0LW11dGVkIGhlbHBcIiB0aXRsZT1cIlRoZSBsaXN0IG9mIHdvcmRzIGZpbGVzIGZvdW5kIGluIHNlcnZlcidzIDxlbT5jb25maWc8L2VtPiBmb2xkZXIuIENsaWNrIG9uIHRoZSBBc3NpZ24gYnV0dG9uIHRvIHNldCB0aGUgPHN0cm9uZz5Xb3JkcyBmaWxlPC9zdHJvbmc+IGZpZWxkIHRvIHRoZSBzZWxlY3RlZCBmaWxlLiBDb25maWd1cmF0aW9uIHN1Ym1pc3Npb24gaXMgdGhlbiByZXF1aXJlZCB0byBzYXZlIHRoZSBuZXcgdmFsdWUuXCI+PC9pPjwvbGFiZWw+XG5cdFx0XHQ8L2Rpdj5cblx0XHRcdDxkaXYgY2xhc3M9XCJpbm5lci13aWRnZXQgYWxpZ24tc2VsZi1jZW50ZXIgY29sLXNtXCI+XG5cdFx0XHRcdDxkaXY+XG5cdFx0XHRcdFx0PGRpdiBpZD1cIndvcmRzRmlsZXNcIiBjbGFzcz1cImxpbWl0ZWQtaGVpZ2h0XCI+PC9kaXY+XG5cdFx0XHRcdFx0PGRpdiBpZD1cImFjdGlvbkJ1dHRvbnNcIiBjbGFzcz1cIm10LTIgdGV4dC1yaWdodFwiIGRpc2FibGVkPlxuXHRcdFx0XHRcdFx0PGkgaWQ9XCJhY3Rpb25TcGlubmVyXCIgY2xhc3M9XCJmYXMgZmEtY2lyY2xlLW5vdGNoIGZhLXNwaW4gdGV4dC1wcmltYXJ5IGFsaWduLW1pZGRsZVwiPjwvaT5cblx0XHRcdFx0XHRcdDxidXR0b24gaWQ9XCJyZWZyZXNoQnV0dG9uXCIgdHlwZT1cImJ1dHRvblwiIGNsYXNzPVwiay1idXR0b24gay1zZWNvbmRhcnkgbWwtMlwiIHRpdGxlPVwiUmVmcmVzaFwiPjxpIGNsYXNzPVwiZmFzIGZhLXJlZG8tYWx0XCI+PC9pPjwvYnV0dG9uPlxuXHRcdFx0XHRcdFx0PGJ1dHRvbiBpZD1cImFkZEJ1dHRvblwiIHR5cGU9XCJidXR0b25cIiBjbGFzcz1cImstYnV0dG9uIGstc2Vjb25kYXJ5IG1sLTJcIiB0aXRsZT1cIkFkZFwiPjxpIGNsYXNzPVwiZmFzIGZhLXBsdXNcIj48L2k+PC9idXR0b24+XG5cdFx0XHRcdFx0XHQ8YnV0dG9uIGlkPVwiZWRpdEJ1dHRvblwiIHR5cGU9XCJidXR0b25cIiBjbGFzcz1cImstYnV0dG9uIGstc2Vjb25kYXJ5IG1sLTJcIiB0aXRsZT1cIkVkaXRcIiBkaXNhYmxlZD48aSBjbGFzcz1cImZhcyBmYS1wZW5cIj48L2k+PC9idXR0b24+XG5cdFx0XHRcdFx0XHQ8YnV0dG9uIGlkPVwicmVtb3ZlQnV0dG9uXCIgdHlwZT1cImJ1dHRvblwiIGNsYXNzPVwiay1idXR0b24gay1zZWNvbmRhcnkgbWwtMlwiIHRpdGxlPVwiUmVtb3ZlXCIgZGlzYWJsZWQ+PGkgY2xhc3M9XCJmYXMgZmEtdGltZXNcIj48L2k+PC9idXR0b24+XG5cdFx0XHRcdFx0XHQ8YnV0dG9uIGlkPVwiYXNzaWduQnV0dG9uXCIgdHlwZT1cImJ1dHRvblwiIGNsYXNzPVwiay1idXR0b24gay1zZWNvbmRhcnkgbWwtMlwiIHRpdGxlPVwiQXNzaWduXCIgZGlzYWJsZWQ+PGkgY2xhc3M9XCJmYXMgZmEtY29tcHJlc3MtYXJyb3dzLWFsdFwiPjwvaT48L2J1dHRvbj5cblx0XHRcdFx0XHQ8L2Rpdj5cblx0XHRcdFx0PC9kaXY+XG5cdFx0XHQ8L2Rpdj5cblx0XHRgKTtcblxuXHRcdC8vLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLVxuXG5cdFx0Ly8gSW5pdGlhbGl6ZSBncmlkXG5cdFx0dGhpcy5fd29yZHNGaWxlc0dyaWQgPSAkKCcjd29yZHNGaWxlcycsICQodGhpcykpLmtlbmRvR3JpZCh7XG5cdFx0XHRyZXNpemFibGU6IHRydWUsXG5cdFx0XHRzZWxlY3RhYmxlOiAncm93Jyxcblx0XHRcdGNoYW5nZTogJC5wcm94eSh0aGlzLl9vbldvcmRzRmlsZXNHcmlkU2VsZWN0aW9uQ2hhbmdlLCB0aGlzKSxcblx0XHRcdGNvbHVtbnM6IFtcblx0XHRcdFx0e1xuXHRcdFx0XHRcdGZpZWxkOiAnbmFtZScsXG5cdFx0XHRcdFx0dGl0bGU6ICdGaWxlbmFtZScsXG5cdFx0XHRcdFx0d2lkdGg6IDEyMFxuXHRcdFx0XHR9LFxuXHRcdFx0XHR7XG5cdFx0XHRcdFx0ZmllbGQ6ICdkYXRlJyxcblx0XHRcdFx0XHR0aXRsZTogJ0RhdGUnLFxuXHRcdFx0XHRcdHdpZHRoOiA4MFxuXHRcdFx0XHR9LFxuXHRcdFx0XHR7XG5cdFx0XHRcdFx0ZmllbGQ6ICdzaXplJyxcblx0XHRcdFx0XHR0aXRsZTogJ1NpemUnLFxuXHRcdFx0XHRcdHdpZHRoOiA4MFxuXHRcdFx0XHR9XG5cdFx0XHRdLFxuXHRcdFx0bm9SZWNvcmRzOiB7XG5cdFx0XHRcdHRlbXBsYXRlOiAnTm8gZmlsZXMuJ1xuXHRcdFx0fVxuXHRcdH0pLmRhdGEoJ2tlbmRvR3JpZCcpO1xuXG5cdFx0Ly8gQWRkIGxpc3RlbmVycyB0byBidXR0b24gY2xpY2tzXG5cdFx0JCgnI3JlZnJlc2hCdXR0b24nLCAkKHRoaXMpKS5vbignY2xpY2snLCAkLnByb3h5KHRoaXMuX29uUmVsb2FkQ2xpY2ssIHRoaXMpKTtcblx0XHQkKCcjYWRkQnV0dG9uJywgJCh0aGlzKSkub24oJ2NsaWNrJywgJC5wcm94eSh0aGlzLl9vbkFkZENsaWNrLCB0aGlzKSk7XG5cdFx0JCgnI2VkaXRCdXR0b24nLCAkKHRoaXMpKS5vbignY2xpY2snLCAkLnByb3h5KHRoaXMuX29uRWRpdENsaWNrLCB0aGlzKSk7XG5cdFx0JCgnI3JlbW92ZUJ1dHRvbicsICQodGhpcykpLm9uKCdjbGljaycsICQucHJveHkodGhpcy5fb25SZW1vdmVDbGljaywgdGhpcykpO1xuXHRcdCQoJyNhc3NpZ25CdXR0b24nLCAkKHRoaXMpKS5vbignY2xpY2snLCAkLnByb3h5KHRoaXMuX29uQXNzaWduQ2xpY2ssIHRoaXMpKTtcblx0fVxuXG5cdGRlc3Ryb3koKVxuXHR7XG5cdFx0Ly8gRGVzdHJveSBncmlkXG5cdFx0dGhpcy5fd29yZHNGaWxlc0dyaWQuZGVzdHJveSgpO1xuXG5cdFx0Ly8gUmVtb3ZlIGV2ZW50IGxpc3RlbmVyc1xuXHRcdCQoJyNyZWZyZXNoQnV0dG9uJywgJCh0aGlzKSkub2ZmKCdjbGljaycpO1xuXHRcdCQoJyNhZGRCdXR0b24nLCAkKHRoaXMpKS5vZmYoJ2NsaWNrJyk7XG5cdFx0JCgnI2VkaXRCdXR0b24nLCAkKHRoaXMpKS5vZmYoJ2NsaWNrJyk7XG5cdFx0JCgnI3JlbW92ZUJ1dHRvbicsICQodGhpcykpLm9mZignY2xpY2snKTtcblx0XHQkKCcjYXNzaWduQnV0dG9uJywgJCh0aGlzKSkub2ZmKCdjbGljaycpO1xuXG5cdFx0Ly8gSGlkZSBtb2RhbCAod2hpY2ggaW4gdHVybiBkZXN0cm95cyBpdClcblx0XHRsZXQgbW9kYWxFbGVtZW50ID0gJCgnI2VkaXRNb2RhbCcsICQodGhpcykpO1xuXG5cdFx0aWYgKG1vZGFsRWxlbWVudClcblx0XHRcdG1vZGFsRWxlbWVudC5tb2RhbCgnaGlkZScpO1xuXHR9XG5cblx0Z2V0IGVuYWJsZWQoKVxuXHR7XG5cdFx0cmV0dXJuIHRoaXMuX2lzRW5hYmxlZDtcblx0fVxuXG5cdHNldCBlbmFibGVkKHZhbHVlKVxuXHR7XG5cdFx0dGhpcy5faXNFbmFibGVkID0gdmFsdWU7XG5cblx0XHQvLyBFbmFibGUvZGlzYWJsZSBidXR0b25zXG5cdFx0JCgnI2FjdGlvbkJ1dHRvbnMnLCB0aGlzKS5hdHRyKCdkaXNhYmxlZCcsICF2YWx1ZSk7XG5cblx0XHQvLyBIaWRlIHNwaW5uZXJcblx0XHRpZiAodmFsdWUpXG5cdFx0XHR0aGlzLmFjdGlvblNwaW5uZXJWaXNpYmxlID0gZmFsc2U7XG5cblx0XHQvLyBFbmFibGUvZGlzYWJsZSBtb2RhbFxuXHRcdGxldCBtb2RhbEVsZW1lbnQgPSAkKCcjZWRpdE1vZGFsJywgJCh0aGlzKSk7XG5cblx0XHRpZiAobW9kYWxFbGVtZW50KVxuXHRcdHtcblx0XHRcdC8vIERpc2FibGUgbW9kYWwgY2xvc2UgYnV0dG9uc1xuXHRcdFx0JCgnYnV0dG9uW2RhdGEtZGlzbWlzcz1cIm1vZGFsXCJdJywgbW9kYWxFbGVtZW50KS5hdHRyKCdkaXNhYmxlZCcsICF2YWx1ZSk7XG5cblx0XHRcdC8vIERpc2FibGUgc2F2ZSBidXR0b25cblx0XHRcdCQoJyNzYXZlV29yZEZpbGVCdXR0b24nLCBtb2RhbEVsZW1lbnQpLmF0dHIoJ2Rpc2FibGVkJywgIXZhbHVlKTtcblxuXHRcdFx0Ly8gRGlzYWJsZSBmaWVsZHNldFxuXHRcdFx0JCgnI2VkaXRGaWVsZHNldCcsIG1vZGFsRWxlbWVudCkuYXR0cignZGlzYWJsZWQnLCAhdmFsdWUpO1xuXG5cdFx0XHQvLyBIaWRlIHNwaW5uZXJcblx0XHRcdGlmICh2YWx1ZSlcblx0XHRcdFx0dGhpcy5zYXZlU3Bpbm5lclZpc2libGUgPSBmYWxzZTtcblx0XHR9XG5cdH1cblxuXHRzZXQgYWN0aW9uU3Bpbm5lclZpc2libGUodmFsdWUpXG5cdHtcblx0XHRpZiAodmFsdWUpXG5cdFx0XHQkKCcjYWN0aW9uU3Bpbm5lcicsICQodGhpcykpLnNob3coKTtcblx0XHRlbHNlXG5cdFx0XHQkKCcjYWN0aW9uU3Bpbm5lcicsICQodGhpcykpLmhpZGUoKTtcblx0fVxuXG5cdHNldCBzYXZlU3Bpbm5lclZpc2libGUodmFsdWUpXG5cdHtcblx0XHRsZXQgbW9kYWxFbGVtZW50ID0gJCgnI2VkaXRNb2RhbCcsICQodGhpcykpO1xuXG5cdFx0aWYgKG1vZGFsRWxlbWVudClcblx0XHR7XG5cdFx0XHRpZiAodmFsdWUpXG5cdFx0XHRcdCQoJyNzYXZlU3Bpbm5lcicsIG1vZGFsRWxlbWVudCkuc2hvdygpO1xuXHRcdFx0ZWxzZVxuXHRcdFx0XHQkKCcjc2F2ZVNwaW5uZXInLCBtb2RhbEVsZW1lbnQpLmhpZGUoKTtcblx0XHR9XG5cdH1cblxuXHRyZWZyZXNoV29yZHNGaWxlc0xpc3Qod29yZHNGaWxlc0xpc3QsIGhpZGVFZGl0TW9kYWwpXG5cdHtcblx0XHRpZiAoaGlkZUVkaXRNb2RhbClcblx0XHR7XG5cdFx0XHRsZXQgbW9kYWxFbGVtZW50ID0gJCgnI2VkaXRNb2RhbCcsICQodGhpcykpO1xuXG5cdFx0XHRpZiAobW9kYWxFbGVtZW50KVxuXHRcdFx0XHRtb2RhbEVsZW1lbnQubW9kYWwoJ2hpZGUnKTtcblx0XHR9XG5cblx0XHRsZXQgZmlsZXMgPSBbXTtcblx0XHR0aGlzLl9leGlzdGluZ0ZpbGVuYW1lcyA9IFtdO1xuXG5cdFx0Zm9yIChsZXQgZiA9IDA7IGYgPCB3b3Jkc0ZpbGVzTGlzdC5zaXplKCk7IGYrKylcblx0XHR7XG5cdFx0XHRjb25zdCBmaWxlID0gd29yZHNGaWxlc0xpc3QuZ2V0U0ZTT2JqZWN0KGYpO1xuXG5cdFx0XHRjb25zdCBmaWxlT2JqID0ge307XG5cdFx0XHRmaWxlT2JqLm5hbWUgPSBmaWxlLmdldFV0ZlN0cmluZygnbmFtZScpO1xuXHRcdFx0ZmlsZU9iai5kYXRlID0gZmlsZS5nZXRVdGZTdHJpbmcoJ2RhdGUnKSArICcgJyArIGZpbGUuZ2V0VXRmU3RyaW5nKCd0aW1lJyk7XG5cdFx0XHRmaWxlT2JqLnNpemUgPSBieXRlc1RvU2l6ZShmaWxlLmdldExvbmcoJ3NpemUnKSwgMik7XG5cblx0XHRcdC8vIFBvcHVsYXRlIGZpbGVzIGxpc3Rcblx0XHRcdGZpbGVzLnB1c2goZmlsZU9iaik7XG5cblx0XHRcdC8vIFNhdmUgcmVmIHRvIGV4aXN0aW5nIGZpbGVuYW1lcywgdG8gY2hlY2sgdGhlbSB3aGVuIGEgbmV3IGZpbGUgaXMgY3JlYXRlZFxuXHRcdFx0dGhpcy5fZXhpc3RpbmdGaWxlbmFtZXMucHVzaChmaWxlT2JqLm5hbWUpO1xuXHRcdH1cblxuXHRcdC8vIEFzc2lnbiBkYXRhIHNvdXJjZSB0byBncmlkXG5cdFx0dGhpcy5fc2V0V29yZHNGaWxlc0dyaWREYXRhU291cmNlKGZpbGVzKTtcblx0XHR0aGlzLl9vbldvcmRzRmlsZXNHcmlkU2VsZWN0aW9uQ2hhbmdlKCk7XG5cblx0XHQvLyBFbmFibGVcblx0XHR0aGlzLmVuYWJsZWQgPSB0cnVlO1xuXHR9XG5cblx0Z2V0U2VsZWN0ZWRXb3Jkc0ZpbGVOYW1lKClcblx0e1xuXHRcdGlmICh0aGlzLl93b3Jkc0ZpbGVzR3JpZC5zZWxlY3QoKSAhPSBudWxsKVxuXHRcdHtcblx0XHRcdGxldCBzZWxlY3RlZEluZGV4ID0gdGhpcy5fd29yZHNGaWxlc0dyaWQuc2VsZWN0KCkuaW5kZXgoKTtcblx0XHRcdHJldHVybiB0aGlzLl93b3Jkc0ZpbGVzR3JpZC5kYXRhU291cmNlLmF0KHNlbGVjdGVkSW5kZXgpLm5hbWU7XG5cdFx0fVxuXHRcdGVsc2Vcblx0XHRcdHJldHVybiBudWxsO1xuXHR9XG5cblx0ZWRpdFdvcmRzRmlsZShmaWxlbmFtZSwgY29udGVudClcblx0e1xuXHRcdHRoaXMuX2lzTmV3RmlsZSA9IGZhbHNlO1xuXG5cdFx0dGhpcy5lbmFibGVkID0gdHJ1ZTtcblxuXHRcdC8vIFNob3cgbW9kYWxcblx0XHR0aGlzLl9zaG93TW9kYWwoKTtcblxuXHRcdC8vIFJlbW92ZSBkZWZhdWx0IGV4dGVuc2lvbiBmcm9tIGZpbGVuYW1lXG5cdFx0ZmlsZW5hbWUgPSBmaWxlbmFtZS5zdWJzdHJpbmcoMCwgZmlsZW5hbWUubGFzdEluZGV4T2YodGhpcy5XT1JEU19GSUxFX0VYVCkpO1xuXG5cdFx0Ly8gRW50ZXIgY29udGVudCBmaWxlbmFtZSBhbmQgY29udGVudCBpbiBtb2RhbCBmb3JtXG5cdFx0JCgnI2VkaXRNb2RhbCAjZmlsZW5hbWUnLCAkKHRoaXMpKS52YWwoZmlsZW5hbWUpO1xuXHRcdCQoJyNlZGl0TW9kYWwgI2NvbnRlbnQnLCAkKHRoaXMpKS52YWwoY29udGVudCk7XG5cblx0XHQvLyBTZXQgZmlsZW5hbWUgZmllbGQgYXMgbm90IGVkaXRhYmxlIGFuZCBoaWRlIG5vdGVcblx0XHQkKCcjZWRpdE1vZGFsICNmaWxlbmFtZScsICQodGhpcykpLmF0dHIoJ2Rpc2FibGVkJywgdHJ1ZSk7XG5cdFx0JCgnI2VkaXRNb2RhbCAjZmlsZW5hbWVOb3RlJywgJCh0aGlzKSkuaGlkZSgpO1xuXHR9XG5cblx0Z2V0RXhpc3RpbmdGaWxlbmFtZXMoKVxuXHR7XG5cdFx0cmV0dXJuIHRoaXMuX2V4aXN0aW5nRmlsZW5hbWVzO1xuXHR9XG5cblx0X3NldFdvcmRzRmlsZXNHcmlkRGF0YVNvdXJjZShkcylcblx0e1xuXHRcdC8vIFJlYWQgY3VycmVudCBob3Jpem9udGFsIHNjcm9sbCB2YWx1ZVxuXHQgICBjb25zdCBzY3JvbGxMZWZ0ID0gJCgnI3dvcmRzRmlsZXMgLmstZ3JpZC1jb250ZW50JywgdGhpcy5fd29yZHNGaWxlc0dyaWQud3JhcHBlcikuc2Nyb2xsTGVmdCgpO1xuXG5cdCAgIC8vIEFzc2lnbiBkYXRhIHNvdXJjZSB0byBncmlkXG5cdCAgIHRoaXMuX3dvcmRzRmlsZXNHcmlkLnNldERhdGFTb3VyY2UoZHMpO1xuXG5cdCAgIC8vIFNldCBob3Jpem9udGFsIHNjcm9sbFxuXHQgICAkKCcjd29yZHNGaWxlcyAuay1ncmlkLWNvbnRlbnQnLCB0aGlzLl93b3Jkc0ZpbGVzR3JpZC53cmFwcGVyKS5zY3JvbGxMZWZ0KHNjcm9sbExlZnQpO1xuXHR9XG5cblx0X29uV29yZHNGaWxlc0dyaWRTZWxlY3Rpb25DaGFuZ2UoKVxuXHR7XG5cdFx0Ly8gRW5hYmxlL2Rpc2FibGUgYnV0dG9uc1xuXHRcdGNvbnN0IHNlbGVjdGVkUm93cyA9IHRoaXMuX3dvcmRzRmlsZXNHcmlkLnNlbGVjdCgpO1xuXHRcdCQoJyNlZGl0QnV0dG9uJykuYXR0cignZGlzYWJsZWQnLCBzZWxlY3RlZFJvd3MubGVuZ3RoID09IDApO1xuXHRcdCQoJyNyZW1vdmVCdXR0b24nKS5hdHRyKCdkaXNhYmxlZCcsIHNlbGVjdGVkUm93cy5sZW5ndGggPT0gMCk7XG5cdFx0JCgnI2Fzc2lnbkJ1dHRvbicpLmF0dHIoJ2Rpc2FibGVkJywgc2VsZWN0ZWRSb3dzLmxlbmd0aCA9PSAwKTtcblx0fVxuXG5cdF9vblJlbG9hZENsaWNrKClcblx0e1xuXHRcdC8vIEZpcmUgZXZlbnQgdG8gcmVxdWVzdCBmaWxlIGNvbnRlbnQgdG8gc2VydmVyXG5cdFx0bGV0IGV2dCA9IG5ldyBDdXN0b21FdmVudCh0aGlzLlJFRlJFU0hfV09SRFNfRklMRVNfQ0xJQ0tfRVZFTlQsIHtcblx0ICAgIFx0ZGV0YWlsOiBudWxsLFxuXHRcdFx0YnViYmxlczogZmFsc2UsXG5cdFx0XHRjYW5jZWxhYmxlOiBmYWxzZVxuXHRcdH0pO1xuXG5cdFx0dGhpcy5kaXNwYXRjaEV2ZW50KGV2dCk7XG5cdH1cblxuXHRfb25BZGRDbGljaygpXG5cdHtcblx0XHR0aGlzLl9pc05ld0ZpbGUgPSB0cnVlO1xuXG5cdFx0Ly8gU2hvdyBtb2RhbFxuXHRcdHRoaXMuX3Nob3dNb2RhbCgpO1xuXHR9XG5cblx0X29uRWRpdENsaWNrKClcblx0e1xuXHRcdC8vIERpc2FibGUgYnV0dG9uc1xuXHRcdHRoaXMuZW5hYmxlZCA9IGZhbHNlO1xuXHRcdHRoaXMuYWN0aW9uU3Bpbm5lclZpc2libGUgPSB0cnVlO1xuXG5cdFx0Ly8gRmlyZSBldmVudCB0byByZXF1ZXN0IGZpbGUgY29udGVudCB0byBzZXJ2ZXJcblx0XHRsZXQgZXZ0ID0gbmV3IEN1c3RvbUV2ZW50KHRoaXMuRURJVF9XT1JEU19GSUxFX0NMSUNLX0VWRU5ULCB7XG5cdCAgICBcdGRldGFpbDogdGhpcy5nZXRTZWxlY3RlZFdvcmRzRmlsZU5hbWUoKSxcblx0XHRcdGJ1YmJsZXM6IGZhbHNlLFxuXHRcdFx0Y2FuY2VsYWJsZTogZmFsc2Vcblx0XHR9KTtcblxuXHRcdHRoaXMuZGlzcGF0Y2hFdmVudChldnQpO1xuXHR9XG5cblx0X29uUmVtb3ZlQ2xpY2soKVxuXHR7XG5cdFx0Ly8gRmlyZSBldmVudCB0byByZXF1ZXN0IGZpbGUgcmVtb3ZhbCB0byBzZXJ2ZXJcblx0XHRsZXQgZXZ0ID0gbmV3IEN1c3RvbUV2ZW50KHRoaXMuUkVNT1ZFX1dPUkRTX0ZJTEVfQ0xJQ0tfRVZFTlQsIHtcblx0ICAgIFx0ZGV0YWlsOiB0aGlzLmdldFNlbGVjdGVkV29yZHNGaWxlTmFtZSgpLFxuXHRcdFx0YnViYmxlczogZmFsc2UsXG5cdFx0XHRjYW5jZWxhYmxlOiBmYWxzZVxuXHRcdH0pO1xuXG5cdFx0dGhpcy5kaXNwYXRjaEV2ZW50KGV2dCk7XG5cdH1cblxuXHRfb25Bc3NpZ25DbGljaygpXG5cdHtcblx0XHQvLyBGaXJlIGV2ZW50IHRvIHN1YnN0aXR1dGUgcGF0aCBpbiBjb25maWd1cmF0aW9uXG5cdFx0bGV0IGV2dCA9IG5ldyBDdXN0b21FdmVudCh0aGlzLkFTU0lHTl9XT1JEU19GSUxFX0NMSUNLX0VWRU5ULCB7XG5cdCAgICBcdGRldGFpbDogdGhpcy5DT05GSUdfRk9MREVSICsgdGhpcy5nZXRTZWxlY3RlZFdvcmRzRmlsZU5hbWUoKSxcblx0XHRcdGJ1YmJsZXM6IGZhbHNlLFxuXHRcdFx0Y2FuY2VsYWJsZTogZmFsc2Vcblx0XHR9KTtcblxuXHRcdHRoaXMuZGlzcGF0Y2hFdmVudChldnQpO1xuXHR9XG5cblx0X29uU2F2ZVdvcmRGaWxlQ2xpY2soKVxuXHR7XG5cdFx0aWYgKHRoaXMuX3ZhbGlkYXRvci52YWxpZGF0ZSgpKVxuXHRcdHtcblx0XHRcdC8vIFNob3cgc3Bpbm5lclxuXHRcdFx0JCgnI2VkaXRNb2RhbCAjc2F2ZVNwaW5uZXInLCAkKHRoaXMpKS5zaG93KCk7XG5cblx0XHRcdC8vIEZpcmUgZXZlbnQgdG8gcmVxdWVzdCBmaWxlIHRvIGJlIHNhdmVkIGJ5IHNlcnZlclxuXHRcdFx0bGV0IGV2dCA9IG5ldyBDdXN0b21FdmVudCh0aGlzLlNBVkVfV09SRFNfRklMRV9DTElDS19FVkVOVCwge1xuXHRcdCAgICBcdGRldGFpbDoge1xuXHRcdFx0XHRcdGZpbGVuYW1lOiAkKCcjZWRpdE1vZGFsICNmaWxlbmFtZScsICQodGhpcykpLnZhbCgpICsgdGhpcy5XT1JEU19GSUxFX0VYVCxcblx0XHRcdFx0XHRpc05ldzogdGhpcy5faXNOZXdGaWxlLFxuXHRcdFx0XHRcdGNvbnRlbnQ6ICQoJyNlZGl0TW9kYWwgI2NvbnRlbnQnLCAkKHRoaXMpKS52YWwoKVxuXHRcdFx0XHR9LFxuXHRcdFx0XHRidWJibGVzOiBmYWxzZSxcblx0XHRcdFx0Y2FuY2VsYWJsZTogZmFsc2Vcblx0XHRcdH0pO1xuXG5cdFx0XHR0aGlzLmRpc3BhdGNoRXZlbnQoZXZ0KTtcblx0XHR9XG5cdH1cblxuXHRfc2hvd01vZGFsKClcblx0e1xuXHRcdC8vIEFwcGVuZCBtb2RhbCBodG1sXG5cdFx0JCh0aGlzKS5hcHBlbmQodGhpcy5fbW9kYWxIdG1sKTtcblxuXHRcdGxldCBtb2RhbEVsZW1lbnQgPSAkKCcjZWRpdE1vZGFsJywgJCh0aGlzKSk7XG5cblx0XHQvLyBJbml0aWFsaXplIGtlbmRvIHZhbGlkYXRpb25cblx0XHR0aGlzLl92YWxpZGF0b3IgPSBtb2RhbEVsZW1lbnQuZmluZCgnI2VkaXRGaWVsZHNldCcpLmtlbmRvVmFsaWRhdG9yKHtcblx0XHRcdHZhbGlkYXRlT25CbHVyOiB0cnVlLFxuXHRcdFx0cnVsZXM6IHtcblx0XHRcdFx0cmVxdWlyZWRGaWxlbmFtZTogJC5wcm94eShmdW5jdGlvbihpbnB1dCkge1xuXHRcdFx0XHRcdGxldCB2YWxpZCA9IHRydWU7XG5cdFx0XHRcdFx0aWYgKGlucHV0LmlzKCdbbmFtZT1maWxlbmFtZV0nKSlcblx0XHRcdFx0XHRcdHZhbGlkID0gaW5wdXQudmFsKCkgIT09ICcnO1xuXHRcdFx0XHRcdHJldHVybiB2YWxpZDtcblx0XHRcdFx0fSwgdGhpcyksXG5cdFx0XHRcdHZhbGlkRmlsZW5hbWU6ICQucHJveHkoZnVuY3Rpb24oaW5wdXQpIHtcblx0XHRcdFx0XHRsZXQgdmFsaWQgPSB0cnVlO1xuXHRcdFx0XHRcdGlmIChpbnB1dC5pcygnW25hbWU9ZmlsZW5hbWVdJykpXG5cdFx0XHRcdFx0XHR2YWxpZCA9ICghaW5wdXQudmFsKCkuaW5jbHVkZXMoJ1xcXFwnKSAmJiAhaW5wdXQudmFsKCkuaW5jbHVkZXMoJy8nKSk7XG5cdFx0XHRcdFx0cmV0dXJuIHZhbGlkO1xuXHRcdFx0XHR9LCB0aGlzKVxuXHRcdFx0fSxcblx0XHRcdG1lc3NhZ2VzOiB7XG5cdFx0XHRcdHJlcXVpcmVkRmlsZW5hbWU6ICdSZXF1aXJlZCcsXG5cdFx0XHRcdHZhbGlkRmlsZW5hbWU6ICdDb250YWlucyBpbnZhbGlkIGNoYXJhY3RlcnMgKHNsYXNoLCBiYWNrc2xhc2gpJ1xuXHRcdFx0fVxuXHRcdH0pLmRhdGEoJ2tlbmRvVmFsaWRhdG9yJyk7XG5cblx0XHQvLyBIaWRlIHNhdmUgc3Bpbm5lclxuXHRcdCQoJyNzYXZlU3Bpbm5lcicsIG1vZGFsRWxlbWVudCkuaGlkZSgpO1xuXG5cdFx0Ly8gQWRkIGxpc3RlbmVyIHRvIFNhdmUgYnV0dG9uIGNsaWNrXG5cdFx0JCgnI3NhdmVXb3JkRmlsZUJ1dHRvbicsIG1vZGFsRWxlbWVudCkub24oJ2NsaWNrJywgJC5wcm94eSh0aGlzLl9vblNhdmVXb3JkRmlsZUNsaWNrLCB0aGlzKSk7XG5cblx0XHQvLyBBZGQgbGlzdGVuZXIgdG8gbW9kYWwgaGlkZSBldmVudFxuXHRcdG1vZGFsRWxlbWVudC5vbignaGlkZGVuLmJzLm1vZGFsJywgJC5wcm94eSh0aGlzLl9kZXN0cm95TW9kYWwsIHRoaXMpKTtcblxuXHRcdC8vIEluaXRpYWxpemUgYm9vdHN0cmFwIG1vZGFsXG5cdFx0bW9kYWxFbGVtZW50Lm1vZGFsKHtcblx0XHRcdGJhY2tkcm9wOiAnc3RhdGljJyxcblx0XHRcdGtleWJvYXJkOiBmYWxzZSxcblx0XHR9KTtcblx0fVxuXG5cdF9kZXN0cm95TW9kYWwoKVxuXHR7XG5cdFx0bGV0IG1vZGFsRWxlbWVudCA9ICQoJyNlZGl0TW9kYWwnLCAkKHRoaXMpKTtcblxuXHRcdGlmIChtb2RhbEVsZW1lbnQpXG5cdFx0e1xuXHRcdFx0Ly8gUmVtb3ZlIGxpc3RlbmVyc1xuXHRcdFx0JCgnI3NhdmVXb3JkRmlsZUJ1dHRvbicsIG1vZGFsRWxlbWVudCkub2ZmKCdjbGljaycpO1xuXHRcdFx0bW9kYWxFbGVtZW50Lm9mZignaGlkZGVuLmJzLm1vZGFsJyk7XG5cblx0XHRcdC8vIERlc3Ryb3kgZXZlcnl0aGluZyBLZW5kb1xuXHRcdFx0a2VuZG8uZGVzdHJveShtb2RhbEVsZW1lbnQpO1xuXG5cdFx0XHQvLyBEaXNwb3NlIG1vZGFsXG5cdFx0XHRtb2RhbEVsZW1lbnQubW9kYWwoJ2Rpc3Bvc2UnKTtcblxuXHRcdFx0Ly8gUmVtb3ZlIGh0bWxcblx0XHRcdG1vZGFsRWxlbWVudC5yZW1vdmUoKTtcblx0XHRcdG1vZGFsRWxlbWVudCA9IG51bGw7XG5cdFx0fVxuXHR9XG59XG5cbi8vIERFRklORSBDT01QT05FTlRcbmlmICghd2luZG93LmN1c3RvbUVsZW1lbnRzLmdldCgnd29yZHMtZmlsZXMtbWFuYWdlcicpKVxuXHR3aW5kb3cuY3VzdG9tRWxlbWVudHMuZGVmaW5lKCd3b3Jkcy1maWxlcy1tYW5hZ2VyJywgV29yZHNGaWxlc01hbmFnZXIpO1xuIiwiZXhwb3J0IGNsYXNzIFNpZGViYXJMYXlvdXQgZXh0ZW5kcyBIVE1MRWxlbWVudFxue1xuXHRjb25zdHJ1Y3RvcigpXG5cdHtcblx0ICAgIHN1cGVyKCk7XG5cblx0XHQvLyBBdHRhY2ggYSBzaGFkb3cgcm9vdFxuXHRcdGNvbnN0IHNoYWRvd1Jvb3QgPSB0aGlzLmF0dGFjaFNoYWRvdyh7bW9kZTogJ29wZW4nfSk7XG5cdFx0c2hhZG93Um9vdC5pbm5lckhUTUwgPSBgXG5cdFx0XHQ8c3R5bGU+XG5cdFx0XHRcdDpob3N0IHtcblx0XHRcdFx0XHRkaXNwbGF5OiBmbGV4O1xuXHRcdFx0XHRcdGZsZXgtZGlyZWN0aW9uOiByb3c7XG5cdFx0XHRcdH1cblxuXHRcdFx0XHRAbWVkaWEgKG1heC13aWR0aDogNTc1Ljk4cHgpIHtcblx0XHRcdFx0XHQ6aG9zdCguc3BsaXQteHMpIDo6c2xvdHRlZCg6bm90KFthcmlhLXNlbGVjdGVkPVwidHJ1ZVwiXSkpIHtcblx0XHRcdFx0XHRcdGRpc3BsYXk6IG5vbmUgIWltcG9ydGFudDtcblx0XHRcdFx0ICAgIH1cblxuXHRcdFx0XHRcdDpob3N0KC5zcGxpdC14cykgOjpzbG90dGVkKFthcmlhLXNlbGVjdGVkPVwidHJ1ZVwiXSkge1xuXHRcdFx0XHRcdFx0ZmxleC1ncm93OiAxO1xuXHRcdFx0XHQgICAgfVxuXHRcdFx0XHR9XG5cblx0XHRcdFx0QG1lZGlhIChtYXgtd2lkdGg6IDc2Ny45OHB4KSB7XG5cdFx0XHRcdFx0Omhvc3QoLnNwbGl0LXNtKSA6OnNsb3R0ZWQoOm5vdChbYXJpYS1zZWxlY3RlZD1cInRydWVcIl0pKSB7XG5cdFx0XHRcdFx0XHRkaXNwbGF5OiBub25lICFpbXBvcnRhbnQ7XG5cdFx0XHRcdCAgICB9XG5cblx0XHRcdFx0XHQ6aG9zdCguc3BsaXQtc20pIDo6c2xvdHRlZChbYXJpYS1zZWxlY3RlZD1cInRydWVcIl0pIHtcblx0XHRcdFx0XHRcdGZsZXgtZ3JvdzogMTtcblx0XHRcdFx0ICAgIH1cblx0XHRcdFx0fVxuXG5cdFx0XHRcdEBtZWRpYSAobWF4LXdpZHRoOiA5OTEuOThweCkge1xuXHRcdFx0XHRcdDpob3N0KC5zcGxpdC1tZCkgOjpzbG90dGVkKDpub3QoW2FyaWEtc2VsZWN0ZWQ9XCJ0cnVlXCJdKSkge1xuXHRcdFx0XHRcdFx0ZGlzcGxheTogbm9uZSAhaW1wb3J0YW50O1xuXHRcdFx0XHQgICAgfVxuXG5cdFx0XHRcdFx0Omhvc3QoLnNwbGl0LW1kKSA6OnNsb3R0ZWQoW2FyaWEtc2VsZWN0ZWQ9XCJ0cnVlXCJdKSB7XG5cdFx0XHRcdFx0XHRmbGV4LWdyb3c6IDE7XG5cdFx0XHRcdCAgICB9XG5cdFx0XHRcdH1cblxuXHRcdFx0XHRAbWVkaWEgKG1heC13aWR0aDogMTE5OS45OHB4KSB7XG5cdFx0XHRcdFx0Omhvc3QoLnNwbGl0LWxnKSA6OnNsb3R0ZWQoOm5vdChbYXJpYS1zZWxlY3RlZD1cInRydWVcIl0pKSB7XG5cdFx0XHRcdFx0XHRkaXNwbGF5OiBub25lICFpbXBvcnRhbnQ7XG5cdFx0XHRcdCAgICB9XG5cblx0XHRcdFx0XHQ6aG9zdCguc3BsaXQtbGcpIDo6c2xvdHRlZChbYXJpYS1zZWxlY3RlZD1cInRydWVcIl0pIHtcblx0XHRcdFx0XHRcdGZsZXgtZ3JvdzogMTtcblx0XHRcdFx0ICAgIH1cblx0XHRcdFx0fVxuXG5cdFx0XHRcdC5zaWRlLWNvbDo6c2xvdHRlZCgqKSB7XG5cdFx0XHRcdH1cblxuXHRcdFx0XHQubWFpbi1jb2w6OnNsb3R0ZWQoKikge1xuXHRcdFx0XHRcdGZsZXgtZ3JvdzogMTtcblx0XHRcdFx0fVxuXHRcdFx0PC9zdHlsZT5cblxuXHRcdFx0PHNsb3QgY2xhc3M9XCJzaWRlLWNvbFwiIG5hbWU9XCJzaWRlLWNvbHVtblwiPjwvc2xvdD5cblx0XHRcdDxzbG90IGNsYXNzPVwibWFpbi1jb2xcIiBuYW1lPVwibWFpbi1jb2x1bW5cIj48L3Nsb3Q+XG5cdFx0YDtcblxuXHRcdC8vIFNldCBpbml0aWFsIHNlbGVjdGlvblxuXHRcdHRoaXMuc2VsZWN0ZWRJbmRleCA9IDA7XG5cdH1cblxuXHRnZXQgc2VsZWN0ZWRQYW5lbCgpXG5cdHtcblx0XHRyZXR1cm4gdGhpcy5fc2VsZWN0ZWRQYW5lbDtcblx0fVxuXG5cdHNldCBzZWxlY3RlZFBhbmVsKGVsZW1lbnQpIC8vICdzaWRlJyBvciAnbWFpbidcblx0e1xuXHRcdGlmIChlbGVtZW50ICE9IG51bGwgJiYgZWxlbWVudC5wYXJlbnROb2RlID09IHRoaXMpXG5cdFx0e1xuXHRcdFx0dGhpcy5fc2VsZWN0ZWRQYW5lbCA9IGVsZW1lbnQ7XG5cblx0XHRcdGZvciAobGV0IGVsZW1lbnQgb2YgdGhpcy5jaGlsZHJlbilcblx0XHRcdHtcblx0XHRcdFx0aWYgKGVsZW1lbnQgPT0gdGhpcy5fc2VsZWN0ZWRQYW5lbClcblx0XHRcdFx0XHRlbGVtZW50LnNldEF0dHJpYnV0ZSgnYXJpYS1zZWxlY3RlZCcsICd0cnVlJyk7XG5cdFx0XHRcdGVsc2Vcblx0XHRcdFx0XHRlbGVtZW50LnJlbW92ZUF0dHJpYnV0ZSgnYXJpYS1zZWxlY3RlZCcpO1xuXHRcdFx0fVxuXHRcdH1cblx0XHRlbHNlXG5cdFx0e1xuXHRcdFx0Y29uc29sZS5lcnJvcignRWxlbWVudCBpcyBub3QgYSBjaGlsZCBvZiBTaWRlYmFyTGF5b3V0Jyk7XG5cdFx0fVxuXHR9XG5cblx0Z2V0IHNlbGVjdGVkSW5kZXgoKVxuXHR7XG5cdFx0cmV0dXJuIEFycmF5LmZyb20odGhpcy5jaGlsZHJlbikuaW5kZXhPZih0aGlzLl9zZWxlY3RlZFBhbmVsKTtcblx0fVxuXG5cdHNldCBzZWxlY3RlZEluZGV4KGluZGV4KVxuXHR7XG5cdFx0aWYgKHRoaXMuY2hpbGRyZW4ubGVuZ3RoID4gMClcblx0XHR7XG5cdFx0XHRpZiAodGhpcy5jaGlsZHJlbltpbmRleF0gPT0gbnVsbClcblx0XHRcdHtcblx0XHRcdFx0Y29uc29sZS5lcnJvcignSW52YWxpZCBTaWRlYmFyTGF5b3V0IGluZGV4Jyk7XG5cdFx0XHRcdHJldHVybjtcblx0XHRcdH1cblxuXHRcdFx0bGV0IGVsZW1lbnQgPSB0aGlzLmNoaWxkcmVuW2luZGV4XTtcblx0XHRcdHRoaXMuc2VsZWN0ZWRQYW5lbCA9IGVsZW1lbnQ7XG5cdFx0fVxuXHR9XG59XG5cbi8vIERFRklORSBDT01QT05FTlRcbmlmICghd2luZG93LmN1c3RvbUVsZW1lbnRzLmdldCgnc2lkZWJhci1sYXlvdXQnKSlcblx0d2luZG93LmN1c3RvbUVsZW1lbnRzLmRlZmluZSgnc2lkZWJhci1sYXlvdXQnLCBTaWRlYmFyTGF5b3V0KTtcbiIsImltcG9ydCB7QmFzZU1vZHVsZX0gZnJvbSAnLi9iYXNlLW1vZHVsZSc7XG5pbXBvcnQge1ZpZXdTdGFja30gZnJvbSAnLi4vY29tcG9uZW50cy92aWV3LXN0YWNrJztcbmltcG9ydCB7U2lkZWJhckxheW91dH0gZnJvbSAnLi4vY29tcG9uZW50cy9zaWRlYmFyLWxheW91dCc7XG5pbXBvcnQge0NvbmZpZ0ludGVyZmFjZUJ1aWxkZXJ9IGZyb20gJy4uL3V0aWxzL3VpYnVpbGRlci9jb25maWctaW50ZXJmYWNlLWJ1aWxkZXInO1xuaW1wb3J0IHtjYXBpdGFsaXplRmlyc3QsIGZpbHRlckNsYXNzTmFtZX0gZnJvbSAnLi4vdXRpbHMvdXRpbGl0aWVzJztcbmltcG9ydCB7V29yZHNGaWxlc01hbmFnZXJ9IGZyb20gJy4uL2NvbXBvbmVudHMvbW9kdWxlLXNwZWNpZmljL3dvcmRzLWZpbGVzLW1hbmFnZXInO1xuXG5leHBvcnQgZGVmYXVsdCBjbGFzcyBab25lQ29uZmlndXJhdG9yIGV4dGVuZHMgQmFzZU1vZHVsZVxue1xuXHRjb25zdHJ1Y3RvcigpXG5cdHtcblx0ICAgIHN1cGVyKCd6b25lQ29uZmlnJyk7XG5cblx0XHR0aGlzLklURU1fVFlQRV9aT05FID0gJ3pvbmUnO1xuXHRcdHRoaXMuSVRFTV9UWVBFX1JPT00gPSAncm9vbSc7XG5cblx0XHQvLyBPdXRnb2luZyByZXF1ZXN0c1xuXHRcdHRoaXMuUkVRX0dFVF9aT05FUyA9ICdnZXRab25lcyc7XG5cblx0XHR0aGlzLlJFUV9HRVRfWk9ORV9DT05GSUcgPSAnZ2V0Wm9uZUNvbmZpZyc7XG5cdFx0dGhpcy5SRVFfU0FWRV9aT05FX0NPTkZJRyA9ICdzYXZlWm9uZUNvbmZpZyc7XG5cdFx0dGhpcy5SRVFfTkVXX1pPTkVfQ09ORklHID0gJ25ld1pvbmVDb25maWcnO1xuXHRcdHRoaXMuUkVRX0RFTEVURV9aT05FX0NPTkZJRyA9ICdkZWxab25lQ29uZmlnJztcblx0XHR0aGlzLlJFUV9BQ1RJVkFURV9aT05FID0gJ2FjdFpvbmUnO1xuXG5cdFx0dGhpcy5SRVFfR0VUX1JPT01fQ09ORklHID0gJ2dldFJvb21Db25maWcnO1xuXHRcdHRoaXMuUkVRX1NBVkVfUk9PTV9DT05GSUcgPSAnc2F2ZVJvb21Db25maWcnO1xuXHRcdHRoaXMuUkVRX05FV19ST09NX0NPTkZJRyA9ICduZXdSb29tQ29uZmlnJztcblx0XHR0aGlzLlJFUV9ERUxFVEVfUk9PTV9DT05GSUcgPSAnZGVsUm9vbUNvbmZpZyc7XG5cblx0XHR0aGlzLlJFUV9SRUZSRVNIX1dPUkRTX0ZJTEUgPSAncmVmcmVzaFdvcmRzRmlsZXMnO1xuXHRcdHRoaXMuUkVRX0VESVRfV09SRFNfRklMRSA9ICdlZGl0V29yZHNGaWxlJztcblx0XHR0aGlzLlJFUV9TQVZFX1dPUkRTX0ZJTEUgPSAnc2F2ZVdvcmRzRmlsZSc7XG5cdFx0dGhpcy5SRVFfREVMRVRFX1dPUkRTX0ZJTEUgPSAnZGVsV29yZHNGaWxlJztcblxuXHRcdC8vIEluY29taW5nIHJlc3BvbnNlc1xuXHRcdHRoaXMuUkVTUF9aT05FUyA9ICd6b25lcyc7XG5cblx0XHR0aGlzLlJFU1BfWk9ORV9DT05GSUcgPSAnem9uZUNvbmZpZyc7XG5cdFx0dGhpcy5SRVNQX1pPTkVfQ09ORklHX1VQREFURV9DT05GSVJNID0gJ3pvbmVDZmdVcGQnO1xuXHRcdHRoaXMuUkVTUF9aT05FX0FEREVEID0gJ3pvbmVBZGRlZCc7XG5cdFx0dGhpcy5SRVNQX1pPTkVfUkVGVVNFRCA9ICd6b25lUmVmdXNlZCc7XG5cdFx0dGhpcy5SRVNQX1pPTkVfREVMRVRFRCA9ICd6b25lRGVsJztcblx0XHR0aGlzLlJFU1BfWk9ORV9BQ1RJVkFURUQgPSAnem9uZUFjdCc7XG5cdFx0dGhpcy5SRVNQX1pPTkVfQUNUSVZBVElPTl9FUlJPUiA9ICd6b25lQWN0RXJyJztcblxuXHRcdHRoaXMuUkVTUF9ST09NX0NPTkZJRyA9ICdyb29tQ29uZmlnJztcblx0XHR0aGlzLlJFU1BfUk9PTV9DT05GSUdfVVBEQVRFX0NPTkZJUk0gPSAncm9vbUNmZ1VwZCc7XG5cdFx0dGhpcy5SRVNQX1JPT01fQURERUQgPSAncm9vbUFkZGVkJztcblx0XHR0aGlzLlJFU1BfUk9PTV9SRUZVU0VEID0gJ3Jvb21SZWZ1c2VkJztcblx0XHR0aGlzLlJFU1BfUk9PTV9ERUxFVEVEID0gJ3Jvb21EZWwnO1xuXG5cdFx0dGhpcy5SRVNQX1JFRlJFU0hfV09SRFNfRklMRVMgPSAncmVmcmVzaFdvcmRzRmlsZXMnO1xuXHRcdHRoaXMuUkVTUF9XT1JEU19GSUxFX0NPTlRFTlQgPSAnd29yZHNGaWxlJztcblx0XHR0aGlzLlJFU1BfV09SRFNfRklMRV9FUlJPUiA9ICd3b3Jkc0ZpbGVFcnInO1xuXHR9XG5cblx0Ly8tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS1cblx0Ly8gQ09NTU9OIE1PRFVMRSBJTlRFUkZBQ0UgTUVUSE9EU1xuXHQvLyBUaGlzIG1lbWJlcnMgYXJlIHVzZWQgYnkgdGhlIG1haW4gY29udHJvbGxlclxuXHQvLyB0byBjb21tdW5pY2F0ZSB3aXRoIHRoZSBtb2R1bGUncyBjb250cm9sbGVyLlxuXHQvLyBUaGlzIG1ldGhvZHMgb3ZlcnJpZGUgdGhvc2UgaW4gQmFzZU1vZHVsZSBjbGFzcy5cblx0Ly8tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS1cblxuXHRpbml0aWFsaXplKGlkRGF0YSwgc2hlbGxDb250cm9sbGVyKVxuXHR7XG5cdFx0Ly8gQ2FsbCBzdXBlciBtZXRob2Rcblx0XHRzdXBlci5pbml0aWFsaXplKGlkRGF0YSwgc2hlbGxDb250cm9sbGVyKTtcblxuXHRcdC8vIENyZWF0ZSBpbnRlcmZhY2UgYnVpbGRlciBpbnN0YW5jZVxuXHRcdHRoaXMuX2ludGVyZmFjZUJ1aWxkZXIgPSBuZXcgQ29uZmlnSW50ZXJmYWNlQnVpbGRlcigpO1xuXG5cdFx0Ly8gU2V0IGxpc3RlbmVyIGZvciBjdXN0b20gYWN0aW9ucyB0cmlnZ2VyZWQgYnkgY29uZmlndXJhdGlvbiBpbnRlcmZhY2Vcblx0XHQkKCcjem5jLXRhYk5hdmlnYXRvcicpLm9uKCd2YWx1ZS1zZXQnLCAkLnByb3h5KHRoaXMuX29uQ29uZmlnVmFsdWVTZXQsIHRoaXMpKTtcblxuXHRcdC8vIEluaXRpYWxpemUgWm9uZXMvUm9vbXMgdHJlZXZpZXdcblx0XHR0aGlzLl90cmVldmlldyA9ICQoJyN6bmMtdHJlZVZpZXcnKS5rZW5kb1RyZWVWaWV3KHtcblx0XHRcdGxvYWRPbkRlbWFuZDogZmFsc2UsXG5cdFx0XHRkYXRhVGV4dEZpZWxkOiAnbmFtZScsXG5cdFx0XHR0ZW1wbGF0ZToga2VuZG8udGVtcGxhdGUoJzxzcGFuIGNsYXNzPVwiIyBpZiAoIWl0ZW0uYWN0aXZlKSB7ICMgaW5hY3RpdmUtbGlzdC1pdGVtICMgfSAjXCI+IzogaXRlbS5uYW1lICM8L3NwYW4+JyksXG5cdFx0XHRjaGFuZ2U6ICQucHJveHkodGhpcy5fb25ab25lUm9vbUNoYW5nZSwgdGhpcyksXG5cdFx0fSkuZGF0YSgna2VuZG9UcmVlVmlldycpO1xuXG5cdFx0Ly8gTGlzdGVuIHRvIHRyZWV2aWV3IGRvdWJsZS1jbGljayBldmVudFxuXHRcdCQoJyN6bmMtdHJlZVZpZXcnKS5vbignZGJsY2xpY2snLCAkLnByb3h5KHRoaXMuX29uVHJlZUl0ZW1Eb3VibGVDbGljaywgdGhpcykpO1xuXG5cdFx0Ly8gUmVxdWVzdCB6b25lcyAmIHJvb21zIGxpc3QgdG8gc2VydmVyIGluc3RhbmNlXG5cdFx0dGhpcy5zZW5kRXh0ZW5zaW9uUmVxdWVzdCh0aGlzLlJFUV9HRVRfWk9ORVMpO1xuXG5cdFx0Ly8gSW5pdGlhbGl6ZSBwcm9ncmVzcyBiYXJcblx0XHQkKCcjem5jLXByb2dyZXNzQmFyJykua2VuZG9Qcm9ncmVzc0Jhcih7XG5cdFx0XHRtaW46IDAsXG4gICAgICAgICAgICBtYXg6IDEwMCxcblx0XHRcdHZhbHVlOiBmYWxzZSxcbiAgICAgICAgICAgIHR5cGU6ICd2YWx1ZScsXG4gICAgICAgICAgICBhbmltYXRpb246IHtcbiAgICAgICAgICAgICAgICBkdXJhdGlvbjogNDAwXG4gICAgICAgICAgICB9XG4gICAgICAgIH0pO1xuXG5cdFx0Ly8gQWRkIGxpc3RlbmVycyB0byB1dGlsaXR5IGJ1dHRvbnNcblx0XHQkKCcjem5jLWFkZFpvbmVCdXR0b24nKS5vbignY2xpY2snLCAkLnByb3h5KHRoaXMuX29uQWRkWm9uZUNsaWNrLCB0aGlzKSk7XG5cdFx0JCgnI3puYy1hZGRSb29tQnV0dG9uJykub24oJ2NsaWNrJywgJC5wcm94eSh0aGlzLl9vbkFkZFJvb21DbGljaywgdGhpcykpO1xuXHRcdCQoJyN6bmMtZWRpdEJ1dHRvbicpLm9uKCdjbGljaycsICQucHJveHkodGhpcy5fb25FZGl0Q2xpY2ssIHRoaXMpKTtcblx0XHQkKCcjem5jLXJlbW92ZUJ1dHRvbicpLm9uKCdjbGljaycsICQucHJveHkodGhpcy5fb25SZW1vdmVDbGljaywgdGhpcykpO1xuXHRcdCQoJyN6bmMtYWN0aXZhdGVCdXR0b24nKS5vbignY2xpY2snLCAkLnByb3h5KHRoaXMuX29uQWN0aXZhdGVDbGljaywgdGhpcykpO1xuXG5cdFx0Ly8gQWRkIGxpc3RlbmVyIHRvIGludGVyZmFjZSBidXR0b25zXG5cdFx0JCgnI3puYy1jYW5jZWxCdXR0b24nKS5vbignY2xpY2snLCAkLnByb3h5KHRoaXMuX29uQ2FuY2VsQ2xpY2ssIHRoaXMpKTtcblx0XHQkKCcjem5jLXJlbG9hZEJ1dHRvbicpLm9uKCdjbGljaycsICQucHJveHkodGhpcy5fb25SZWxvYWRDbGljaywgdGhpcykpO1xuXHRcdCQoJyN6bmMtc3VibWl0QnV0dG9uJykub24oJ2NsaWNrJywgJC5wcm94eSh0aGlzLl9vblN1Ym1pdENsaWNrLCB0aGlzKSk7XG5cblx0XHQvLyBTYXZlIHJlZiB0byB3b3JkcyBmaWxlcyBtYW5hZ2VyIGFuZCBhZGQgY3VzdG9tIGV2ZW50IGxpc3RlbmVyc1xuXHRcdHRoaXMuX3dvcmRzRmlsZXNNYW5hZ2VyID0gZG9jdW1lbnQuZ2V0RWxlbWVudEJ5SWQoJ3puYy13b3Jkc0ZpbGVzTWFuYWdlcicpO1xuXHRcdCQodGhpcy5fd29yZHNGaWxlc01hbmFnZXIpLm9uKHRoaXMuX3dvcmRzRmlsZXNNYW5hZ2VyLlJFRlJFU0hfV09SRFNfRklMRVNfQ0xJQ0tfRVZFTlQsICQucHJveHkodGhpcy5fb25Xb3Jkc0ZpbGVSZWxvYWRDbGljaywgdGhpcykpO1xuXHRcdCQodGhpcy5fd29yZHNGaWxlc01hbmFnZXIpLm9uKHRoaXMuX3dvcmRzRmlsZXNNYW5hZ2VyLkVESVRfV09SRFNfRklMRV9DTElDS19FVkVOVCwgJC5wcm94eSh0aGlzLl9vbldvcmRzRmlsZUVkaXRDbGljaywgdGhpcykpO1xuXHRcdCQodGhpcy5fd29yZHNGaWxlc01hbmFnZXIpLm9uKHRoaXMuX3dvcmRzRmlsZXNNYW5hZ2VyLlNBVkVfV09SRFNfRklMRV9DTElDS19FVkVOVCwgJC5wcm94eSh0aGlzLl9vbldvcmRzRmlsZVNhdmVDbGljaywgdGhpcykpO1xuXHRcdCQodGhpcy5fd29yZHNGaWxlc01hbmFnZXIpLm9uKHRoaXMuX3dvcmRzRmlsZXNNYW5hZ2VyLlJFTU9WRV9XT1JEU19GSUxFX0NMSUNLX0VWRU5ULCAkLnByb3h5KHRoaXMuX29uV29yZHNGaWxlUmVtb3ZlQ2xpY2ssIHRoaXMpKTtcblx0XHQkKHRoaXMuX3dvcmRzRmlsZXNNYW5hZ2VyKS5vbih0aGlzLl93b3Jkc0ZpbGVzTWFuYWdlci5BU1NJR05fV09SRFNfRklMRV9DTElDS19FVkVOVCwgJC5wcm94eSh0aGlzLl9vbldvcmRzRmlsZUFzc2lnbkNsaWNrLCB0aGlzKSk7XG5cdH1cblxuXHRkZXN0cm95KClcblx0e1xuXHRcdC8vIENhbGwgc3VwZXIgbWV0aG9kXG5cdFx0c3VwZXIuZGVzdHJveSgpO1xuXG5cdFx0Ly8gUmVtb3ZlIHRyZWUgdmlldyBkb3VibGVjbGljayBsaXN0ZW5lclxuXHRcdCQoJyN6bmMtdHJlZVZpZXcnKS5vZmYoJ2RibGNsaWNrJyk7XG5cblx0XHQvLyBSZW1vdmUgbGlzdGVuZXIgZm9yIGN1c3RvbSBhY3Rpb25zIHRyaWdnZXJlZCBieSBjb25maWd1cmF0aW9uIGludGVyZmFjZVxuXHRcdCQoJyN6bmMtdGFiTmF2aWdhdG9yJykub2ZmKCd2YWx1ZS1zZXQnKTtcblxuXHRcdC8vIFJlbW92ZSBsaXN0ZW5lciBmb3Igem9uZS9yb29tIGFjdGl2YXRpb24gZXZlbnRcblx0XHQkKCcjem5jLWFkZFpvbmVCdXR0b24nKS5vZmYoJ2NsaWNrJyk7XG5cdFx0JCgnI3puYy1hZGRSb29tQnV0dG9uJykub2ZmKCdjbGljaycpO1xuXHRcdCQoJyN6bmMtZWRpdEJ1dHRvbicpLm9mZignY2xpY2snKTtcblx0XHQkKCcjem5jLXJlbW92ZUJ1dHRvbicpLm9mZignY2xpY2snKTtcblx0XHQkKCcjem5jLWFjdGl2YXRlQnV0dG9uJykub2ZmKCdjbGljaycpO1xuXG5cdFx0Ly8gUmVtb3ZlIGludGVyZmFjZSBidXR0b25zIGNsaWNrIGxpc3RlbmVyc1xuXHRcdCQoJyN6bmMtY2FuY2VsQnV0dG9uJykub2ZmKCdjbGljaycpO1xuXHRcdCQoJyN6bmMtcmVsb2FkQnV0dG9uJykub2ZmKCdjbGljaycpO1xuXHRcdCQoJyN6bmMtc3VibWl0QnV0dG9uJykub2ZmKCdjbGljaycpO1xuXG5cdFx0Ly8gUmVtb3ZlIGxpc3RlbmVycyB0byB3b3JkcyBmaWxlcyBtYW5hZ2VyXG5cdFx0JCh0aGlzLl93b3Jkc0ZpbGVzTWFuYWdlcikub2ZmKHRoaXMuX3dvcmRzRmlsZXNNYW5hZ2VyLlJFRlJFU0hfV09SRFNfRklMRVNfQ0xJQ0tfRVZFTlQpO1xuXHRcdCQodGhpcy5fd29yZHNGaWxlc01hbmFnZXIpLm9mZih0aGlzLl93b3Jkc0ZpbGVzTWFuYWdlci5FRElUX1dPUkRTX0ZJTEVfQ0xJQ0tfRVZFTlQpO1xuXHRcdCQodGhpcy5fd29yZHNGaWxlc01hbmFnZXIpLm9mZih0aGlzLl93b3Jkc0ZpbGVzTWFuYWdlci5TQVZFX1dPUkRTX0ZJTEVfQ0xJQ0tfRVZFTlQpO1xuXHRcdCQodGhpcy5fd29yZHNGaWxlc01hbmFnZXIpLm9mZih0aGlzLl93b3Jkc0ZpbGVzTWFuYWdlci5SRU1PVkVfV09SRFNfRklMRV9DTElDS19FVkVOVCk7XG5cdFx0JCh0aGlzLl93b3Jkc0ZpbGVzTWFuYWdlcikub2ZmKHRoaXMuX3dvcmRzRmlsZXNNYW5hZ2VyLkFTU0lHTl9XT1JEU19GSUxFX0NMSUNLX0VWRU5UKTtcblxuXHRcdC8vIENsZWFyIHRhYnMgY29udGFpbmVyXG5cdFx0dGhpcy5fY2xlYXJUYWJzKCk7XG5cdH1cblxuXHRvbkV4dGVuc2lvbkNvbW1hbmQoY29tbWFuZCwgZGF0YSlcblx0e1xuXHRcdGNvbnN0IHVzZXJuYW1lID0gZGF0YS5nZXRVdGZTdHJpbmcoJ3VzZXInKTtcblxuXHRcdC8qKioqKiogWk9ORVMgJiBST09NUyAqKioqKiovXG5cblx0XHQvLyBab25lcyAmIHJvb21zIGxpc3QgcmVjZWl2ZWRcblx0XHRpZiAoY29tbWFuZCA9PSB0aGlzLlJFU1BfWk9ORVMpXG5cdFx0XHR0aGlzLl9wb3B1bGF0ZVRyZWUoZGF0YSk7XG5cblx0XHQvLyBab25lIG9yIHJvb20gY29uZmlndXJhdGlvbiBkYXRhIHJlY2VpdmVkXG5cdFx0ZWxzZSBpZiAoY29tbWFuZCA9PSB0aGlzLlJFU1BfWk9ORV9DT05GSUcgfHwgY29tbWFuZCA9PSB0aGlzLlJFU1BfUk9PTV9DT05GSUcpXG5cdFx0e1xuXHRcdFx0Ly8gQnVpbGQgdXNlciBpbnRlcmZhY2UgYmFzZWQgb24gcmVjZWl2ZWQgZGF0YVxuXHRcdFx0dGhpcy5faW50ZXJmYWNlQnVpbGRlci5idWlsZEludGVyZmFjZShkYXRhLmdldFNGU0FycmF5KCdzZXR0aW5ncycpLCAnem5jLXRhYk5hdmlnYXRvcicsIGZhbHNlKTtcblxuXHRcdFx0Ly8gRW5hYmxlIHNjcm9sbGluZyB0YWJzIChpZiBuZWVkZWQpXG5cdFx0XHRpZiAodGhpcy5fcmVpbml0VGFicylcblx0XHRcdHtcblx0XHRcdFx0JCgnI3puYy10YWJOYXZpZ2F0b3IgI3RhYnMnKS5zY3JvbGxpbmdUYWJzKHtcblx0XHRcdFx0XHRib290c3RyYXBWZXJzaW9uOiA0LFxuXHRcdFx0XHRcdHNjcm9sbFRvVGFiRWRnZTogdHJ1ZSxcblx0XHRcdFx0XHRlbmFibGVTd2lwaW5nOiB0cnVlLFxuXHRcdFx0XHRcdGRpc2FibGVTY3JvbGxBcnJvd3NPbkZ1bGx5U2Nyb2xsZWQ6IHRydWUsXG5cdFx0XHRcdFx0Y3NzQ2xhc3NMZWZ0QXJyb3c6ICdmYSBmYS1jaGV2cm9uLWxlZnQnLFxuXHRcdFx0XHRcdGNzc0NsYXNzUmlnaHRBcnJvdzogJ2ZhIGZhLWNoZXZyb24tcmlnaHQnXG5cdFx0XHRcdH0pO1xuXHRcdFx0fVxuXG5cdFx0XHQvLyBFbmFibGUgaW50ZXJmYWNlXG5cdFx0XHR0aGlzLl9lbmFibGVDb25maWdJbnRlcmZhY2UodHJ1ZSk7XG5cdFx0fVxuXG5cdFx0LyoqKioqKiBaT05FUyAqKioqKiovXG5cblx0XHQvLyBab25lIGNvbmZpZ3VyYXRpb24gdXBkYXRlIGNvbmZpcm1hdGlvblxuXHRcdGVsc2UgaWYgKGNvbW1hbmQgPT0gdGhpcy5SRVNQX1pPTkVfQ09ORklHX1VQREFURV9DT05GSVJNKVxuXHRcdHtcblx0XHRcdC8vIElmIGEgJ25hbWUnIHBhcmFtZXRlciBpcyByZWNlaXZlZCwgaXQgbWVhbnMgdGhlIHpvbmUgbmFtZSBjaGFuZ2VkLCBhbmQgd2UgaGF2ZSB0byB1cGRhdGUgdGhlIHpvbmVzIGxpc3Rcblx0XHRcdGlmIChkYXRhLmdldFV0ZlN0cmluZygnek5hbWUnKSAhPSBudWxsKVxuXHRcdFx0XHR0aGlzLl91cGRhdGVab25lTmFtZUluTGlzdChkYXRhLmdldEludCgneklkJyksIGRhdGEuZ2V0VXRmU3RyaW5nKCd6TmFtZScpKTtcblxuXHRcdFx0Ly8gSWYgdGhlIGN1cnJlbnQgdXNlciBpcyB0aGUgdXBkYXRlciwgc2hvdyBhIG5vdGlmaWNhdGlvbjsgb3RoZXJ3aXNlLCBzaG93IGEgZGlhbG9nIGJveCBzdWdnZXN0aW5nIHRvIHJlbG9hZFxuXHRcdFx0aWYgKHVzZXJuYW1lID09IHRoaXMuc21hcnRGb3gubXlTZWxmLm5hbWUpXG5cdFx0XHR7XG5cdFx0XHRcdC8vIEVuYWJsZSBpbnRlcmZhY2Vcblx0XHRcdFx0dGhpcy5fZW5hYmxlQ29uZmlnSW50ZXJmYWNlKHRydWUpO1xuXG5cdFx0XHRcdC8vIERpc3BsYXkgbm90aWZpY2F0aW9uXG5cdFx0XHRcdHRoaXMuc2hlbGxDdHJsLnNob3dOb3RpZmljYXRpb24oJ1pvbmUgbW9kaWZpZWQnLCBgWm9uZSBzZXR0aW5ncyB1cGRhdGVkIHN1Y2Nlc3NmdWxseTsgY2hhbmdlcyB3aWxsIGJlIGFwcGxpZWQgb24gbmV4dCA8c3Ryb25nPnNlcnZlciByZXN0YXJ0PC9zdHJvbmc+YCk7XG5cblx0XHRcdFx0Ly8gUmVzZXQgdGhlICdtb2RpZmllZCcgZmxhZ1xuXHRcdFx0XHR0aGlzLl9pbnRlcmZhY2VCdWlsZGVyLnJlc2V0SXNNb2RpZmllZCgpO1xuXHRcdFx0fVxuXHRcdFx0ZWxzZVxuXHRcdFx0e1xuXHRcdFx0XHQvLyBBbiBhbGVydCBib3ggaXMgZGlzcGxheWVkIGlmIHRoZSB1c2VyIGlzIGN1cnJlbnRseSBlZGl0aW5nIHRoZSBzYW1lIHpvbmVcblx0XHRcdFx0aWYgKGRhdGEuZ2V0SW50KCd6SWQnKSA9PSB0aGlzLl9lZGl0ZWRab25lSWQpXG5cdFx0XHRcdHtcblx0XHRcdFx0XHQvLyBTaG93IGFsZXJ0XG5cdFx0XHRcdFx0dGhpcy5zaGVsbEN0cmwuc2hvd1NpbXBsZUFsZXJ0KGBBZG1pbmlzdHJhdG9yICR7dXNlcm5hbWV9IGhhcyBtb2RpZmllZCB0aGUgWm9uZSB5b3UgYXJlIGN1cnJlbnRseSBlZGl0aW5nOyBwbGVhc2UgcmVsb2FkIHRvIHVwZGF0ZSB5b3VyIHZpZXcuYCk7XG5cblx0XHRcdFx0XHQvLyBEaXNhYmxlIHN1Ym1pdCBidXR0b25cblx0XHRcdFx0XHQkKCcjem5jLXN1Ym1pdEJ1dHRvbicpLmF0dHIoJ2Rpc2FibGVkJywgdHJ1ZSk7XG5cdFx0XHRcdH1cblx0XHRcdFx0ZWxzZVxuXHRcdFx0XHR7XG5cdFx0XHRcdFx0Ly8gRGlzcGxheSBub3RpZmljYXRpb25cblx0XHRcdFx0XHRpZiAoZGF0YS5nZXRVdGZTdHJpbmcoJ3pOYW1lJykgIT0gbnVsbClcblx0XHRcdFx0XHRcdHRoaXMuc2hlbGxDdHJsLnNob3dOb3RpZmljYXRpb24oJ1pvbmUgcmVuYW1lZCcsIGBBZG1pbmlzdHJhdG9yICR7dXNlcm5hbWV9IGhhcyBjaGFuZ2VkIHRoZSBuYW1lIG9uIG9uZSBvZiB0aGUgWm9uZXNgKTtcblx0XHRcdFx0fVxuXHRcdFx0fVxuXHRcdH1cblxuXHRcdC8vIE5ldyB6b25lIGFkZGVkXG5cdFx0ZWxzZSBpZiAoY29tbWFuZCA9PSB0aGlzLlJFU1BfWk9ORV9BRERFRClcblx0XHR7XG5cdFx0XHRjb25zdCB6b25lTmFtZSA9IGRhdGEuZ2V0U0ZTT2JqZWN0KCd6b25lJykuZ2V0VXRmU3RyaW5nKCduYW1lJyk7XG5cblx0XHRcdC8vIElmIHRoZSBjdXJyZW50IHVzZXIgaXMgdGhlIHVwZGF0ZXIsIHJlc2V0IHRoZSBpbnRlcmZhY2U7IG90aGVyd2lzZSwganVzdCBzaG93IGEgbm90aWZpY2F0aW9uXG5cdFx0XHRpZiAodXNlcm5hbWUgPT0gdGhpcy5zbWFydEZveC5teVNlbGYubmFtZSlcblx0XHRcdHtcblx0XHRcdFx0Ly8gUmVzZXQgaW50ZXJmYWNlXG5cdFx0XHRcdHRoaXMuX29uQ2FuY2VsQ2xpY2soKTtcblxuXHRcdFx0XHQvLyBEaXNwbGF5IG5vdGlmaWNhdGlvblxuXHRcdFx0XHR0aGlzLnNoZWxsQ3RybC5zaG93Tm90aWZpY2F0aW9uKCdab25lIGFkZGVkJywgYFpvbmUgJyR7em9uZU5hbWV9JyBjcmVhdGVkIHN1Y2Nlc3NmdWxseWApO1xuXHRcdFx0fVxuXHRcdFx0ZWxzZVxuXHRcdFx0e1xuXHRcdFx0XHQvLyBEaXNwbGF5IG5vdGlmaWNhdGlvblxuXHRcdFx0XHR0aGlzLnNoZWxsQ3RybC5zaG93Tm90aWZpY2F0aW9uKCdab25lIGFkZGVkJywgYEFkbWluaXN0cmF0b3IgJHt1c2VybmFtZX0gY3JlYXRlZCBab25lICcke3pvbmVOYW1lfSdgKTtcblx0XHRcdH1cblxuXHRcdFx0Ly8gQWRkIG5ldyB6b25lIHRvIHRyZWVcblx0XHRcdGxldCB6b25lc0RTID0gdGhpcy5fdHJlZXZpZXcuZGF0YVNvdXJjZTtcblx0XHRcdHpvbmVzRFMuYWRkKHRoaXMuX2NyZWF0ZVpvbmVPYmplY3QoZGF0YS5nZXRTRlNPYmplY3QoJ3pvbmUnKSkpO1xuXHRcdFx0em9uZXNEUy5zeW5jKCk7XG5cdFx0fVxuXG5cdFx0Ly8gTmV3IHpvbmUgY3JlYXRpb24gcmVmdXNlZCBkdWUgdG8gaW52YWxpZCB6b25lIG5hbWVcblx0XHRlbHNlIGlmIChjb21tYW5kID09IHRoaXMuUkVTUF9aT05FX1JFRlVTRUQpXG5cdFx0e1xuXHRcdFx0Ly8gUmUtZW5hYmxlIGludGVyZmFjZVxuXHRcdFx0dGhpcy5fZW5hYmxlQ29uZmlnSW50ZXJmYWNlKHRydWUpO1xuXG5cdFx0XHQvLyBTaG93IHdhcm5pbmdcblx0XHRcdHRoaXMuc2hlbGxDdHJsLnNob3dTaW1wbGVBbGVydCgnWm9uZSBjb25maWd1cmF0aW9uIGNhblxcJ3QgYmUgc2F2ZWQgYmVjYXVzZSBhbm90aGVyIFpvbmUgd2l0aCB0aGUgc2FtZSBuYW1lIGFscmVhZHkgZXhpc3RzLicsIHRydWUpO1xuXHRcdH1cblxuXHRcdC8vIEV4aXN0aW5nIHpvbmUgZGVsZXRlZFxuXHRcdGVsc2UgaWYgKGNvbW1hbmQgPT0gdGhpcy5SRVNQX1pPTkVfREVMRVRFRClcblx0XHR7XG5cdFx0XHQvLyBJZiB0aGUgY3VycmVudCB1c2VyIGlzIHRoZSBkZWxldGVyLCByZXNldCB0aGUgaW50ZXJmYWNlOyBvdGhlcndpc2UsIGp1c3Qgc2hvdyBhIG5vdGlmaWNhdGlvblxuXHRcdFx0aWYgKHVzZXJuYW1lID09IHRoaXMuc21hcnRGb3gubXlTZWxmLm5hbWUpXG5cdFx0XHR7XG5cdFx0XHRcdC8vIFJlLWVuYWJsZSBpbnRlcmZhY2Vcblx0XHRcdFx0dGhpcy5fZW5hYmxlTGlzdEludGVyZmFjZSh0cnVlKTtcblxuXHRcdFx0XHQvLyBEaXNwbGF5IG5vdGlmaWNhdGlvblxuXHRcdFx0XHR0aGlzLnNoZWxsQ3RybC5zaG93Tm90aWZpY2F0aW9uKCdab25lIHJlbW92ZWQnLCBgWm9uZSAnJHtkYXRhLmdldFV0ZlN0cmluZygnek5hbWUnKX0nIGRlbGV0ZWQgc3VjY2Vzc2Z1bGx5YCk7XG5cdFx0XHR9XG5cdFx0XHRlbHNlXG5cdFx0XHR7XG5cdFx0XHRcdC8vIEFuIGFsZXJ0IGJveCBpcyBkaXNwbGF5ZWQgaWYgdGhlIHVzZXIgaXMgY3VycmVudGx5IGVkaXRpbmcgdGhlIHNhbWUgem9uZVxuXHRcdFx0XHRpZiAoZGF0YS5nZXRJbnQoJ3pJZCcpID09IHRoaXMuX2VkaXRlZFpvbmVJZClcblx0XHRcdFx0e1xuXHRcdFx0XHRcdC8vIFNob3cgYWxlcnRcblx0XHRcdFx0XHR0aGlzLnNoZWxsQ3RybC5zaG93U2ltcGxlQWxlcnQoYEFkbWluaXN0cmF0b3IgJHt1c2VybmFtZX0gaGFzIGRlbGV0ZWQgdGhlIFpvbmUgeW91IGFyZSBjdXJyZW50bHkgbW9kaWZ5aW5nOyB5b3UgaGF2ZSB0byBjYW5jZWwgeW91ciBlZGl0aW5nLmApO1xuXG5cdFx0XHRcdFx0Ly8gRGlzYWJsZSBzdWJtaXQgYW5kIHJlbG9hZCBidXR0b25zXG5cdFx0XHRcdFx0JCgnI3puYy1yZWxvYWRCdXR0b24nKS5hdHRyKCdkaXNhYmxlZCcsIHRydWUpO1xuXHRcdFx0XHRcdCQoJyN6bmMtc3VibWl0QnV0dG9uJykuYXR0cignZGlzYWJsZWQnLCB0cnVlKTtcblx0XHRcdFx0fVxuXHRcdFx0XHRlbHNlXG5cdFx0XHRcdHtcblx0XHRcdFx0XHQvLyBEaXNwbGF5IG5vdGlmaWNhdGlvblxuXHRcdFx0XHRcdHRoaXMuc2hlbGxDdHJsLnNob3dOb3RpZmljYXRpb24oJ1pvbmUgcmVtb3ZlZCcsIGBBZG1pbmlzdHJhdG9yICR7dXNlcm5hbWV9IGRlbGV0ZWQgWm9uZSAnJHtkYXRhLmdldFV0ZlN0cmluZygnek5hbWUnKX0nYCk7XG5cdFx0XHRcdH1cblx0XHRcdH1cblxuXHRcdFx0Ly8gUmVzZXQgc2VsZWN0aW9uIGlmIHRoZSBjdXJyZW50bHkgc2VsZWN0ZWQgaXRlbSBvciBpdHMgcGFyZW50IGlzIGJlaW5nIHJlbW92ZWRcblx0XHRcdGxldCBzZWxlY3RlZE5vZGUgPSB0aGlzLl90cmVldmlldy5zZWxlY3QoKTtcblx0XHRcdGxldCBzZWxlY3RlZERhdGFJdGVtID0gdGhpcy5fdHJlZXZpZXcuZGF0YUl0ZW0oc2VsZWN0ZWROb2RlKTtcblx0XHRcdGlmIChzZWxlY3RlZERhdGFJdGVtKVxuXHRcdFx0e1xuXHRcdFx0XHRpZiAoc2VsZWN0ZWREYXRhSXRlbS50eXBlID09IHRoaXMuSVRFTV9UWVBFX1pPTkUgJiYgc2VsZWN0ZWREYXRhSXRlbS5pZCA9PSBkYXRhLmdldEludCgneklkJykpXG5cdFx0XHRcdFx0dGhpcy5fZGVzZWxlY3RUcmVlSXRlbSgpO1xuXG5cdFx0XHRcdGlmIChzZWxlY3RlZERhdGFJdGVtLnR5cGUgPT0gdGhpcy5JVEVNX1RZUEVfUk9PTSlcblx0XHRcdFx0e1xuXHRcdFx0XHRcdGxldCBwYXJlbnREYXRhSXRlbSA9IHRoaXMuX3RyZWV2aWV3LmRhdGFJdGVtKHRoaXMuX3RyZWV2aWV3LnBhcmVudChzZWxlY3RlZE5vZGUpKTtcblxuXHRcdFx0XHRcdGlmIChwYXJlbnREYXRhSXRlbS5pZCA9PSBkYXRhLmdldEludCgneklkJykpXG5cdFx0XHRcdFx0XHR0aGlzLl9kZXNlbGVjdFRyZWVJdGVtKCk7XG5cdFx0XHRcdH1cblx0XHRcdH1cblxuXHRcdFx0Ly8gUmVtb3ZlIHpvbmUgZnJvbSB0cmVlXG5cdFx0XHRsZXQgZGF0YUl0ZW0gPSB0aGlzLl9nZXRab25lRGF0YUl0ZW1CeUlkKGRhdGEuZ2V0SW50KCd6SWQnKSk7XG5cdFx0XHRsZXQgem9uZXNEUyA9IHRoaXMuX3RyZWV2aWV3LmRhdGFTb3VyY2U7XG5cdFx0XHR6b25lc0RTLnJlbW92ZShkYXRhSXRlbSk7XG5cdFx0XHR6b25lc0RTLnN5bmMoKTtcblx0XHR9XG5cblx0XHQvLyBab25lIGFjdGl2YXRlZFxuXHRcdGVsc2UgaWYgKGNvbW1hbmQgPT0gdGhpcy5SRVNQX1pPTkVfQUNUSVZBVEVEKVxuXHRcdHtcblx0XHRcdC8vIFNldCB6b25lIGFjdGl2YXRpb24gc3RhdHVzXG5cdFx0XHRjb25zdCB6b25lTmFtZSA9IHRoaXMuX3NldFpvbmVBY3RpdmF0aW9uU3RhdHVzKGRhdGEuZ2V0SW50KCd6SWQnKSwgZGF0YS5nZXRVdGZTdHJpbmcoJ2FjdFJvb21zJyksIHRydWUpO1xuXG5cdFx0XHQvLyBEaXNwbGF5IG5vdGlmaWNhdGlvblxuXHRcdFx0aWYgKHVzZXJuYW1lID09IHRoaXMuc21hcnRGb3gubXlTZWxmLm5hbWUpXG5cdFx0XHRcdHRoaXMuc2hlbGxDdHJsLnNob3dOb3RpZmljYXRpb24oJ1pvbmUgYWN0aXZhdGVkJywgYFpvbmUgJyR7em9uZU5hbWV9JyBhY3RpdmF0ZWQgc3VjY2Vzc2Z1bGx5YCk7XG5cdFx0XHRlbHNlXG5cdFx0XHRcdHRoaXMuc2hlbGxDdHJsLnNob3dOb3RpZmljYXRpb24oJ1pvbmUgYWN0aXZhdGVkJywgYEFkbWluaXN0cmF0b3IgJHt1c2VybmFtZX0gYWN0aXZhdGVkIFpvbmUgJyR7em9uZU5hbWV9J2ApO1xuXHRcdH1cblxuXHRcdC8vIFpvbmUgYWN0aXZhdGlvbiBlcnJvclxuXHRcdGVsc2UgaWYgKGNvbW1hbmQgPT0gdGhpcy5SRVNQX1pPTkVfQUNUSVZBVElPTl9FUlJPUilcblx0XHR7XG5cdFx0XHQvLyBTZXQgem9uZSBhY3RpdmF0aW9uIHN0YXR1c1xuXHRcdFx0dGhpcy5fc2V0Wm9uZUFjdGl2YXRpb25TdGF0dXMoZGF0YS5nZXRJbnQoJ3pJZCcpLCAnJywgZmFsc2UpO1xuXG5cdFx0XHQvLyBTaG93IGFsZXJ0XG5cdFx0XHR0aGlzLnNoZWxsQ3RybC5zaG93U2ltcGxlQWxlcnQoZGF0YS5nZXRVdGZTdHJpbmcoJ2Vycm9yJyksIHRydWUpO1xuXHRcdH1cblxuXHRcdC8qKioqKiogUk9PTVMgKioqKioqL1xuXG5cdFx0Ly8gUm9vbSBjb25maWd1cmF0aW9uIHVwZGF0ZSBjb25maXJtYXRpb25cblx0XHRlbHNlIGlmIChjb21tYW5kID09IHRoaXMuUkVTUF9ST09NX0NPTkZJR19VUERBVEVfQ09ORklSTSlcblx0XHR7XG5cdFx0XHRpZiAoZGF0YS5nZXRVdGZTdHJpbmcoJ3JOYW1lJykgIT0gbnVsbClcblx0XHRcdFx0dGhpcy5fdXBkYXRlUm9vbU5hbWVJbkxpc3QoZGF0YS5nZXRJbnQoJ3pJZCcpLCBkYXRhLmdldEludCgncklkJyksIGRhdGEuZ2V0VXRmU3RyaW5nKCdyTmFtZScpKTtcblxuXHRcdFx0Ly8gSWYgdGhlIGN1cnJlbnQgdXNlciBpcyB0aGUgdXBkYXRlciwgc2hvdyBhIG5vdGlmaWNhdGlvbjsgb3RoZXJ3aXNlLCBzaG93IGEgZGlhbG9nIGJveCBzdWdnZXN0aW5nIHRvIHJlbG9hZFxuXHRcdFx0aWYgKHVzZXJuYW1lID09IHRoaXMuc21hcnRGb3gubXlTZWxmLm5hbWUpXG5cdFx0XHR7XG5cdFx0XHRcdC8vIEVuYWJsZSBpbnRlcmZhY2Vcblx0XHRcdFx0dGhpcy5fZW5hYmxlQ29uZmlnSW50ZXJmYWNlKHRydWUpO1xuXG5cdFx0XHRcdC8vIERpc3BsYXkgbm90aWZpY2F0aW9uXG5cdFx0XHRcdHRoaXMuc2hlbGxDdHJsLnNob3dOb3RpZmljYXRpb24oJ1Jvb20gbW9kaWZpZWQnLCBgUm9vbSBzZXR0aW5ncyB1cGRhdGVkIHN1Y2Nlc3NmdWxseTsgY2hhbmdlcyB3aWxsIGJlIGFwcGxpZWQgb24gbmV4dCA8c3Ryb25nPnNlcnZlciByZXN0YXJ0PC9zdHJvbmc+YCk7XG5cblx0XHRcdFx0Ly8gUmVzZXQgdGhlICdtb2RpZmllZCcgZmxhZ1xuXHRcdFx0XHR0aGlzLl9pbnRlcmZhY2VCdWlsZGVyLnJlc2V0SXNNb2RpZmllZCgpO1xuXHRcdFx0fVxuXHRcdFx0ZWxzZVxuXHRcdFx0e1xuXHRcdFx0XHQvLyBBbiBhbGVydCBib3ggaXMgZGlzcGxheWVkIGlmIHRoZSB1c2VyIGlzIGN1cnJlbnRseSBlZGl0aW5nIHRoZSBzYW1lIHJvb21cblx0XHRcdFx0aWYgKGRhdGEuZ2V0SW50KCdySWQnKSA9PSB0aGlzLl9lZGl0ZWRSb29tSWQpXG5cdFx0XHRcdHtcblx0XHRcdFx0XHQvLyBTaG93IGFsZXJ0XG5cdFx0XHRcdFx0dGhpcy5zaGVsbEN0cmwuc2hvd1NpbXBsZUFsZXJ0KGBBZG1pbmlzdHJhdG9yICR7dXNlcm5hbWV9IGhhcyBtb2RpZmllZCB0aGUgUm9vbSB5b3UgYXJlIGN1cnJlbnRseSBlZGl0aW5nOyBwbGVhc2UgcmVsb2FkIHRvIHVwZGF0ZSB5b3VyIHZpZXcuYCk7XG5cblx0XHRcdFx0XHQvLyBEaXNhYmxlIHN1Ym1pdCBidXR0b25cblx0XHRcdFx0XHQkKCcjem5jLXN1Ym1pdEJ1dHRvbicpLmF0dHIoJ2Rpc2FibGVkJywgdHJ1ZSk7XG5cdFx0XHRcdH1cblx0XHRcdFx0ZWxzZVxuXHRcdFx0XHR7XG5cdFx0XHRcdFx0Ly8gRGlzcGxheSBub3RpZmljYXRpb25cblx0XHRcdFx0XHRpZiAoZGF0YS5nZXRVdGZTdHJpbmcoJ3JOYW1lJykgIT0gbnVsbClcblx0XHRcdFx0XHRcdHRoaXMuc2hlbGxDdHJsLnNob3dOb3RpZmljYXRpb24oJ1Jvb20gcmVuYW1lZCcsIGBBZG1pbmlzdHJhdG9yICR7dXNlcm5hbWV9IGhhcyBjaGFuZ2VkIHRoZSBuYW1lIG9uIG9uZSBvZiB0aGUgUm9vbXNgKTtcblx0XHRcdFx0fVxuXHRcdFx0fVxuXHRcdH1cblxuXHRcdC8vIE5ldyByb29tIGFkZGVkXG5cdFx0ZWxzZSBpZiAoY29tbWFuZCA9PSB0aGlzLlJFU1BfUk9PTV9BRERFRClcblx0XHR7XG5cdFx0XHRjb25zdCByb29tRGF0YSA9IGRhdGEuZ2V0U0ZTT2JqZWN0KCdyb29tJyk7XG5cdFx0XHRjb25zdCB6b25lSWQgPSBkYXRhLmdldEludCgneklkJyk7XG5cblx0XHRcdGxldCB6b25lc0RTID0gdGhpcy5fdHJlZXZpZXcuZGF0YVNvdXJjZTtcblx0XHRcdGxldCB6b25lSXRlbSA9IHpvbmVzRFMuZ2V0KHpvbmVJZCk7XG5cblx0XHRcdC8vIElmIHRoZSBjdXJyZW50IHVzZXIgaXMgdGhlIHVwZGF0ZXIsIHJlc2V0IHRoZSBpbnRlcmZhY2U7IG90aGVyd2lzZSwganVzdCBzaG93IGEgbm90aWZpY2F0aW9uXG5cdFx0XHRpZiAodXNlcm5hbWUgPT0gdGhpcy5zbWFydEZveC5teVNlbGYubmFtZSlcblx0XHRcdHtcblx0XHRcdFx0Ly8gUmVzZXQgaW50ZXJmYWNlXG5cdFx0XHRcdHRoaXMuX29uQ2FuY2VsQ2xpY2soKTtcblxuXHRcdFx0XHQvLyBEaXNwbGF5IG5vdGlmaWNhdGlvblxuXHRcdFx0XHR0aGlzLnNoZWxsQ3RybC5zaG93Tm90aWZpY2F0aW9uKCdSb29tIGFkZGVkJywgYFJvb20gJyR7cm9vbURhdGEuZ2V0VXRmU3RyaW5nKCduYW1lJyl9JyBjcmVhdGVkIHN1Y2Nlc3NmdWxseWApO1xuXHRcdFx0fVxuXHRcdFx0ZWxzZVxuXHRcdFx0e1xuXHRcdFx0XHQvLyBEaXNwbGF5IG5vdGlmaWNhdGlvblxuXHRcdFx0XHR0aGlzLnNoZWxsQ3RybC5zaG93Tm90aWZpY2F0aW9uKCdSb29tIGFkZGVkJywgYEFkbWluaXN0cmF0b3IgJHt1c2VybmFtZX0gY3JlYXRlZCBSb29tICcke3Jvb21EYXRhLmdldFV0ZlN0cmluZygnbmFtZScpfScgaW4gWm9uZSAnJHt6b25lSXRlbS5uYW1lfSdgKTtcblx0XHRcdH1cblxuXHRcdFx0Ly8gQWRkIG5ldyByb29tIHRvIHRyZWVcblx0XHRcdHpvbmVJdGVtLmFwcGVuZCh0aGlzLl9jcmVhdGVSb29tT2JqZWN0KHJvb21EYXRhLCB6b25lSWQpKTtcblx0XHRcdHpvbmVzRFMuc3luYygpO1xuXG5cdFx0XHQvLyBFeHBhbmQgem9uZSBub2RlIHdoZXJlIHJvb20gd2FzIGFkZGVkXG5cdFx0XHR0aGlzLl90cmVldmlldy5leHBhbmQodGhpcy5fdHJlZXZpZXcuc2VsZWN0KCkpO1xuXHRcdH1cblxuXHRcdC8vIE5ldyByb29tIGNyZWF0aW9uIHJlZnVzZWQgZHVlIHRvIGludmFsaWQgcm9vbSBuYW1lXG5cdFx0ZWxzZSBpZiAoY29tbWFuZCA9PSB0aGlzLlJFU1BfUk9PTV9SRUZVU0VEKVxuXHRcdHtcblx0XHRcdC8vIFJlLWVuYWJsZSBpbnRlcmZhY2Vcblx0XHRcdHRoaXMuX2VuYWJsZUNvbmZpZ0ludGVyZmFjZSh0cnVlKTtcblxuXHRcdFx0Ly8gU2hvdyB3YXJuaW5nXG5cdFx0XHR0aGlzLnNoZWxsQ3RybC5zaG93U2ltcGxlQWxlcnQoJ1Jvb20gY29uZmlndXJhdGlvbiBjYW5cXCd0IGJlIHNhdmVkIGJlY2F1c2UgYW5vdGhlciBSb29tIHdpdGggdGhlIHNhbWUgbmFtZSBhbHJlYWR5IGV4aXN0cy4nLCB0cnVlKTtcblx0XHR9XG5cblx0XHQvLyBFeGlzdGluZyByb29tIGRlbGV0ZWRcblx0XHRlbHNlIGlmIChjb21tYW5kID09IHRoaXMuUkVTUF9ST09NX0RFTEVURUQpXG5cdFx0e1xuXHRcdFx0bGV0IHpvbmVJdGVtID0gdGhpcy5fZ2V0Wm9uZURhdGFJdGVtQnlJZChkYXRhLmdldEludCgneklkJykpO1xuXHRcdFx0bGV0IHJvb21JdGVtID0gdGhpcy5fZ2V0Um9vbURhdGFJdGVtQnlJZChkYXRhLmdldEludCgneklkJyksIGRhdGEuZ2V0SW50KCdySWQnKSk7XG5cblx0XHRcdC8vIElmIHRoZSBjdXJyZW50IHVzZXIgaXMgdGhlIGRlbGV0ZXIsIHJlc2V0IHRoZSBpbnRlcmZhY2U7IG90aGVyd2lzZSwganVzdCBzaG93IGEgbm90aWZpY2F0aW9uXG5cdFx0XHRpZiAodXNlcm5hbWUgPT0gdGhpcy5zbWFydEZveC5teVNlbGYubmFtZSlcblx0XHRcdHtcblx0XHRcdFx0Ly8gUmUtZW5hYmxlIGludGVyZmFjZVxuXHRcdFx0XHR0aGlzLl9lbmFibGVMaXN0SW50ZXJmYWNlKHRydWUpO1xuXG5cdFx0XHRcdC8vIERpc3BsYXkgbm90aWZpY2F0aW9uXG5cdFx0XHRcdHRoaXMuc2hlbGxDdHJsLnNob3dOb3RpZmljYXRpb24oJ1Jvb20gcmVtb3ZlZCcsIGBSb29tICcke3Jvb21JdGVtLm5hbWV9JyBkZWxldGVkIHN1Y2Nlc3NmdWxseWApO1xuXHRcdFx0fVxuXHRcdFx0ZWxzZVxuXHRcdFx0e1xuXHRcdFx0XHQvLyBBbiBhbGVydCBib3ggaXMgZGlzcGxheWVkIGlmIHRoZSB1c2VyIGlzIGN1cnJlbnRseSBlZGl0aW5nIHRoZSBzYW1lIHJvb21cblx0XHRcdFx0aWYgKGRhdGEuZ2V0SW50KCdySWQnKSA9PSB0aGlzLl9lZGl0ZWRSb29tSWQpXG5cdFx0XHRcdHtcblx0XHRcdFx0XHQvLyBTaG93IGFsZXJ0XG5cdFx0XHRcdFx0dGhpcy5zaGVsbEN0cmwuc2hvd1NpbXBsZUFsZXJ0KGBBZG1pbmlzdHJhdG9yICR7dXNlcm5hbWV9IGhhcyBkZWxldGVkIHRoZSBSb29tIHlvdSBhcmUgY3VycmVudGx5IG1vZGlmeWluZzsgeW91IGhhdmUgdG8gY2FuY2VsIHlvdXIgZWRpdGluZy5gKTtcblxuXHRcdFx0XHRcdC8vIERpc2FibGUgc3VibWl0IGFuZCByZWxvYWQgYnV0dG9uc1xuXHRcdFx0XHRcdCQoJyN6bmMtcmVsb2FkQnV0dG9uJykuYXR0cignZGlzYWJsZWQnLCB0cnVlKTtcblx0XHRcdFx0XHQkKCcjem5jLXN1Ym1pdEJ1dHRvbicpLmF0dHIoJ2Rpc2FibGVkJywgdHJ1ZSk7XG5cdFx0XHRcdH1cblx0XHRcdFx0ZWxzZVxuXHRcdFx0XHR7XG5cdFx0XHRcdFx0Ly8gRGlzcGxheSBub3RpZmljYXRpb25cblx0XHRcdFx0XHR0aGlzLnNoZWxsQ3RybC5zaG93Tm90aWZpY2F0aW9uKCdSb29tIHJlbW92ZWQnLCBgQWRtaW5pc3RyYXRvciAke3VzZXJuYW1lfSBkZWxldGVkIFJvb20gJyR7cm9vbUl0ZW0ubmFtZX0nIGZyb20gWm9uZSAnJHt6b25lSXRlbS5uYW1lfSdgKTtcblx0XHRcdFx0fVxuXHRcdFx0fVxuXG5cdFx0XHQvLyBSZXNldCBzZWxlY3Rpb24gaWYgdGhlIGN1cnJlbnRseSBzZWxlY3RlZCBpdGVtIG9yIGl0cyBwYXJlbnQgaXMgYmVpbmcgcmVtb3ZlZFxuXHRcdFx0bGV0IHNlbGVjdGVkTm9kZSA9IHRoaXMuX3RyZWV2aWV3LnNlbGVjdCgpO1xuXHRcdFx0bGV0IHNlbGVjdGVkRGF0YUl0ZW0gPSB0aGlzLl90cmVldmlldy5kYXRhSXRlbShzZWxlY3RlZE5vZGUpO1xuXHRcdFx0aWYgKHNlbGVjdGVkRGF0YUl0ZW0pXG5cdFx0XHR7XG5cdFx0XHRcdGlmIChzZWxlY3RlZERhdGFJdGVtLnR5cGUgPT0gdGhpcy5JVEVNX1RZUEVfUk9PTSAmJiBzZWxlY3RlZERhdGFJdGVtLmlkID09IGRhdGEuZ2V0SW50KCdySWQnKSlcblx0XHRcdFx0XHR0aGlzLl9kZXNlbGVjdFRyZWVJdGVtKCk7XG5cdFx0XHR9XG5cblx0XHRcdC8vIFJlbW92ZSByb29tIGZyb20gdHJlZVxuXHRcdFx0em9uZUl0ZW0uY2hpbGRyZW4ucmVtb3ZlKHJvb21JdGVtKTtcblx0XHRcdHRoaXMuX3RyZWV2aWV3LmRhdGFTb3VyY2Uuc3luYygpO1xuXHRcdH1cblxuXHRcdC8qKioqKiogV09SRFMgRklMRVMgKioqKioqL1xuXG5cdFx0Ly8gV29yZHMgZmlsZXMgbGlzdCByZWNlaXZlZFxuXHRcdGVsc2UgaWYgKGNvbW1hbmQgPT0gdGhpcy5SRVNQX1JFRlJFU0hfV09SRFNfRklMRVMpXG5cdFx0e1xuXHRcdFx0dGhpcy5fd29yZHNGaWxlc01hbmFnZXIucmVmcmVzaFdvcmRzRmlsZXNMaXN0KGRhdGEuZ2V0U0ZTQXJyYXkoJ3dmJyksIHVzZXJuYW1lID09IHRoaXMuc21hcnRGb3gubXlTZWxmLm5hbWUpO1xuXG5cdFx0XHQvLyBJZiBhbm90aGVyIHVzZXIgY2F1c2VkIGEgcmVmcmVzaCAoZm9yIGV4YW1wbGUgZGVsZXRpbmcgYSBmaWxlLCBvciBhZGRpbmcgYSBuZXcgb25lKSBzaG93IGEgbm90aWZpY2F0aW9uXG5cdFx0XHRpZiAodXNlcm5hbWUgIT0gbnVsbCAmJiB1c2VybmFtZSAhPSB0aGlzLnNtYXJ0Rm94Lm15U2VsZi5uYW1lICYmIHRoaXMuX2VkaXRlZFpvbmVJZCA+IC0xKVxuXHRcdFx0XHR0aGlzLnNoZWxsQ3RybC5zaG93Tm90aWZpY2F0aW9uKCdXb3JkcyBmaWxlcyBtb2RpZmllZCcsIGBBZG1pbmlzdHJhdG9yICR7dXNlcm5hbWV9IGhhcyBhZGRlZCwgbW9kaWZpZWQgb3IgZGVsZXRlZCBhIHdvcmRzIGZpbGUuYCk7XG5cdFx0fVxuXG5cdFx0Ly8gV29yZHMgZmlsZSBjb250ZW50IHJlY2VpdmVkXG5cdFx0ZWxzZSBpZiAoY29tbWFuZCA9PSB0aGlzLlJFU1BfV09SRFNfRklMRV9DT05URU5UKVxuXHRcdHtcblx0XHRcdHRoaXMuX3dvcmRzRmlsZXNNYW5hZ2VyLmVkaXRXb3Jkc0ZpbGUoZGF0YS5nZXRVdGZTdHJpbmcoJ2ZpbGVuYW1lJyksIGRhdGEuZ2V0VGV4dCgnY29udGVudCcpKTtcblx0XHR9XG5cblx0XHQvLyBXb3JkcyBmaWxlIGVycm9yIChlZGl0L3NhdmUpXG5cdFx0ZWxzZSBpZiAoY29tbWFuZCA9PSB0aGlzLlJFU1BfV09SRFNfRklMRV9FUlJPUilcblx0XHR7XG5cdFx0XHQvLyBFbmFibGUgYnV0dG9uc1xuXHRcdFx0dGhpcy5fd29yZHNGaWxlc01hbmFnZXIuZW5hYmxlZCA9IHRydWU7XG5cblx0XHRcdC8vIFNob3cgYWxlcnRcblx0XHRcdHRoaXMuc2hlbGxDdHJsLnNob3dTaW1wbGVBbGVydChkYXRhLmdldFV0ZlN0cmluZygnZXJyb3InKSwgdHJ1ZSk7XG5cdFx0fVxuXG5cdFx0Ly8gZWxzZSBpZiAoKVxuXHR9XG5cblx0Ly8tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS1cblx0Ly8gVUkgRVZFTlQgTElTVEVORVJTXG5cdC8vLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tXG5cblx0X29uVHJlZUl0ZW1Eb3VibGVDbGljayhlKVxuXHR7XG5cdFx0Ly8gR2V0IGV2ZW50IHRhcmdldCdzIGNsb3Nlc3QgdHJlZSBub2RlXG5cdFx0bGV0IHRyZWVOb2RlID0gJChlLnRhcmdldCkuY2xvc2VzdCgnLmstaXRlbVtyb2xlPXRyZWVpdGVtXScpO1xuXG5cdFx0Ly8gR2V0IGFzc29jaWF0ZWQgZGF0YSBpdGVtXG5cdFx0bGV0IGRhdGFJdGVtID0gdGhpcy5fdHJlZXZpZXcuZGF0YUl0ZW0odHJlZU5vZGUpO1xuXG5cdFx0Ly8gTG9hZCBjb25maWd1cmF0aW9uXG5cdFx0dGhpcy5fbG9hZENvbmZpZ3VyYXRpb24oZGF0YUl0ZW0udHlwZSk7XG5cdH1cblxuXHRfb25ab25lUm9vbUNoYW5nZSgpXG5cdHtcblx0XHQvLyBSZXNldCB1dGlsaXR5IGJ1dHRvbnNcblx0XHR0aGlzLl9zZXRVdGlsaXR5QnV0dG9uc1N0YXRlKHRoaXMuX3NlbGVjdGVkSXRlbSk7XG5cdH1cblxuXHQvLyBVdGlsaXR5IGJ1dHRvbnMgbGlzdGVuZXJzXG5cblx0X29uQWRkWm9uZUNsaWNrKClcblx0e1xuXHRcdC8vIERlc2VsZWN0IGxpc3QgaXRlbVxuXHRcdHRoaXMuX2Rlc2VsZWN0VHJlZUl0ZW0oKTtcblxuXHRcdC8vIExvYWQgY29uZmlndXJhdGlvblxuXHRcdHRoaXMuX2xvYWRDb25maWd1cmF0aW9uKHRoaXMuSVRFTV9UWVBFX1pPTkUpO1xuXHR9XG5cblx0X29uQWRkUm9vbUNsaWNrKClcblx0e1xuXHRcdC8vIFNlbGVjdCBwYXJlbnQgbGlzdCBpdGVtXG5cdFx0dGhpcy5fc2VsZWN0UGFyZW50VHJlZUl0ZW0oKTtcblxuXHRcdC8vIExvYWQgY29uZmlndXJhdGlvblxuXHRcdHRoaXMuX2xvYWRDb25maWd1cmF0aW9uKHRoaXMuSVRFTV9UWVBFX1JPT00pO1xuXHR9XG5cblx0X29uRWRpdENsaWNrKClcblx0e1xuXHRcdC8vIExvYWQgY29uZmlndXJhdGlvblxuXHRcdHRoaXMuX2xvYWRDb25maWd1cmF0aW9uKHRoaXMuX3NlbGVjdGVkSXRlbS50eXBlKTtcblx0fVxuXG5cdF9vblJlbW92ZUNsaWNrKClcblx0e1xuXHRcdHRoaXMuc2hlbGxDdHJsLnNob3dDb25maXJtV2FybmluZyhgQXJlIHlvdSBzdXJlIHlvdSB3YW50IHRvIGRlbGV0ZSB0aGUgc2VsZWN0ZWQgJHt0aGlzLl9zZWxlY3RlZEl0ZW0udHlwZSA9PSB0aGlzLklURU1fVFlQRV9aT05FID8gJ1pvbmUnIDogJ1Jvb20nfSBjb25maWd1cmF0aW9uP2AsICQucHJveHkodGhpcy5fb25SZW1vdmVDb25maXJtLCB0aGlzKSk7XG5cdH1cblxuXHRfb25SZW1vdmVDb25maXJtKClcblx0e1xuXHRcdC8vIERpc2FibGUgem9uZS9yb29tIHNlbGVjdGlvbiBsaXN0XG5cdFx0dGhpcy5fZW5hYmxlTGlzdEludGVyZmFjZShmYWxzZSk7XG5cblx0XHRsZXQgcGFyYW1zID0gbmV3IFNGUzJYLlNGU09iamVjdCgpO1xuXG5cdFx0Ly8gUmVxdWVzdCB6b25lIHJlbW92YWxcblx0XHRpZiAodGhpcy5fc2VsZWN0ZWRJdGVtLnR5cGUgPT0gdGhpcy5JVEVNX1RZUEVfWk9ORSlcblx0XHR7XG5cdFx0XHRwYXJhbXMucHV0SW50KCd6SWQnLCB0aGlzLl9zZWxlY3RlZEl0ZW0uaWQpO1xuXHRcdFx0dGhpcy5zZW5kRXh0ZW5zaW9uUmVxdWVzdCh0aGlzLlJFUV9ERUxFVEVfWk9ORV9DT05GSUcsIHBhcmFtcyk7XG5cdFx0fVxuXHRcdGVsc2Vcblx0XHR7XG5cdFx0XHRwYXJhbXMucHV0SW50KCd6SWQnLCB0aGlzLl9zZWxlY3RlZEl0ZW1QYXJlbnQuaWQpO1xuXHRcdFx0cGFyYW1zLnB1dEludCgncklkJywgdGhpcy5fc2VsZWN0ZWRJdGVtLmlkKTtcblx0XHRcdHRoaXMuc2VuZEV4dGVuc2lvblJlcXVlc3QodGhpcy5SRVFfREVMRVRFX1JPT01fQ09ORklHLCBwYXJhbXMpO1xuXHRcdH1cblx0fVxuXG5cdF9vbkFjdGl2YXRlQ2xpY2soKVxuXHR7XG5cdFx0Ly8gR2V0IHNlbGVjdGVkIGRhdGEgaXRlbVxuXHRcdGlmICh0aGlzLl9zZWxlY3RlZEl0ZW0udHlwZSA9PSB0aGlzLklURU1fVFlQRV9aT05FKVxuXHRcdHtcblx0XHRcdGxldCBwYXJhbXMgPSBuZXcgU0ZTMlguU0ZTT2JqZWN0KCk7XG5cdFx0XHRwYXJhbXMucHV0SW50KCd6SWQnLCB0aGlzLl9zZWxlY3RlZEl0ZW0uaWQpO1xuXG5cdFx0XHR0aGlzLnNlbmRFeHRlbnNpb25SZXF1ZXN0KHRoaXMuUkVRX0FDVElWQVRFX1pPTkUsIHBhcmFtcyk7XG5cdFx0fVxuXHR9XG5cblx0Ly8gQ29uZmlndXJhdGlvbiBidXR0b25zIGxpc3RlbmVyc1xuXG5cdF9vbkNhbmNlbENsaWNrKClcblx0e1xuXHRcdC8vIEVuYWJsZSB6b25lL3Jvb20gc2VsZWN0aW9uIGxpc3RzXG5cdFx0dGhpcy5fZW5hYmxlTGlzdEludGVyZmFjZSh0cnVlKTtcblxuXHRcdC8vIERpc2FibGUgY29uZmlndXJhdGlvbiBpbnRlcmZhY2Vcblx0XHR0aGlzLl9lbmFibGVDb25maWdJbnRlcmZhY2UoZmFsc2UpO1xuXG5cdFx0Ly8gQ2xlYXIgbWFpbiBjb250YWluZXJcblx0XHR0aGlzLl9yZXNldFRhYnNDb250YWluZXIoZmFsc2UsIHRydWUpO1xuXG5cdFx0Ly8gU2V0IGlzRWRpdGluZyBmbGFnXG5cdFx0dGhpcy5faXNFZGl0aW5nID0gZmFsc2U7XG5cdFx0dGhpcy5fZWRpdGVkSXRlbVR5cGUgPSAnJztcblxuXHRcdC8vIFN3aXRjaCBwYW5lbFxuXHRcdHRoaXMuX3N3aXRjaFBhbmVsKCd6bmMtc2lkZWJhclBhbmVsJyk7XG5cdH1cblxuXHRfb25SZWxvYWRDbGljaygpXG5cdHtcblx0XHQvLyBIaWRlIHZhbGlkYXRpb24gbWVzc2FnZXNcblx0XHR0aGlzLl9pbnRlcmZhY2VCdWlsZGVyLnJlc2V0VmFsaWRhdGlvbigpO1xuXG5cdFx0Ly8gUmVsb2FkIGNvbmZpZ3VyYXRpb25cblx0XHR0aGlzLl9sb2FkQ29uZmlndXJhdGlvbih0aGlzLl9lZGl0ZWRJdGVtVHlwZSwgZmFsc2UpO1xuXHR9XG5cblx0X29uU3VibWl0Q2xpY2soKVxuXHR7XG5cdFx0Ly8gQ2hlY2sgdmFsaWRpdHlcblx0XHRpZiAodGhpcy5faW50ZXJmYWNlQnVpbGRlci5jaGVja0lzVmFsaWQoKSlcblx0XHR7XG5cdFx0XHRsZXQgY2hhbmdlcyA9IHRoaXMuX2ludGVyZmFjZUJ1aWxkZXIuZ2V0Q2hhbmdlZERhdGEoKTtcblxuXHRcdFx0aWYgKGNoYW5nZXMuc2l6ZSgpID4gMClcblx0XHRcdHtcblx0XHRcdFx0Ly9jb25zb2xlLmxvZyhjaGFuZ2VzLmdldER1bXAoKSlcblx0XHRcdFx0Ly9yZXR1cm47XG5cblx0XHRcdFx0Ly8gSW4gY2FzZSB0aGUgem9uZS9yb29tIG5hbWUgY2hhbmdlZCwgY2hlY2sgaXQgYWdhaW5zdCB0aGUgbGlzdCAoZHVwbGljYXRlIG5hbWVzIG5vdCBhbGxvd2VkISlcblx0XHRcdFx0aWYgKHRoaXMuX3ZhbGlkYXRlTmFtZShjaGFuZ2VzKSlcblx0XHRcdFx0e1xuXHRcdFx0XHRcdC8vIERpc2FibGUgY29uZmlndXJhdGlvbiBpbnRlcmZhY2Vcblx0XHRcdFx0XHR0aGlzLl9lbmFibGVDb25maWdJbnRlcmZhY2UoZmFsc2UpO1xuXG5cdFx0XHRcdFx0Ly8gU2VuZCBzZXR0aW5ncyB0byBzZXJ2ZXIgaW5zdGFuY2Vcblx0XHRcdFx0XHRsZXQgcGFyYW1zID0gbmV3IFNGUzJYLlNGU09iamVjdCgpO1xuXHRcdFx0XHRcdHBhcmFtcy5wdXRTRlNBcnJheSgnc2V0dGluZ3MnLCBjaGFuZ2VzKTtcblx0XHRcdFx0XHRwYXJhbXMucHV0Qm9vbCgnYmFja3VwJywgJCgnI3puYy1iYWNrdXBDaGVjaycpLnByb3AoJ2NoZWNrZWQnKSk7XG5cdFx0XHRcdFx0cGFyYW1zLnB1dEludCgneklkJywgdGhpcy5fZWRpdGVkWm9uZUlkKTtcblx0XHRcdFx0XHRwYXJhbXMucHV0SW50KCdySWQnLCB0aGlzLl9lZGl0ZWRSb29tSWQpO1xuXG5cdFx0XHRcdFx0aWYgKHRoaXMuX2VkaXRlZEl0ZW1UeXBlID09IHRoaXMuSVRFTV9UWVBFX1pPTkUpXG5cdFx0XHRcdFx0e1xuXHRcdFx0XHRcdFx0Ly8gU3VibWl0IHpvbmUgc2V0dGluZ3Ncblx0XHRcdFx0XHRcdGlmICh0aGlzLl9lZGl0ZWRab25lSWQgPiAtMSlcblx0XHRcdFx0XHRcdFx0dGhpcy5zZW5kRXh0ZW5zaW9uUmVxdWVzdCh0aGlzLlJFUV9TQVZFX1pPTkVfQ09ORklHLCBwYXJhbXMpO1xuXHRcdFx0XHRcdFx0ZWxzZVxuXHRcdFx0XHRcdFx0XHR0aGlzLnNlbmRFeHRlbnNpb25SZXF1ZXN0KHRoaXMuUkVRX05FV19aT05FX0NPTkZJRywgcGFyYW1zKTtcblx0XHRcdFx0XHR9XG5cdFx0XHRcdFx0ZWxzZVxuXHRcdFx0XHRcdHtcblx0XHRcdFx0XHRcdC8vIFN1Ym1pdCByb29tIHNldHRpbmdzXG5cdFx0XHRcdFx0XHRpZiAodGhpcy5fZWRpdGVkUm9vbUlkID4gLTEpXG5cdFx0XHRcdFx0XHRcdHRoaXMuc2VuZEV4dGVuc2lvblJlcXVlc3QodGhpcy5SRVFfU0FWRV9ST09NX0NPTkZJRywgcGFyYW1zKTtcblx0XHRcdFx0XHRcdGVsc2Vcblx0XHRcdFx0XHRcdFx0dGhpcy5zZW5kRXh0ZW5zaW9uUmVxdWVzdCh0aGlzLlJFUV9ORVdfUk9PTV9DT05GSUcsIHBhcmFtcyk7XG5cdFx0XHRcdFx0fVxuXHRcdFx0XHR9XG5cdFx0XHRcdGVsc2Vcblx0XHRcdFx0e1xuXHRcdFx0XHRcdC8vIFNob3cgYWxlcnRcblx0XHRcdFx0XHR0aGlzLnNoZWxsQ3RybC5zaG93U2ltcGxlQWxlcnQoYFVuYWJsZSB0byBzdWJtaXQgY29uZmlndXJhdGlvbiBiZWNhdXNlIHRoZSAke2NhcGl0YWxpemVGaXJzdCh0aGlzLl9lZGl0ZWRJdGVtVHlwZSl9IG5hbWUgYWxyZWFkeSBleGlzdHM7IGR1cGxpY2F0ZSBuYW1lcyBhcmUgbm90IGFsbG93ZWQuYCwgdHJ1ZSk7XG5cdFx0XHRcdH1cblx0XHRcdH1cblx0XHR9XG5cdFx0ZWxzZVxuXHRcdHtcblx0XHRcdC8vIFNob3cgYWxlcnRcblx0XHRcdHRoaXMuc2hlbGxDdHJsLnNob3dTaW1wbGVBbGVydCgnVW5hYmxlIHRvIHN1Ym1pdCBjb25maWd1cmF0aW9uIGNoYW5nZXMgZHVlIHRvIGFuIGludmFsaWQgdmFsdWU7IHBsZWFzZSB2ZXJpZnkgdGhlIGhpZ2hsaWdodGVkIGZvcm0gZmllbGRzIGluIGFsbCB0YWJzLicsIHRydWUpO1xuXHRcdH1cblx0fVxuXG5cdF9vbkNvbmZpZ1ZhbHVlU2V0KGUpIC8vIFNBTUUgTUVUSE9EIERVUExJQ0FURUQgSU4gem9uZS1tb25pdG9yLmpzXG5cdHtcblx0XHRjb25zdCBjb25maWdQYXJhbSA9IGUudGFyZ2V0LmRhdGE7XG5cblx0XHQvLyBIYW5kbGUgZXh0ZW5zaW9uIG5hbWUvdHlwZSBkcm9wZG93bnMgdXBkYXRlIGFuZCB1cGRhdGUgdGhlIG1haW4gY2xhc3MgZHJvcGRvd24gZGF0YXNvdXJjZSBhY2NvcmRpbmdseVxuXHRcdGlmIChjb25maWdQYXJhbS5uYW1lID09ICdleHRlbnNpb24ubmFtZScgfHwgY29uZmlnUGFyYW0ubmFtZSA9PSAnZXh0ZW5zaW9uLnR5cGUnIHx8IGNvbmZpZ1BhcmFtLm5hbWUgPT0gJ2V4dGVuc2lvbi5maWx0ZXJDbGFzcycpXG5cdFx0e1xuXHRcdFx0Ly8gQWxsIGludm9sdmVkIENvbmZpZ0Zvcm1JdGVtcyBtdXN0IGJlIGF2YWlsYWJsZSBhbmQgaW5pdGlhbGl6ZWQgdG8gcHJvY2VlZFxuXHRcdFx0Y29uc3QgbmFtZUZvcm1JdGVtID0gdGhpcy5faW50ZXJmYWNlQnVpbGRlci5nZXRDb25maWdGb3JtSXRlbSgnZXh0ZW5zaW9uLm5hbWUnKTtcblx0XHRcdGNvbnN0IHR5cGVGb3JtSXRlbSA9IHRoaXMuX2ludGVyZmFjZUJ1aWxkZXIuZ2V0Q29uZmlnRm9ybUl0ZW0oJ2V4dGVuc2lvbi50eXBlJyk7XG5cdFx0XHRjb25zdCBjbGFzc0Zvcm1JdGVtID0gdGhpcy5faW50ZXJmYWNlQnVpbGRlci5nZXRDb25maWdGb3JtSXRlbSgnZXh0ZW5zaW9uLmZpbGUnKTtcblx0XHRcdGNvbnN0IGZpbHRlckZvcm1JdGVtID0gdGhpcy5faW50ZXJmYWNlQnVpbGRlci5nZXRDb25maWdGb3JtSXRlbSgnZXh0ZW5zaW9uLmZpbHRlckNsYXNzJyk7XG5cblx0XHRcdGlmIChuYW1lRm9ybUl0ZW0gIT0gbnVsbCAmJiB0eXBlRm9ybUl0ZW0gIT0gbnVsbCAmJiBjbGFzc0Zvcm1JdGVtICE9IG51bGwgJiYgZmlsdGVyRm9ybUl0ZW0gIT0gbnVsbClcblx0XHRcdHtcblx0XHRcdFx0Y29uc3Qgc291cmNlID0gbmFtZUZvcm1JdGVtLmRhdGE7XG5cdFx0XHRcdGxldCBjbGFzc2VzTGlzdCA9IFtdO1xuXG5cdFx0XHRcdGxldCBkYXRhID0gc291cmNlLnRyaWdnZXJEYXRhO1xuXHRcdFx0XHRmb3IgKGxldCBpID0gMDsgaSA8IGRhdGEuc2l6ZSgpOyBpKyspXG5cdFx0XHRcdHtcblx0XHRcdFx0XHRsZXQgZXh0ID0gZGF0YS5nZXRTRlNPYmplY3QoaSk7XG5cblx0XHRcdFx0XHRpZiAoZXh0LmdldFV0ZlN0cmluZygnbmFtZScpID09IG5hbWVGb3JtSXRlbS5kYXRhLnZhbHVlICYmIGV4dC5nZXRVdGZTdHJpbmcoJ3R5cGUnKSA9PSB0eXBlRm9ybUl0ZW0uZGF0YS52YWx1ZSlcblx0XHRcdFx0XHR7XG5cdFx0XHRcdFx0XHRsZXQgY2xhc3NlcyA9IGV4dC5nZXQoJ2NsYXNzZXNTdHJpbmcnKS5zcGxpdCgnLCcpOyAvLyBXZSBkb24ndCB1c2UgXCJnZXRVVGZTdHJpbmdcIiBiZWNhdXNlIHRoZSB0eXBlIGlzIFRleHQgaW4gY2FzZSBhIHZlcnkgbGFyZ2UgbGlzdCBvZiBjbGFzc2VzIGlzIHJldHVybmVkXG5cblx0XHRcdFx0XHRcdGlmIChmaWx0ZXJGb3JtSXRlbS5kYXRhLnZhbHVlID09IHRydWUpXG5cdFx0XHRcdFx0XHR7XG5cdFx0XHRcdFx0XHRcdGxldCBmaWx0ZXJlZENsYXNzZXMgPSBjbGFzc2VzLmZpbHRlcihmaWx0ZXJDbGFzc05hbWUpO1xuXHRcdFx0XHRcdFx0XHRjbGFzc2VzID0gZmlsdGVyZWRDbGFzc2VzO1xuXHRcdFx0XHRcdFx0fVxuXG5cdFx0XHRcdFx0XHRjbGFzc2VzTGlzdCA9IGNsYXNzZXNMaXN0LmNvbmNhdChjbGFzc2VzKTtcblx0XHRcdFx0XHR9XG5cdFx0XHRcdH1cblxuXHRcdFx0XHRsZXQgY3VycmVudENsYXNzID0gY2xhc3NGb3JtSXRlbS5kYXRhLnZhbHVlO1xuXG5cdFx0XHRcdC8vIElmIHRoZSBjbGFzc2VzIGxpc3QgZG9lc24ndCBjb250YWluIHRoZSBjdXJyZW50IHZhbHVlLCBjcmVhdGUgYW4gZW1wdHkgZW50cnkgYW5kIHJlc2V0IHRoZSB2YWx1ZVxuXHRcdFx0XHRpZiAoY2xhc3Nlc0xpc3QuaW5kZXhPZihjdXJyZW50Q2xhc3MpIDwgMClcblx0XHRcdFx0e1xuXHRcdFx0XHRcdGlmIChjbGFzc2VzTGlzdC5sZW5ndGggPT0gMClcblx0XHRcdFx0XHR7XG5cdFx0XHRcdFx0XHRjbGFzc2VzTGlzdC5wdXNoKCcnKTtcblx0XHRcdFx0XHRcdGN1cnJlbnRDbGFzcyA9ICcnO1xuXHRcdFx0XHRcdH1cblx0XHRcdFx0XHRlbHNlXG5cdFx0XHRcdFx0XHRjdXJyZW50Q2xhc3MgPSBjbGFzc2VzTGlzdFswXTtcblx0XHRcdFx0fVxuXG5cdFx0XHRcdGxldCBtYWluQ2xhc3NEcm9wRG93biA9IGNsYXNzRm9ybUl0ZW0uX2lubmVyV2lkZ2V0O1xuXHRcdFx0XHRtYWluQ2xhc3NEcm9wRG93bi5zZXREYXRhU291cmNlKGNsYXNzZXNMaXN0KTtcblxuXHRcdFx0XHRjbGFzc0Zvcm1JdGVtLmRhdGEudmFsdWUgPSBjdXJyZW50Q2xhc3M7XG5cdFx0XHRcdGNsYXNzRm9ybUl0ZW0uX3NldFdpZGdldFZhbHVlKCk7XG5cdFx0XHR9XG5cdFx0fVxuXHR9XG5cblx0X29uV29yZHNGaWxlUmVsb2FkQ2xpY2soZXZ0KVxuXHR7XG5cdFx0Ly8gU2VuZCByZXF1ZXN0IHRvIHNlcnZlclxuXHRcdHRoaXMuc2VuZEV4dGVuc2lvblJlcXVlc3QodGhpcy5SRVFfUkVGUkVTSF9XT1JEU19GSUxFKTtcblx0fVxuXG5cdF9vbldvcmRzRmlsZUVkaXRDbGljayhldnQpXG5cdHtcblx0XHQvLyBTZW5kIHJlcXVlc3QgdG8gc2VydmVyXG5cdFx0bGV0IHBhcmFtcyA9IG5ldyBTRlMyWC5TRlNPYmplY3QoKTtcblx0XHRwYXJhbXMucHV0VXRmU3RyaW5nKCdmaWxlbmFtZScsIGV2dC5kZXRhaWwpO1xuXHRcdHRoaXMuc2VuZEV4dGVuc2lvblJlcXVlc3QodGhpcy5SRVFfRURJVF9XT1JEU19GSUxFLCBwYXJhbXMpO1xuXHR9XG5cblx0X29uV29yZHNGaWxlU2F2ZUNsaWNrKGV2dClcblx0e1xuXHRcdHRoaXMuX3RlbXBXb3Jkc0ZpbGVEYXRhID0gZXZ0LmRldGFpbDtcblxuXHRcdC8vIENoZWNrIGlmIGEgbmV3IGZpbGUgaXMgYmVpbmcgY3JlYXRlZFxuXHRcdGlmICh0aGlzLl90ZW1wV29yZHNGaWxlRGF0YS5pc05ldylcblx0XHR7XG5cdFx0XHQvLyBJZiB5ZXMsIGNoZWNrIGlmIG5hbWUgYWxyZWFkeSBleGlzdHNcblx0XHRcdGlmICh0aGlzLl93b3Jkc0ZpbGVzTWFuYWdlci5nZXRFeGlzdGluZ0ZpbGVuYW1lcygpLmluY2x1ZGVzKHRoaXMuX3RlbXBXb3Jkc0ZpbGVEYXRhLmZpbGVuYW1lKSlcblx0XHRcdHtcblx0XHRcdFx0Ly8gU2hvdyBjb25maXJtIGRpYWxvZ1xuXHRcdFx0XHR0aGlzLnNoZWxsQ3RybC5zaG93Q29uZmlybVdhcm5pbmcoJ0Egd29yZHMgZmlsZSB3aXRoIHRoZSBlbnRlcmVkIG5hbWUgYWxyZWFkeSBleGlzdHM7IGRvIHlvdSB3YW50IHRvIHByb2NlZWQgYW55d2F5PyBUaGUgZXhpc3RpbmcgZmlsZSB3aWxsIGJlIG92ZXJ3cml0dGVuLicsICQucHJveHkodGhpcy5fb25Xb3Jkc0ZpbGVTYXZlQ29uZmlybSwgdGhpcykpO1xuXHRcdFx0XHRyZXR1cm47XG5cdFx0XHR9XG5cdFx0fVxuXG5cdFx0Ly8gUHJvY2VlZFxuXHRcdHRoaXMuX29uV29yZHNGaWxlU2F2ZUNvbmZpcm0oKTtcblx0fVxuXG5cdF9vbldvcmRzRmlsZVNhdmVDb25maXJtKClcblx0e1xuXHRcdC8vIERpc2FibGUgd29yZHMgZmlsZXMgbWFuYWdlciBidXR0b25zXG5cdFx0dGhpcy5fd29yZHNGaWxlc01hbmFnZXIuZW5hYmxlZCA9IGZhbHNlO1xuXHRcdHRoaXMuX3dvcmRzRmlsZXNNYW5hZ2VyLnNhdmVTcGlubmVyVmlzaWJsZSA9IHRydWU7XG5cblx0XHQvLyBTZW5kIHJlcXVlc3QgdG8gc2VydmVyXG5cdFx0bGV0IHBhcmFtcyA9IG5ldyBTRlMyWC5TRlNPYmplY3QoKTtcblx0XHRwYXJhbXMucHV0VXRmU3RyaW5nKCdmaWxlbmFtZScsIHRoaXMuX3RlbXBXb3Jkc0ZpbGVEYXRhLmZpbGVuYW1lKTtcblx0XHRwYXJhbXMucHV0VGV4dCgnY29udGVudCcsIHRoaXMuX3RlbXBXb3Jkc0ZpbGVEYXRhLmNvbnRlbnQpO1xuXHRcdHRoaXMuc2VuZEV4dGVuc2lvblJlcXVlc3QodGhpcy5SRVFfU0FWRV9XT1JEU19GSUxFLCBwYXJhbXMpO1xuXHR9XG5cblx0X29uV29yZHNGaWxlUmVtb3ZlQ2xpY2soZXZ0KVxuXHR7XG5cdFx0dGhpcy5zaGVsbEN0cmwuc2hvd0NvbmZpcm1XYXJuaW5nKCdBcmUgeW91IHN1cmUgeW91IHdhbnQgdG8gZGVsZXRlIHRoZSBzZWxlY3RlZCB3b3JkcyBmaWxlPycsICQucHJveHkodGhpcy5fb25Xb3Jkc0ZpbGVSZW1vdmVDb25maXJtLCB0aGlzKSk7XG5cdH1cblxuXHRfb25Xb3Jkc0ZpbGVSZW1vdmVDb25maXJtKClcblx0e1xuXHRcdGxldCB3b3Jkc0ZpbGUgPSB0aGlzLl93b3Jkc0ZpbGVzTWFuYWdlci5nZXRTZWxlY3RlZFdvcmRzRmlsZU5hbWUoKTtcblxuXHRcdGlmICh3b3Jkc0ZpbGUgIT0gbnVsbClcblx0XHR7XG5cdFx0XHQvLyBEaXNhYmxlIHdvcmRzIGZpbGVzIG1hbmFnZXIgYnV0dG9uc1xuIFx0XHRcdHRoaXMuX3dvcmRzRmlsZXNNYW5hZ2VyLmVuYWJsZWQgPSBmYWxzZTtcblx0XHRcdHRoaXMuX3dvcmRzRmlsZXNNYW5hZ2VyLmFjdGlvblNwaW5uZXJWaXNpYmxlID0gdHJ1ZTtcblxuXHRcdFx0Ly8gU2VuZCByZXF1ZXN0IHRvIHNlcnZlclxuXHRcdFx0bGV0IHBhcmFtcyA9IG5ldyBTRlMyWC5TRlNPYmplY3QoKTtcblx0XHRcdHBhcmFtcy5wdXRVdGZTdHJpbmcoJ2ZpbGVuYW1lJywgd29yZHNGaWxlKTtcblx0XHRcdHRoaXMuc2VuZEV4dGVuc2lvblJlcXVlc3QodGhpcy5SRVFfREVMRVRFX1dPUkRTX0ZJTEUsIHBhcmFtcyk7XG5cdFx0fVxuXHR9XG5cblx0X29uV29yZHNGaWxlQXNzaWduQ2xpY2soZXZ0KVxuXHR7XG5cdFx0bGV0IHBhdGggPSBldnQuZGV0YWlsO1xuXG5cdFx0Ly8gV3JpdGUgcGF0aCBvZiB0aGUgc2VsZWN0ZWQgd29yZHMgZmlsZSBpbiBcIndvcmRzRmlsdGVyLndvcmRzRmlsZVwiIGR5bmFtaWNhbGx5IGNyZWF0ZWQgZmllbGRcblx0XHRjb25zdCB3b3Jkc0ZpbGVGb3JtSXRlbSA9IHRoaXMuX2ludGVyZmFjZUJ1aWxkZXIuZ2V0Q29uZmlnRm9ybUl0ZW0oJ3dvcmRzRmlsdGVyLndvcmRzRmlsZScpO1xuXHRcdHdvcmRzRmlsZUZvcm1JdGVtLmRhdGEudmFsdWUgPSBwYXRoO1xuXHRcdHdvcmRzRmlsZUZvcm1JdGVtLl9zZXRXaWRnZXRWYWx1ZSgpO1xuXHR9XG5cblx0Ly8tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS1cblx0Ly8gUFJJVkFURSBNRVRIT0RTXG5cdC8vLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tXG5cblx0X2VuYWJsZUxpc3RJbnRlcmZhY2UoZW5hYmxlZClcblx0e1xuXHRcdCQoJyN6bmMtdXRpbEJ1dHRvbnMnKS5hdHRyKCdkaXNhYmxlZCcsICFlbmFibGVkKTtcblx0XHQkKCcjem5jLXRyZWVWaWV3JykuYXR0cignZGlzYWJsZWQnLCAhZW5hYmxlZCk7XG5cdH1cblxuXHRfc2V0VXRpbGl0eUJ1dHRvbnNTdGF0ZShkYXRhSXRlbSA9IG51bGwpXG5cdHtcblx0XHRsZXQgZGlzYWJsZSA9IHRydWU7XG5cblx0XHRpZiAoZGF0YUl0ZW0pXG5cdFx0e1xuXHRcdFx0Ly8gRW5hYmxlICdhY3RpdmF0ZSB6b25lJyBidXR0b24gaWYgem9uZSBpcyBpbmFjdGl2ZVxuXHRcdFx0JCgnI3puYy1hY3RpdmF0ZUJ1dHRvbicpLmF0dHIoJ2Rpc2FibGVkJywgKGRhdGFJdGVtLnR5cGUgIT0gdGhpcy5JVEVNX1RZUEVfWk9ORSB8fCBkYXRhSXRlbS5hY3RpdmUpKTtcblxuXHRcdFx0ZGlzYWJsZSA9IGZhbHNlO1xuXHRcdH1cblx0XHRlbHNlXG5cdFx0e1xuXHRcdFx0Ly8gRGlzYWJsZSAnYWN0aXZhdGUgem9uZScgYnV0dG9uXG5cdFx0XHQkKCcjem5jLWFjdGl2YXRlQnV0dG9uJykuYXR0cignZGlzYWJsZWQnLCB0cnVlKTtcblx0XHR9XG5cblx0XHQvLyBFbmFibGUvZGlzYWJsZSBvdGhlciB1dGlsaXR5IGJ1dHRvbnNcblx0XHQkKCcjem5jLWFkZFpvbmVCdXR0b24nKS5hdHRyKCdkaXNhYmxlZCcsIGZhbHNlKTsgLy8gQWx3YXlzIGVuYWJsZWRcblx0XHQkKCcjem5jLWFkZFJvb21CdXR0b24nKS5hdHRyKCdkaXNhYmxlZCcsIGRpc2FibGUpO1xuXHRcdCQoJyN6bmMtZWRpdEJ1dHRvbicpLmF0dHIoJ2Rpc2FibGVkJywgZGlzYWJsZSk7XG5cdFx0JCgnI3puYy1yZW1vdmVCdXR0b24nKS5hdHRyKCdkaXNhYmxlZCcsIGRpc2FibGUpO1xuXHR9XG5cblx0X2VuYWJsZUNvbmZpZ0ludGVyZmFjZShlbmFibGVkKVxuXHR7XG5cdFx0JCgnI3puYy1jYW5jZWxCdXR0b24nKS5hdHRyKCdkaXNhYmxlZCcsICFlbmFibGVkKTtcblx0XHQkKCcjem5jLXJlbG9hZEJ1dHRvbicpLmF0dHIoJ2Rpc2FibGVkJywgIWVuYWJsZWQpO1xuXHRcdCQoJyN6bmMtc3VibWl0QnV0dG9uJykuYXR0cignZGlzYWJsZWQnLCAhZW5hYmxlZCk7XG5cdFx0JCgnI3puYy1iYWNrdXBDaGVjaycpLmF0dHIoJ2Rpc2FibGVkJywgIWVuYWJsZWQpO1xuXG5cdFx0dGhpcy5faW50ZXJmYWNlQnVpbGRlci5kaXNhYmxlSW50ZXJmYWNlKCFlbmFibGVkKTtcblxuXHRcdC8vIEFsc28gc3dpdGNoIHZpZXcgd2hlbiBlbmFibGVkXG5cdFx0aWYgKGVuYWJsZWQpXG5cdFx0XHR0aGlzLl9zd2l0Y2hWaWV3KCd6bmMtbWFpbicpO1xuXHR9XG5cblx0X3N3aXRjaFZpZXcodmlld0lkKVxuXHR7XG5cdFx0ZG9jdW1lbnQuZ2V0RWxlbWVudEJ5SWQoJ3puYy12aWV3c3RhY2snKS5zZWxlY3RlZEVsZW1lbnQgPSBkb2N1bWVudC5nZXRFbGVtZW50QnlJZCh2aWV3SWQpO1xuXHR9XG5cblx0X2NsZWFyVGFicygpXG5cdHtcblx0XHQvLyBEZXN0cm95IHNjcm9sbGluZyB0YWJzXG5cdFx0JCgnI3puYy10YWJOYXZpZ2F0b3IgI3RhYnMnKS5zY3JvbGxpbmdUYWJzKCdkZXN0cm95Jyk7XG5cblx0XHQvLyBSZW1vdmUgYWxsIHRhYiBuYXZpZ2F0b3IgY29udGVudFxuXHRcdHRoaXMuX2ludGVyZmFjZUJ1aWxkZXIuZGVzdHJveUludGVyZmFjZSgpO1xuXG5cdFx0Ly8gU2V0IGZsYWcgdG8gcmUtaW5pdGlhbGl6ZSB0YWJzIGlmIG5lZWRlZFxuXHRcdHRoaXMuX3JlaW5pdFRhYnMgPSB0cnVlO1xuXHR9XG5cblx0X3BvcHVsYXRlVHJlZShkYXRhKVxuXHR7XG5cdFx0bGV0IHpEYXRhID0gZGF0YS5nZXRTRlNBcnJheSgnem9uZXMnKTtcblxuXHRcdGxldCB6b25lc0FyciA9IFtdO1xuXHRcdGZvciAobGV0IHogPSAwOyB6IDwgekRhdGEuc2l6ZSgpOyB6KyspXG5cdFx0XHR6b25lc0Fyci5wdXNoKCB0aGlzLl9jcmVhdGVab25lT2JqZWN0KHpEYXRhLmdldFNGU09iamVjdCh6KSkgKTtcblxuXHRcdC8vIENyZWF0ZSBkYXRhc291cmNlXG5cdFx0bGV0IHpvbmVzID0gbmV3IGtlbmRvLmRhdGEuSGllcmFyY2hpY2FsRGF0YVNvdXJjZSh7XG4gICAgICAgICAgICBkYXRhOiB6b25lc0Fycixcblx0XHRcdHNvcnQ6IHtcblx0XHRcdFx0ZmllbGQ6ICduYW1lJyxcblx0XHRcdFx0ZGlyOiAnYXNjJ1xuXHRcdFx0fSxcbiAgICAgICAgICAgIHNjaGVtYToge1xuICAgICAgICAgICAgICAgIG1vZGVsOiB7XG5cdFx0XHRcdFx0aWQ6ICdpZCcsXG4gICAgICAgICAgICAgICAgICAgIGNoaWxkcmVuOiB7XG5cdFx0XHRcdFx0XHRzY2hlbWE6IHtcblx0XHRcdFx0XHRcdFx0ZGF0YTogJ3Jvb21zJyxcblx0XHRcdFx0XHRcdFx0c29ydDoge1xuXHRcdFx0XHRcdFx0XHRcdGZpZWxkOiAnbmFtZScsXG5cdFx0XHRcdFx0XHRcdFx0ZGlyOiAnYXNjJ1xuXHRcdFx0XHRcdFx0XHR9XG5cdFx0XHRcdFx0XHR9XG5cdFx0XHRcdFx0fVxuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgIH1cbiAgICAgICAgfSk7XG5cblx0XHQvLyBTZXQgdHJlZSB2aWV3IGRhdGFwcm92aWRlclxuXHRcdHRoaXMuX3RyZWV2aWV3LnNldERhdGFTb3VyY2Uoem9uZXMpO1xuXG5cdFx0Ly8gU2V0IHV0aWxpdHkgYnV0dG9ucyBzdGF0ZSAoYWRkLCByZW1vdmUsIGVkaXQsIGV0Yylcblx0XHR0aGlzLl9zZXRVdGlsaXR5QnV0dG9uc1N0YXRlKCk7XG5cdH1cblxuXHRfY3JlYXRlWm9uZU9iamVjdCh6b25lRGF0YSlcblx0e1xuXHRcdGxldCB6b25lID0ge1xuXHRcdFx0dHlwZTogdGhpcy5JVEVNX1RZUEVfWk9ORSxcblx0XHRcdG5hbWU6IHpvbmVEYXRhLmdldFV0ZlN0cmluZygnbmFtZScpLFxuXHRcdFx0aWQ6IHpvbmVEYXRhLmdldEludCgnaWQnKSxcblx0XHRcdGFjdGl2ZTogem9uZURhdGEuZ2V0Qm9vbCgnYWN0Jylcblx0XHR9XG5cblx0XHQvLyBDcmVhdGUgcm9vbXMgbGlzdCBkYXRhcHJvdmlkZXJcblx0XHRsZXQgckRhdGEgPSB6b25lRGF0YS5nZXRTRlNBcnJheSgncm9vbXMnKTtcblxuXHRcdGxldCByb29tc0FyciA9IFtdO1xuXHRcdGZvciAobGV0IHIgPSAwOyByIDwgckRhdGEuc2l6ZSgpOyByKyspXG5cdFx0XHRyb29tc0Fyci5wdXNoKCB0aGlzLl9jcmVhdGVSb29tT2JqZWN0KHJEYXRhLmdldFNGU09iamVjdChyKSwgem9uZURhdGEuZ2V0SW50KCdpZCcpKSApO1xuXG5cdFx0em9uZS5yb29tcyA9IHJvb21zQXJyO1xuXG5cdFx0cmV0dXJuIHpvbmU7XG5cdH1cblxuXHRfY3JlYXRlUm9vbU9iamVjdChyb29tRGF0YSwgem9uZUlkKVxuXHR7XG5cdFx0bGV0IHJvb20gPSB7XG5cdFx0XHR0eXBlOiB0aGlzLklURU1fVFlQRV9ST09NLFxuXHRcdFx0bmFtZTogcm9vbURhdGEuZ2V0VXRmU3RyaW5nKCduYW1lJyksXG5cdFx0XHRpZDogcm9vbURhdGEuZ2V0SW50KCdpZCcpLFxuXHRcdFx0YWN0aXZlOiByb29tRGF0YS5nZXRCb29sKCdhY3QnKSxcblx0XHRcdHBhcmVudElkOiB6b25lSWQsXG5cdFx0XHRzcHJpdGVDc3NDbGFzczogdGhpcy5fZ2V0Um9vbUxpc3RJY29uQ3NzQ2xhc3Mocm9vbURhdGEuZ2V0Qm9vbCgnYWN0JykpXG5cdFx0fTtcblxuXHRcdHJldHVybiByb29tO1xuXHR9XG5cblx0X2dldFJvb21MaXN0SWNvbkNzc0NsYXNzKGlzQWN0aXZlKVxuXHR7XG5cdFx0cmV0dXJuIGlzQWN0aXZlID8gJ2ZhcyBmYS1kb29yLW9wZW4nIDogJ2ZhcyBmYS1kb29yLWNsb3NlZCBpbmFjdGl2ZS1saXN0LWl0ZW0nO1xuXHR9XG5cblx0X3NldFpvbmVBY3RpdmF0aW9uU3RhdHVzKHpvbmVJZCwgYWN0aXZlUm9vbXMsIGlzQWN0aXZlKVxuXHR7XG5cdFx0bGV0IHpvbmVESSA9IHRoaXMuX2dldFpvbmVEYXRhSXRlbUJ5SWQoem9uZUlkKTtcblxuXHRcdHpvbmVESS5hY3RpdmUgPSBpc0FjdGl2ZTtcblxuXHRcdGxldCBhY3RpdmVSb29tc0FyciA9IGFjdGl2ZVJvb21zLnNwbGl0KCcsJyk7XG5cblx0XHRpZiAoem9uZURJLmhhc0NoaWxkcmVuKVxuXHRcdHtcblx0XHRcdGZvciAobGV0IGkgPSAwOyBpIDwgem9uZURJLmNoaWxkcmVuLmRhdGEoKS5sZW5ndGg7IGkrKylcblx0XHRcdHtcblx0XHRcdFx0bGV0IHJvb20gPSB6b25lREkuY2hpbGRyZW4uZGF0YSgpW2ldO1xuXHRcdFx0XHRyb29tLmFjdGl2ZSA9IChpc0FjdGl2ZSAmJiBhY3RpdmVSb29tc0Fyci5pbmRleE9mKHJvb20ubmFtZSkgPiAtMSk7XG5cdFx0XHRcdHJvb20uc3ByaXRlQ3NzQ2xhc3MgPSB0aGlzLl9nZXRSb29tTGlzdEljb25Dc3NDbGFzcyhyb29tLmFjdGl2ZSlcblx0XHRcdH1cblx0XHR9XG5cblx0XHQvLyBSZWZyZXNoIGxpc3Rcblx0XHR0aGlzLl90cmVldmlldy5kYXRhU291cmNlLnN5bmMoKTtcblxuXHRcdC8vIFJldHVybiB6b25lIG5hbWVcblx0XHRyZXR1cm4gem9uZURJLm5hbWU7XG5cdH1cblxuXHRfZGVzZWxlY3RUcmVlSXRlbSgpXG5cdHtcblx0XHR0aGlzLl90cmVldmlldy5zZWxlY3QoJCgpKTtcblx0fVxuXG5cdF9zZWxlY3RQYXJlbnRUcmVlSXRlbSgpXG5cdHtcblx0XHRsZXQgc2VsZWN0ZWROb2RlID0gdGhpcy5fdHJlZXZpZXcuc2VsZWN0KCk7XG5cdFx0bGV0IHNlbGVjdGVkRGF0YUl0ZW0gPSB0aGlzLl90cmVldmlldy5kYXRhSXRlbShzZWxlY3RlZE5vZGUpO1xuXG5cdFx0aWYgKHNlbGVjdGVkRGF0YUl0ZW0udHlwZSA9PSB0aGlzLklURU1fVFlQRV9ST09NKVxuXHRcdHtcblx0XHRcdGxldCBwYXJlbnROb2RlID0gdGhpcy5fdHJlZXZpZXcucGFyZW50KHNlbGVjdGVkTm9kZSk7XG5cdFx0XHR0aGlzLl90cmVldmlldy5zZWxlY3QocGFyZW50Tm9kZSk7XG5cdFx0fVxuXHR9XG5cblx0X2xvYWRDb25maWd1cmF0aW9uKHR5cGUsIHJlc2V0VGFicyA9IHRydWUpXG5cdHtcblx0XHQvLyBEaXNhYmxlIHpvbmUvcm9vbSBzZWxlY3Rpb24gbGlzdFxuXHRcdHRoaXMuX2VuYWJsZUxpc3RJbnRlcmZhY2UoZmFsc2UpO1xuXG5cdFx0Ly8gRGlzYWJsZSBjb25maWd1cmF0aW9uIGludGVyZmFjZVxuXHRcdHRoaXMuX2VuYWJsZUNvbmZpZ0ludGVyZmFjZShmYWxzZSk7XG5cblx0XHQvLyBDbGVhciBtYWluIGNvbnRhaW5lclxuXHRcdHRoaXMuX3Jlc2V0VGFic0NvbnRhaW5lcih0cnVlLCByZXNldFRhYnMpO1xuXG5cdFx0Ly8gU2V0IGlzRWRpdGluZyBmbGFnXG5cdFx0dGhpcy5faXNFZGl0aW5nID0gdHJ1ZTtcblx0XHR0aGlzLl9lZGl0ZWRJdGVtVHlwZSA9IHR5cGU7XG5cblx0XHQvLyBSZXF1ZXN0IHpvbmUgb3Igcm9vbSBjb25maWd1cmF0aW9uIGRhdGEgdG8gc2VydmVyIGluc3RhbmNlXG5cdFx0bGV0IHBhcmFtcyA9IG5ldyBTRlMyWC5TRlNPYmplY3QoKTtcblx0XHRwYXJhbXMucHV0SW50KCd6SWQnLCB0aGlzLl9lZGl0ZWRab25lSWQpO1xuXHRcdHBhcmFtcy5wdXRJbnQoJ3JJZCcsIHRoaXMuX2VkaXRlZFJvb21JZCk7XG5cblx0XHQvLyBJZiBubyByb29tIGlzIHNlbGVjdGVkLCB0aGVuIHdlIGFyZSBlZGl0aW5nIGEgem9uZVxuXHRcdGlmICh0aGlzLl9lZGl0ZWRJdGVtVHlwZSA9PSB0aGlzLklURU1fVFlQRV9aT05FKVxuXHRcdFx0dGhpcy5zZW5kRXh0ZW5zaW9uUmVxdWVzdCh0aGlzLlJFUV9HRVRfWk9ORV9DT05GSUcsIHBhcmFtcyk7XG5cdFx0ZWxzZVxuXHRcdFx0dGhpcy5zZW5kRXh0ZW5zaW9uUmVxdWVzdCh0aGlzLlJFUV9HRVRfUk9PTV9DT05GSUcsIHBhcmFtcyk7XG5cblx0XHQvLyBTd2l0Y2ggcGFuZWxcblx0XHR0aGlzLl9zd2l0Y2hQYW5lbCgnem5jLW1haW5QYW5lbCcpO1xuXHR9XG5cblx0X3Jlc2V0VGFic0NvbnRhaW5lcihpc0xvYWRpbmcsIHJlc2V0VGFicylcblx0e1xuXHRcdGlmIChyZXNldFRhYnMpXG5cdFx0XHR0aGlzLl9jbGVhclRhYnMoKTtcblx0XHRlbHNlXG5cdFx0XHR0aGlzLl9yZWluaXRUYWJzID0gZmFsc2U7XG5cblx0XHRpZiAoIWlzTG9hZGluZylcblx0XHRcdHRoaXMuX3N3aXRjaFZpZXcoJ3puYy1zZWxlY3QnKTtcblx0XHRlbHNlXG5cdFx0XHR0aGlzLl9zd2l0Y2hWaWV3KCd6bmMtbG9hZGluZycpO1xuXHR9XG5cblx0X3N3aXRjaFBhbmVsKHBhbmVsSWQpXG5cdHtcblx0XHRkb2N1bWVudC5nZXRFbGVtZW50QnlJZCgnem5jLXZpZXcnKS5zZWxlY3RlZFBhbmVsID0gZG9jdW1lbnQuZ2V0RWxlbWVudEJ5SWQocGFuZWxJZCk7XG5cdH1cblxuXHRfZ2V0Wm9uZURhdGFJdGVtQnlJZCh6b25lSWQpXG5cdHtcblx0XHRsZXQgem9uZXNEUyA9IHRoaXMuX3RyZWV2aWV3LmRhdGFTb3VyY2U7XG5cdFx0cmV0dXJuIHpvbmVzRFMuZ2V0KHpvbmVJZCk7XG5cdH1cblxuXHRfZ2V0Um9vbURhdGFJdGVtQnlJZCh6b25lSWQsIHJvb21JZClcblx0e1xuXHRcdGxldCB6b25lREkgPSB0aGlzLl9nZXRab25lRGF0YUl0ZW1CeUlkKHpvbmVJZCk7XG5cblx0XHRpZiAoem9uZURJLmhhc0NoaWxkcmVuKVxuXHRcdFx0cmV0dXJuIHpvbmVESS5jaGlsZHJlbi5nZXQocm9vbUlkKTtcblxuXHRcdHJldHVybiBudWxsO1xuXHR9XG5cblx0X3VwZGF0ZVpvbmVOYW1lSW5MaXN0KHpvbmVJZCwgem9uZU5hbWUpXG5cdHtcblx0XHR0aGlzLl9nZXRab25lRGF0YUl0ZW1CeUlkKHpvbmVJZCkubmFtZSA9IHpvbmVOYW1lO1xuXHRcdHRoaXMuX3RyZWV2aWV3LmRhdGFTb3VyY2Uuc3luYygpO1xuXHR9XG5cblx0X3VwZGF0ZVJvb21OYW1lSW5MaXN0KHpvbmVJZCwgcm9vbUlkLCByb29tTmFtZSlcblx0e1xuXHRcdHRoaXMuX2dldFJvb21EYXRhSXRlbUJ5SWQoem9uZUlkLCByb29tSWQpLm5hbWUgPSByb29tTmFtZTtcblx0XHR0aGlzLl90cmVldmlldy5kYXRhU291cmNlLnN5bmMoKTtcblx0fVxuXG5cdF92YWxpZGF0ZU5hbWUoY2hhbmdlcylcblx0e1xuXHRcdGNvbnN0IHpvbmVJZCA9IHRoaXMuX2VkaXRlZFpvbmVJZDtcblxuXHRcdGZvciAobGV0IGkgPSAwOyBpIDwgY2hhbmdlcy5zaXplKCk7IGkrKylcblx0XHR7XG5cdFx0XHRjb25zdCBzZXR0aW5nID0gY2hhbmdlcy5nZXRTRlNPYmplY3QoaSk7XG5cblx0XHRcdGlmIChzZXR0aW5nLmNvbnRhaW5zS2V5KCduYW1lJykgJiYgc2V0dGluZy5nZXRVdGZTdHJpbmcoJ25hbWUnKSA9PSAnbmFtZScpXG5cdFx0XHR7XG5cdFx0XHRcdC8vIEdldCBuYW1lIHZhbHVlXG5cdFx0XHRcdGNvbnN0IG5hbWUgPSBzZXR0aW5nLmdldFRleHQoJ3ZhbHVlJyk7XG5cblx0XHRcdFx0Ly8gR2V0IGRhdGEgc291cmNlXG5cdFx0XHRcdGxldCBkcyA9IFtdO1xuXG5cdFx0XHRcdGlmICh0aGlzLl9lZGl0ZWRJdGVtVHlwZSA9PSB0aGlzLklURU1fVFlQRV9aT05FKVxuXHRcdFx0XHRcdGRzID0gdGhpcy5fdHJlZXZpZXcuZGF0YVNvdXJjZS5kYXRhKCk7XG5cdFx0XHRcdGVsc2Vcblx0XHRcdFx0e1xuXHRcdFx0XHRcdGlmICh0aGlzLl9nZXRab25lRGF0YUl0ZW1CeUlkKHpvbmVJZCkuaGFzQ2hpbGRyZW4pXG5cdFx0XHRcdFx0XHRkcyA9IHRoaXMuX2dldFpvbmVEYXRhSXRlbUJ5SWQoem9uZUlkKS5jaGlsZHJlbi5kYXRhKCk7XG5cdFx0XHRcdH1cblxuXG5cdFx0XHRcdC8vIENoZWNrIGlmIG5hbWUgZXhpc3RzIGluIGRhdGEgc291cmNlXG5cdFx0XHRcdGZvciAobGV0IGogPSAwOyBqIDwgZHMubGVuZ3RoOyBqKyspXG5cdFx0XHRcdHtcblx0XHRcdFx0XHRpZiAoZHNbal0ubmFtZSA9PSBuYW1lKVxuXHRcdFx0XHRcdHtcblx0XHRcdFx0XHRcdHJldHVybiBmYWxzZTtcblx0XHRcdFx0XHR9XG5cdFx0XHRcdH1cblxuXHRcdFx0XHRicmVhaztcblx0XHRcdH1cblx0XHR9XG5cblx0XHRyZXR1cm4gdHJ1ZTtcblx0fVxuXG5cdC8vLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tXG5cdC8vIFBSSVZBVEUgR0VUVEVSU1xuXHQvLy0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLVxuXG5cdGdldCBfc2VsZWN0ZWRJdGVtKClcblx0e1xuXHRcdHJldHVybiB0aGlzLl90cmVldmlldy5kYXRhSXRlbSh0aGlzLl90cmVldmlldy5zZWxlY3QoKSk7XG5cdH1cblxuXHRnZXQgX3NlbGVjdGVkSXRlbVBhcmVudCgpXG5cdHtcblx0XHRsZXQgc2VsZWN0ZWROb2RlID0gdGhpcy5fdHJlZXZpZXcuc2VsZWN0KCk7XG5cdFx0bGV0IHBhcmVudE5vZGUgPSB0aGlzLl90cmVldmlldy5wYXJlbnQoc2VsZWN0ZWROb2RlKTtcblxuXHRcdHJldHVybiB0aGlzLl90cmVldmlldy5kYXRhSXRlbShwYXJlbnROb2RlKTtcblx0fVxuXG5cdGdldCBfZWRpdGVkWm9uZUlkKClcblx0e1xuXHRcdGlmICh0aGlzLl9pc0VkaXRpbmcgJiYgdGhpcy5fc2VsZWN0ZWRJdGVtKVxuXHRcdHtcblx0XHRcdGlmICh0aGlzLl9zZWxlY3RlZEl0ZW0udHlwZSA9PSB0aGlzLklURU1fVFlQRV9aT05FKVxuXHRcdFx0XHRyZXR1cm4gdGhpcy5fc2VsZWN0ZWRJdGVtLmlkO1xuXHRcdFx0ZWxzZVxuXHRcdFx0XHRyZXR1cm4gdGhpcy5fc2VsZWN0ZWRJdGVtUGFyZW50LmlkO1xuXHRcdH1cblxuXHRcdHJldHVybiAtMTtcblx0fVxuXG5cdGdldCBfZWRpdGVkUm9vbUlkKClcblx0e1xuXHRcdGlmICh0aGlzLl9pc0VkaXRpbmcgJiYgdGhpcy5fc2VsZWN0ZWRJdGVtKVxuXHRcdHtcblx0XHRcdGlmICh0aGlzLl9zZWxlY3RlZEl0ZW0udHlwZSA9PSB0aGlzLklURU1fVFlQRV9ST09NKVxuXHRcdFx0XHRyZXR1cm4gdGhpcy5fc2VsZWN0ZWRJdGVtLmlkO1xuXHRcdH1cblxuXHRcdHJldHVybiAtMTtcblx0fVxufVxuIl0sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7OztBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTs7Ozs7Ozs7Ozs7Ozs7QUNwY0E7QUFBQTtBQUFBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTs7Ozs7Ozs7Ozs7OztBQ3ZIQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBOzs7Ozs7QSIsInNvdXJjZVJvb3QiOiIifQ==