LyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogQ29weXJpZ2h0IChjKSAyMDA0IEV0ZXJhdGlvbiBCaWxpc2ltIEEuUy4KICogQWxsIHJpZ2h0cyByZXNlcnZlZC4goCBUaGlzIHByb2dyYW0gYW5kIHRoZSBhY2NvbXBhbnlpbmcgbWF0ZXJpYWxzCiAqIGFyZSBtYWRlIGF2YWlsYWJsZSB1bmRlciB0aGUgdGVybXMgb2YgdGhlIENvbW1vbiBQdWJsaWMgTGljZW5zZSB2MS4wCiAqIHdoaWNoIGFjY29tcGFuaWVzIHRoaXMgZGlzdHJpYnV0aW9uLCBhbmQgaXMgYXZhaWxhYmxlIGF0CiAqIGh0dHA6Ly93d3cuZWNsaXBzZS5vcmcvbGVnYWwvY3BsLXYxMC5odG1sCiAqIAogKiBDb250cmlidXRvcnM6CiAqICAgICBOYWNpIE0uIERhaSAtIGluaXRpYWwgQVBJIGFuZCBpbXBsZW1lbnRhdGlvbgogKiAKICogVEhJUyBTT0ZUV0FSRSBJUyBQUk9WSURFRCBgYEFTIElTJycgQU5EIEFOWSBFWFBSRVNTRUQgT1IgSU1QTElFRAogKiBXQVJSQU5USUVTLCBJTkNMVURJTkcsIEJVVCBOT1QgTElNSVRFRCBUTywgVEhFIElNUExJRUQgV0FSUkFOVElFUwogKiBPRiBNRVJDSEFOVEFCSUxJVFkgQU5EIEZJVE5FU1MgRk9SIEEgUEFSVElDVUxBUiBQVVJQT1NFIEFSRQogKiBESVNDTEFJTUVELiAgSU4gTk8gRVZFTlQgU0hBTEwgRVRFUkFUSU9OIEEuUy4gT1IKICogSVRTIENPTlRSSUJVVE9SUyBCRSBMSUFCTEUgRk9SIEFOWSBESVJFQ1QsIElORElSRUNULCBJTkNJREVOVEFMLAogKiBTUEVDSUFMLCBFWEVNUExBUlksIE9SIENPTlNFUVVFTlRJQUwgREFNQUdFUyAoSU5DTFVESU5HLCBCVVQgTk9UCiAqIExJTUlURUQgVE8sIFBST0NVUkVNRU5UIE9GIFNVQlNUSVRVVEUgR09PRFMgT1IgU0VSVklDRVM7IExPU1MgT0YKICogVVNFLCBEQVRBLCBPUiBQUk9GSVRTOyBPUiBCVVNJTkVTUyBJTlRFUlJVUFRJT04pIEhPV0VWRVIgQ0FVU0VEIEFORAogKiBPTiBBTlkgVEhFT1JZIE9GIExJQUJJTElUWSwgV0hFVEhFUiBJTiBDT05UUkFDVCwgU1RSSUNUIExJQUJJTElUWSwKICogT1IgVE9SVCAoSU5DTFVESU5HIE5FR0xJR0VOQ0UgT1IgT1RIRVJXSVNFKSBBUklTSU5HIElOIEFOWSBXQVkgT1VUCiAqIE9GIFRIRSBVU0UgT0YgVEhJUyBTT0ZUV0FSRSwgRVZFTiBJRiBBRFZJU0VEIE9GIFRIRSBQT1NTSUJJTElUWSBPRgogKiBTVUNIIERBTUFHRS4KICogPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT0KICoKICogVGhpcyBzb2Z0d2FyZSBjb25zaXN0cyBvZiB2b2x1bnRhcnkgY29udHJpYnV0aW9ucyBtYWRlIGJ5IG1hbnkKICogaW5kaXZpZHVhbHMgb24gYmVoYWxmIG9mIHRoZSBFdGVyYXRpb24gQmlsaXNpbSBBLlMuICBGb3IgbW9yZQogKiBpbmZvcm1hdGlvbiBvbiBldGVyYXRpb24sIHBsZWFzZSBzZWUKICogPGh0dHA6Ly93d3cuZXRlcmF0aW9uLmNvbS8+LgogKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwpwYWNrYWdlIG9yZy5lY2xpcHNlLmpzdC5zZXJ2ZXIuZ2VuZXJpYy5jb3JlLmludGVybmFsOwoKaW1wb3J0IGphdmEuaW8uRmlsZTsKaW1wb3J0IGphdmEuaW8uRmlsZU91dHB1dFN0cmVhbTsKaW1wb3J0IGphdmEuaW8uSU9FeGNlcHRpb247CmltcG9ydCBqYXZhLmlvLklucHV0U3RyZWFtOwppbXBvcnQgamF2YS5pby5PdXRwdXRTdHJlYW07CmltcG9ydCBqYXZhLm5ldC5VUkw7CmltcG9ydCBqYXZhLnV0aWwuSGFzaE1hcDsKaW1wb3J0IGphdmEudXRpbC5JdGVyYXRvcjsKaW1wb3J0IGphdmEudXRpbC5NYXA7CmltcG9ydCBqYXZhLnV0aWwuamFyLkphckVudHJ5OwppbXBvcnQgamF2YS51dGlsLmphci5KYXJGaWxlOwppbXBvcnQgb3JnLmVjbGlwc2UuYW50LmludGVybmFsLnVpLklBbnRVSUNvbnN0YW50czsKaW1wb3J0IG9yZy5lY2xpcHNlLmFudC5pbnRlcm5hbC51aS5sYXVuY2hDb25maWd1cmF0aW9ucy5JQW50TGF1bmNoQ29uZmlndXJhdGlvbkNvbnN0YW50czsKaW1wb3J0IG9yZy5lY2xpcHNlLmNvcmUucnVudGltZS5Db3JlRXhjZXB0aW9uOwppbXBvcnQgb3JnLmVjbGlwc2UuY29yZS5ydW50aW1lLklQcm9ncmVzc01vbml0b3I7CmltcG9ydCBvcmcuZWNsaXBzZS5jb3JlLnJ1bnRpbWUuSVN0YXR1czsKaW1wb3J0IG9yZy5lY2xpcHNlLmNvcmUucnVudGltZS5QbGF0Zm9ybTsKaW1wb3J0IG9yZy5lY2xpcHNlLmNvcmUucnVudGltZS5TdGF0dXM7CmltcG9ydCBvcmcuZWNsaXBzZS5kZWJ1Zy5jb3JlLkRlYnVnUGx1Z2luOwppbXBvcnQgb3JnLmVjbGlwc2UuZGVidWcuY29yZS5JTGF1bmNoQ29uZmlndXJhdGlvbjsKaW1wb3J0IG9yZy5lY2xpcHNlLmRlYnVnLmNvcmUuSUxhdW5jaENvbmZpZ3VyYXRpb25UeXBlOwppbXBvcnQgb3JnLmVjbGlwc2UuZGVidWcuY29yZS5JTGF1bmNoQ29uZmlndXJhdGlvbldvcmtpbmdDb3B5OwppbXBvcnQgb3JnLmVjbGlwc2UuZGVidWcuY29yZS5JTGF1bmNoTWFuYWdlcjsKaW1wb3J0IG9yZy5lY2xpcHNlLmRlYnVnLnVpLklEZWJ1Z1VJQ29uc3RhbnRzOwppbXBvcnQgb3JnLmVjbGlwc2UuamR0LmxhdW5jaGluZy5JSmF2YUxhdW5jaENvbmZpZ3VyYXRpb25Db25zdGFudHM7CmltcG9ydCBvcmcuZWNsaXBzZS5qc3Quc2VydmVyLmdlbmVyaWMuaW50ZXJuYWwuY29yZS51dGlsLkZpbGVVdGlsOwppbXBvcnQgb3JnLmVjbGlwc2UuanN0LnNlcnZlci5nZW5lcmljLnNlcnZlcnR5cGUuZGVmaW5pdGlvbi5Nb2R1bGU7CmltcG9ydCBvcmcuZWNsaXBzZS5qc3Quc2VydmVyLmdlbmVyaWMuc2VydmVydHlwZS5kZWZpbml0aW9uLlB1Ymxpc2hlckRhdGE7CmltcG9ydCBvcmcuZWNsaXBzZS5qc3Quc2VydmVyLmNvcmUuSUVKQk1vZHVsZTsKaW1wb3J0IG9yZy5lY2xpcHNlLmpzdC5zZXJ2ZXIuY29yZS5JV2ViTW9kdWxlOwppbXBvcnQgb3JnLmVjbGlwc2UudWkuZXh0ZXJuYWx0b29scy5pbnRlcm5hbC5tb2RlbC5JRXh0ZXJuYWxUb29sQ29uc3RhbnRzOwppbXBvcnQgb3JnLmVjbGlwc2Uud3N0LnNlcnZlci5jb3JlLklNb2R1bGVBcnRpZmFjdDsKaW1wb3J0IG9yZy5vc2dpLmZyYW1ld29yay5CdW5kbGU7Ci8qKgogKiBBbnQgYmFzZWQgcHVibGlzaGVyLgogKiBBbGwgdGhlIHByb3BlcnRpZXMgZGVmaW5lZCBpbiB0aGUgc2VydmVyIGRlZmluaXRpb24gZmlsZSBhcmUKICogcGFzc2VkIGludG8gdGhlIEFOVCBidWlsZCBmaWxlIGFzIHByb3BlcnRpZXMuCiAqIEluIGFkZGl0aW9uIHRvIHRoZSBwcm9wZXJ0aWVzIGRlZmluZWQgaW4gdGhlIHNlcnZlciBkZWZpbml0aW9uCiAqIDxJPm1vZHVsZS5kaXI8L0k+LCA8ST5tb2R1bGUubmFtZSw8L0k+IGFuZCA8ST5zZXJ2ZXIucHVibGlzaC5kaXI8L0k+IGFyZSBjb21wdXRlZCBhbmQgcGFzc2VkIHRvIHRoZSAKICogZGVmaW5pdGlvbiBmaWxlLgogKiA8dWw+CiAqIDxsaT5tb2R1bGUuZGlyOiBpbmNsdWRlcyB0aGUgcm9vdCBvZiB0aGUgbW9kdWxlIHByb2plY3QgZmlsZTwvbGk+CiAqIDxsaT5tb2R1bGUubmFtZTogdGhlIG5hbWUgb2YgdGhlIG1vZHVsZTwvbGk+CiAqIDxsaT5zZXJ2ZXIucHVibGlzaC5kaXI6IHRoZSBkaXJlY3RvcnkgdG8gcHV0IHRoZSBkZXBsb3ltZW50IHVuaXRzPC9saT4KICogPC91bD4KICoKICogQGF1dGhvciBHb3JrZW0gRXJjYW4KICovCgpwdWJsaWMgY2xhc3MgQW50UHVibGlzaGVyIGV4dGVuZHMgR2VuZXJpY1B1Ymxpc2hlcnsKCXByaXZhdGUgc3RhdGljIGZpbmFsIFN0cmluZyBKQVJfUFJPVE9DT0xfUFJFRklYID0gImphciI7CgoJLyoqCgkgKiBwdWJsaXNoZXIgaWQgZm9yIEFOVCBwdWJsaXNoZXIuCgkgKi8KCXB1YmxpYyBzdGF0aWMgZmluYWwgU3RyaW5nIFBVQkxJU0hFUl9JRD0ib3JnLmVjbGlwc2UuanN0LnNlcnZlci5nZW5lcmljLmFudHB1Ymxpc2hlciI7IC8vJE5PTi1OTFMtMSQKICAgIAoJcHJpdmF0ZSBzdGF0aWMgZmluYWwgU3RyaW5nIFBST1BfU0VSVkVSX1BVQkxJU0hfRElSID0gInNlcnZlci5wdWJsaXNoLmRpciI7Ly8kTk9OLU5MUy0xJAoJcHJpdmF0ZSBzdGF0aWMgZmluYWwgU3RyaW5nIFBST1BfTU9EVUxFX0RJUiA9ICJtb2R1bGUuZGlyIjsvLyROT04tTkxTLTEkCglwcml2YXRlIHN0YXRpYyBmaW5hbCBTdHJpbmcgUFJPUF9NT0RVTEVfTkFNRSA9ICJtb2R1bGUubmFtZSI7Ly8kTk9OLU5MUy0xJAoJcHJpdmF0ZSBzdGF0aWMgZmluYWwgU3RyaW5nIE1PRFVMRV9QVUJMSVNIX1RBUkdFVF9QUkVGSVggPSAidGFyZ2V0LnB1Ymxpc2guIjsgLy8kTk9OLU5MUy0xJAogICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgU3RyaW5nIE1PRFVMRV9VTlBVQkxJU0hfVEFSR0VUX1BSRUZJWCA9ICJ0YXJnZXQudW5wdWJsaXNoLiI7Ly8kTk9OLU5MUy0xJAogICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgU3RyaW5nIERBVEFfTkFNRV9CVUlMRF9GSUxFPSJidWlsZC5maWxlIjsvLyROT04tTkxTLTEkCgogICAgLyogKG5vbi1KYXZhZG9jKQoJICogQHNlZSBvcmcuZWNsaXBzZS53dHAuc2VydmVyLmNvcmUubW9kZWwuSVB1Ymxpc2hlciNwdWJsaXNoKG9yZy5lY2xpcHNlLnd0cC5zZXJ2ZXIuY29yZS5yZXNvdXJjZXMuSU1vZHVsZVJlc291cmNlW10sIG9yZy5lY2xpcHNlLmNvcmUucnVudGltZS5JUHJvZ3Jlc3NNb25pdG9yKQoJICovCglwdWJsaWMgSVN0YXR1c1tdIHB1Ymxpc2goSU1vZHVsZUFydGlmYWN0W10gcmVzb3VyY2UsIElQcm9ncmVzc01vbml0b3IgbW9uaXRvcil7CiAgICAgICAgdHJ5ewogICAgICAgIAlGaWxlIGZpbGUgPSBjb21wdXRlQnVpbGRGaWxlKCk7CiAgICAgICAgCXJ1bkFudChmaWxlLnRvU3RyaW5nKCksZ2V0UHVibGlzaFRhcmdldHNGb3JNb2R1bGUoKSxnZXRQdWJsaXNoUHJvcGVydGllcygpLG1vbml0b3IpOwogICAgICAgIH1jYXRjaChDb3JlRXhjZXB0aW9uIGUpewogICAgICAgICAgICBJU3RhdHVzIHMgPSBuZXcgU3RhdHVzKElTdGF0dXMuRVJST1IsQ29yZVBsdWdpbi5QTFVHSU5fSUQsMCxHZW5lcmljU2VydmVyQ29yZU1lc3NhZ2VzLmVycm9yUHVibGlzaEFudHB1Ymxpc2hlcixlKTsKICAgICAgICAgICAgQ29yZVBsdWdpbi5nZXREZWZhdWx0KCkuZ2V0TG9nKCkubG9nKHMpOwogICAgICAgICAgICByZXR1cm4gbmV3IElTdGF0dXNbXSB7c307CiAgICAgICAgfQoJCXJldHVybiBudWxsOwoJfQoKICAgIC8qKgogICAgICogCiAgICAgKiBAcmV0dXJuCiAgICAgKiBAdGhyb3dzIENvcmVFeGNlcHRpb24KICAgICAqLwogICAgcHJpdmF0ZSBGaWxlIGNvbXB1dGVCdWlsZEZpbGUoKSB0aHJvd3MgQ29yZUV4Y2VwdGlvbiB7CiAgICAgICAgQnVuZGxlIGJ1bmRsZSA9IFBsYXRmb3JtLmdldEJ1bmRsZShnZXRTZXJ2ZXJSdW50aW1lKCkuZ2V0U2VydmVyVHlwZURlZmluaXRpb24oKS5nZXRDb25maWd1cmF0aW9uRWxlbWVudE5hbWVzcGFjZSgpKTsKICAgICAgICBVUkwgYnVuZGxlVXJsID1idW5kbGUuZ2V0RW50cnkoZ2V0QnVpbGRGaWxlKCkpOwogICAgICAgIFVSTCBmaWxlVVJMID0gRmlsZVV0aWwucmVzb2x2ZVVSTChidW5kbGVVcmwpOwogICAgICAgIGlmKGZpbGVVUkwuZ2V0UHJvdG9jb2woKS5lcXVhbHMoSkFSX1BST1RPQ09MX1BSRUZJWCkpewogICAgICAgIAlPdXRwdXRTdHJlYW0gb3M9bnVsbDsKICAgICAgICAJSW5wdXRTdHJlYW0gaXM9bnVsbDsgCiAgICAgICAgCXRyeXsKICAgICAgICAJCVN0cmluZyBmaWxlbmFtZSA9ZmlsZVVSTC5nZXRQYXRoKCk7CiAgICAgICAgCQlTdHJpbmcgamFybmFtZT0gZmlsZVVSTC5nZXRGaWxlKCkuc3Vic3RyaW5nKDAsZmlsZW5hbWUuaW5kZXhPZignIScpKTsKICAgICAgICAJCQogICAgICAgIAkJRmlsZSBqYXJGaWxlID0gbmV3IEZpbGUobmV3IFVSTChqYXJuYW1lKS5nZXRGaWxlKCkpOwogICAgICAgIAkJSmFyRmlsZSBqYXIgPSBuZXcgSmFyRmlsZShqYXJGaWxlKTsKICAgICAgICAJCUZpbGUgdG1wRmlsZSA9IEZpbGVVdGlsLmNyZWF0ZVRlbXBGaWxlKGdldEJ1aWxkRmlsZSgpLENvcmVQbHVnaW4uZ2V0RGVmYXVsdCgpLmdldFN0YXRlTG9jYXRpb24oKS50b09TU3RyaW5nKCkpOwogICAgICAgIAkJb3MgPSBuZXcgRmlsZU91dHB1dFN0cmVhbSh0bXBGaWxlKTsKICAgICAgICAJCVN0cmluZyBlbnRyeW5hbWU9IGdldEJ1aWxkRmlsZSgpOwogICAgICAgIAkJaWYgKGVudHJ5bmFtZS5zdGFydHNXaXRoKCIvIikpLy8kTk9OLU5MUy0xJAogICAgICAgIAkJCWVudHJ5bmFtZT0gZW50cnluYW1lLnN1YnN0cmluZygxKTsKICAgICAgICAJCUphckVudHJ5IGVudHJ5ID0gamFyLmdldEphckVudHJ5KGVudHJ5bmFtZSk7CiAgICAgICAgCQlpcyA9amFyLmdldElucHV0U3RyZWFtKGVudHJ5KTsKICAgICAgICAJCUZpbGVVdGlsLmNvcHkoaXMsb3MpOwogICAgICAgIAkJcmV0dXJuIHRtcEZpbGU7CiAgICAgICAgIAl9CiAgICAgICAgCWNhdGNoIChJT0V4Y2VwdGlvbiBlKSB7CiAgICAgICAgCQlJU3RhdHVzIHMgPSBuZXcgU3RhdHVzKElTdGF0dXMuRVJST1IsQ29yZVBsdWdpbi5QTFVHSU5fSUQsMCwiZXJyb3IgY3JlYXRpbmcgdGVtcG9yYXJ5IGJ1aWxkIGZpbGUiLGUpOy8vJE5PTi1OTFMtMSQKICAgICAgICAgICAgICAgIENvcmVQbHVnaW4uZ2V0RGVmYXVsdCgpLmdldExvZygpLmxvZyhzKTsKCQkJCXRocm93IG5ldyBDb3JlRXhjZXB0aW9uKHMpOwoJCQl9CiAgICAgICAgCWZpbmFsbHl7CiAgICAgICAgCQl0cnkgewogICAgICAgIAkJCWlmKGlzIT1udWxsKQogICAgICAgIAkJCQlpcy5jbG9zZSgpOwoJCQkJCWlmKG9zIT1udWxsKQoJCQkJCQlvcy5jbG9zZSgpOwoJCQkJfSBjYXRjaCAoSU9FeGNlcHRpb24gZSkgewoJCQkJCS8vaWdub3JlCgkJCQl9CiAgICAgICAgCX0KICAgICAgICB9CiAgICAgICAgZWxzZXsKICAgICAgICAJcmV0dXJuIEZpbGVVdGlsLnJlc29sdmVGaWxlKGZpbGVVUkwpOwogICAgICAgIH0gCQogICAgfQogICAKICAgIC8qKgogICAgICogQHJldHVybgogICAgICovCiAgICBwcml2YXRlIFN0cmluZyBnZXRQdWJsaXNoVGFyZ2V0c0Zvck1vZHVsZSgpIHsKICAgIAlyZXR1cm4gZG9HZXRUYXJnZXRzKE1PRFVMRV9QVUJMSVNIX1RBUkdFVF9QUkVGSVgrZ2V0TW9kdWxlVHlwZUlkKCkpOwogICAgfQoKICAgIC8qKgogICAgICogQHJldHVybgogICAgICovCiAgICBwcml2YXRlIFN0cmluZyBnZXRVbnB1Ymxpc2hUYXJnZXRzRm9yTW9kdWxlKCkgewogICAgICAgIHJldHVybiBkb0dldFRhcmdldHMoTU9EVUxFX1VOUFVCTElTSF9UQVJHRVRfUFJFRklYK2dldE1vZHVsZVR5cGVJZCgpKTsKICAgIH0KICAgIAogICAgLyoqCiAgICAgKiBAcGFyYW0gZGF0YW5hbWUKICAgICAqIEByZXR1cm4KICAgICAqLwogICAgcHJpdmF0ZSBTdHJpbmcgZG9HZXRUYXJnZXRzKFN0cmluZyBkYXRhbmFtZSkgewogICAgCVN0cmluZ0J1ZmZlciBidWZmZXIgPSBuZXcgU3RyaW5nQnVmZmVyKCk7CiAgICAJSXRlcmF0b3IgaXRlcmF0b3IgPSBnZXRTZXJ2ZXJSdW50aW1lKCkuZ2V0U2VydmVyVHlwZURlZmluaXRpb24oKS5nZXRQdWJsaXNoZXIoUFVCTElTSEVSX0lEKS5nZXRQdWJsaXNoZXJkYXRhKCkuaXRlcmF0b3IoKTsKICAgICAgICB3aGlsZShpdGVyYXRvci5oYXNOZXh0KCkpewogICAgICAgICAgICBQdWJsaXNoZXJEYXRhIGRhdGEgPSAoUHVibGlzaGVyRGF0YSlpdGVyYXRvci5uZXh0KCk7CiAgICAgICAgICAgIGlmKGRhdGFuYW1lLmVxdWFscyhkYXRhLmdldERhdGFuYW1lKCkpKSB7CiAgICAgICAgICAgICAgICBpZihidWZmZXIubGVuZ3RoKCk+MCkKICAgICAgICAgICAgICAgIAlidWZmZXIuYXBwZW5kKCIsIik7Ly8kTk9OLU5MUy0xJAogICAgICAgICAgICAJYnVmZmVyLmFwcGVuZChkYXRhLmdldERhdGF2YWx1ZSgpKTsKICAgICAgICAgICAgfSAgIAogICAgICAgIH0KICAgICAgICByZXR1cm4gYnVmZmVyLnRvU3RyaW5nKCk7CiAgICB9CgogICAgcHJpdmF0ZSBTdHJpbmcgZ2V0TW9kdWxlVHlwZUlkKCl7CiAgICAgICAgcmV0dXJuIGdldE1vZHVsZSgpWzBdLmdldE1vZHVsZVR5cGUoKS5nZXRJZCgpOwogICAgfQogICAgCglwcml2YXRlIFN0cmluZyBnZXRCdWlsZEZpbGUoKQogICAgewogICAgICAgIEl0ZXJhdG9yIGl0ZXJhdG9yID0gZ2V0U2VydmVyUnVudGltZSgpLmdldFNlcnZlclR5cGVEZWZpbml0aW9uKCkuZ2V0UHVibGlzaGVyKFBVQkxJU0hFUl9JRCkuZ2V0UHVibGlzaGVyZGF0YSgpLml0ZXJhdG9yKCk7CiAgICAgICAgd2hpbGUoaXRlcmF0b3IuaGFzTmV4dCgpKQogICAgICAgIHsKICAgICAgICAgICAgUHVibGlzaGVyRGF0YSBkYXRhID0gKFB1Ymxpc2hlckRhdGEpaXRlcmF0b3IubmV4dCgpOwogICAgICAgICAgICBpZihEQVRBX05BTUVfQlVJTERfRklMRS5lcXVhbHMoZGF0YS5nZXREYXRhbmFtZSgpKSkKICAgICAgICAgICAgICAgIHJldHVybiBnZXRTZXJ2ZXJSdW50aW1lKCkuZ2V0U2VydmVyVHlwZURlZmluaXRpb24oKS5nZXRSZXNvbHZlcigpLnJlc29sdmVQcm9wZXJ0aWVzKGRhdGEuZ2V0RGF0YXZhbHVlKCkpOwogICAgICAgIH0KICAgICAgICByZXR1cm4gbnVsbDsKICAgIH0KCQoJcHJpdmF0ZSBNYXAgZ2V0UHVibGlzaFByb3BlcnRpZXMoKQoJewogICAgICAgIE1hcCBwcm9wcyA9IG5ldyBIYXNoTWFwKCk7CiAgICAgICAgLy8gcGFzcyBhbGwgcHJvcGVydGllcyB0byBidWlsZCBmaWxlLgogICAgICAgIE1hcCBwcm9wZXJ0aWVzID0gZ2V0U2VydmVyUnVudGltZSgpLmdldFNlcnZlclR5cGVEZWZpbml0aW9uKCkuZ2V0UmVzb2x2ZXIoKS5nZXRQcm9wZXJ0eVZhbHVlcygpOwogICAgICAgIEl0ZXJhdG9yIHByb3BlcnR5SXRlcmF0b3IgPSBwcm9wZXJ0aWVzLmtleVNldCgpLml0ZXJhdG9yKCk7CiAgICAgICAgd2hpbGUocHJvcGVydHlJdGVyYXRvci5oYXNOZXh0KCkpCiAgICAgICAgewogICAgICAgICAgICBTdHJpbmcgcHJvcGVydHkgPSAoU3RyaW5nKXByb3BlcnR5SXRlcmF0b3IubmV4dCgpOwogICAgICAgICAgICBwcm9wcy5wdXQocHJvcGVydHkscHJvcGVydGllcy5nZXQocHJvcGVydHkpKTsKICAgICAgICB9CiAgICAgICAgTW9kdWxlIG1vZHVsZSA9ICBnZXRTZXJ2ZXJSdW50aW1lKCkuZ2V0U2VydmVyVHlwZURlZmluaXRpb24oKS5nZXRNb2R1bGUoZ2V0TW9kdWxlVHlwZUlkKCkpOwoJCVN0cmluZyBtb2REaXIgPSBtb2R1bGUuZ2V0UHVibGlzaERpcigpOwoJCW1vZERpciA9IGdldFNlcnZlclJ1bnRpbWUoKS5nZXRTZXJ2ZXJUeXBlRGVmaW5pdGlvbigpLmdldFJlc29sdmVyKCkucmVzb2x2ZVByb3BlcnRpZXMobW9kRGlyKTsKCgkJSVdlYk1vZHVsZSB3ZWJNb2R1bGUgPSAoSVdlYk1vZHVsZSlnZXRNb2R1bGUoKVswXS5nZXRBZGFwdGVyKElXZWJNb2R1bGUuY2xhc3MpOwogICAgICAgIElFSkJNb2R1bGUgZWpiTW9kdWxlID0gKElFSkJNb2R1bGUpZ2V0TW9kdWxlKClbMF0uZ2V0QWRhcHRlcihJRUpCTW9kdWxlLmNsYXNzKTsKCQlTdHJpbmcgbW9kdWxlTmFtZT0idW5rbm93bm1vZHVsZSI7Ly8kTk9OLU5MUy0xJAogICAgICAgIFN0cmluZyBtb2R1bGVEaXI9IiI7Ly8kTk9OLU5MUy0xJAogICAgICAgIGlmKHdlYk1vZHVsZSE9bnVsbCl7ICAgIAogICAgICAgICAgICBtb2R1bGVOYW1lID0gdGhpcy5ndWVzc01vZHVsZU5hbWUod2ViTW9kdWxlKTsKICAgICAgICAgICAgbW9kdWxlRGlyID0gd2ViTW9kdWxlLmdldExvY2F0aW9uKCkudG9TdHJpbmcoKTsKICAgICAgICB9CiAgICAgICAgaWYoZWpiTW9kdWxlIT1udWxsKXsgIAogICAgICAgICAgICBtb2R1bGVOYW1lID0gZ2V0TW9kdWxlKClbMF0uZ2V0TmFtZSgpOwogICAgICAgICAgICBtb2R1bGVEaXI9IGVqYk1vZHVsZS5nZXRMb2NhdGlvbigpLnRvU3RyaW5nKCk7CiAgICAgICAgfQoJCXByb3BzLnB1dChQUk9QX01PRFVMRV9OQU1FLG1vZHVsZU5hbWUpOwoJCXByb3BzLnB1dChQUk9QX01PRFVMRV9ESVIsbW9kdWxlRGlyKTsKCQlwcm9wcy5wdXQoUFJPUF9TRVJWRVJfUFVCTElTSF9ESVIsbW9kRGlyKTsKCQlyZXR1cm4gcHJvcHM7Cgl9CgoJcHJpdmF0ZSBTdHJpbmcgZ3Vlc3NNb2R1bGVOYW1lKElXZWJNb2R1bGUgd2ViTW9kdWxlKSB7CgkJU3RyaW5nIG1vZHVsZU5hbWUgPSBnZXRNb2R1bGUoKVswXS5nZXROYW1lKCk7IAoJCVN0cmluZyBjb250ZXh0Um9vdCA9IHdlYk1vZHVsZS5nZXRDb250ZXh0Um9vdCgpOwoJCWlmKGNvbnRleHRSb290LmNoYXJBdCgwKSA9PSAnLycpCgkJCW1vZHVsZU5hbWUgPSBjb250ZXh0Um9vdC5zdWJzdHJpbmcoMSk7CgkJcmV0dXJuIG1vZHVsZU5hbWU7Cgl9CgoJcHJpdmF0ZSB2b2lkIHJ1bkFudChTdHJpbmcgYnVpbGRGaWxlLFN0cmluZyB0YXJnZXRzLE1hcCBwcm9wZXJ0aWVzICxJUHJvZ3Jlc3NNb25pdG9yIG1vbml0b3IpdGhyb3dzIENvcmVFeGNlcHRpb257CgkJSUxhdW5jaE1hbmFnZXIgbGF1bmNoTWFuYWdlciA9IERlYnVnUGx1Z2luLmdldERlZmF1bHQoKS5nZXRMYXVuY2hNYW5hZ2VyKCk7CgkJSUxhdW5jaENvbmZpZ3VyYXRpb25UeXBlIHR5cGUgPSBsYXVuY2hNYW5hZ2VyLmdldExhdW5jaENvbmZpZ3VyYXRpb25UeXBlKElBbnRMYXVuY2hDb25maWd1cmF0aW9uQ29uc3RhbnRzLklEX0FOVF9MQVVOQ0hfQ09ORklHVVJBVElPTl9UWVBFKTsKCgkJSUxhdW5jaENvbmZpZ3VyYXRpb25Xb3JraW5nQ29weSB3Yz0gdHlwZS5uZXdJbnN0YW5jZShudWxsLHByb3BlcnRpZXMuZ2V0KFBST1BfTU9EVUxFX05BTUUpKyIgbW9kdWxlIHB1Ymxpc2hlciIpOwoJCXdjLnNldENvbnRhaW5lcihudWxsKTsKCQl3Yy5zZXRBdHRyaWJ1dGUoSUV4dGVybmFsVG9vbENvbnN0YW50cy5BVFRSX0xPQ0FUSU9OLCBidWlsZEZpbGUpOwoJCXdjLnNldEF0dHJpYnV0ZShJSmF2YUxhdW5jaENvbmZpZ3VyYXRpb25Db25zdGFudHMuQVRUUl9DTEFTU1BBVEhfUFJPVklERVIsIm9yZy5lY2xpcHNlLmFudC51aS5BbnRDbGFzc3BhdGhQcm92aWRlciIpOwoJCXdjLnNldEF0dHJpYnV0ZShJQW50TGF1bmNoQ29uZmlndXJhdGlvbkNvbnN0YW50cy5BVFRSX0FOVF9UQVJHRVRTLHRhcmdldHMpOwoJCXdjLnNldEF0dHJpYnV0ZShJQW50TGF1bmNoQ29uZmlndXJhdGlvbkNvbnN0YW50cy5BVFRSX0FOVF9QUk9QRVJUSUVTLHByb3BlcnRpZXMpOwoJCXdjLnNldEF0dHJpYnV0ZShJRGVidWdVSUNvbnN0YW50cy5BVFRSX0xBVU5DSF9JTl9CQUNLR1JPVU5ELGZhbHNlKTsKCQl3Yy5zZXRBdHRyaWJ1dGUoSURlYnVnVUlDb25zdGFudHMuQVRUUl9DQVBUVVJFX0lOX0NPTlNPTEUsdHJ1ZSk7CgkJd2Muc2V0QXR0cmlidXRlKElEZWJ1Z1VJQ29uc3RhbnRzLkFUVFJfUFJJVkFURSx0cnVlKTsKCQkKCQl3Yy5zZXRBdHRyaWJ1dGUoSUphdmFMYXVuY2hDb25maWd1cmF0aW9uQ29uc3RhbnRzLkFUVFJfU09VUkNFX1BBVEhfUFJPVklERVIsICJvcmcuZWNsaXBzZS5hbnQudWkuQW50Q2xhc3NwYXRoUHJvdmlkZXIiKTsgCgkJd2Muc2V0QXR0cmlidXRlKElKYXZhTGF1bmNoQ29uZmlndXJhdGlvbkNvbnN0YW50cy5BVFRSX1ZNX0lOU1RBTExfTkFNRSxnZXRTZXJ2ZXJSdW50aW1lKCkuZ2V0Vk1JbnN0YWxsKCkuZ2V0TmFtZSgpKTsKCQl3Yy5zZXRBdHRyaWJ1dGUoSUphdmFMYXVuY2hDb25maWd1cmF0aW9uQ29uc3RhbnRzLkFUVFJfVk1fSU5TVEFMTF9UWVBFLGdldFNlcnZlclJ1bnRpbWUoKS5nZXRWTUluc3RhbGwoKS5nZXRWTUluc3RhbGxUeXBlKCkuZ2V0SWQoKSk7CgkJd2Muc2V0QXR0cmlidXRlKElKYXZhTGF1bmNoQ29uZmlndXJhdGlvbkNvbnN0YW50cy5BVFRSX01BSU5fVFlQRV9OQU1FLCAib3JnLmVjbGlwc2UuYW50LmludGVybmFsLnVpLmFudHN1cHBvcnQuSW50ZXJuYWxBbnRSdW5uZXIiKTsKCQl3Yy5zZXRBdHRyaWJ1dGUoRGVidWdQbHVnaW4uQVRUUl9QUk9DRVNTX0ZBQ1RPUllfSUQsIElBbnRVSUNvbnN0YW50cy5SRU1PVEVfQU5UX1BST0NFU1NfRkFDVE9SWV9JRCk7CgkJCgkJSUxhdW5jaENvbmZpZ3VyYXRpb24gbGF1bmNoQ29uZmlnID0gd2MuZG9TYXZlKCk7CiAgICAgICAgbGF1bmNoQ29uZmlnLmxhdW5jaCgicnVuIixtb25pdG9yKTsKCX0KCiAgICAvKiAobm9uLUphdmFkb2MpCiAgICAgKiBAc2VlIG9yZy5lY2xpcHNlLmpzdC5zZXJ2ZXIuZ2VuZXJpYy5pbnRlcm5hbC5jb3JlLkdlbmVyaWNQdWJsaXNoZXIjdW5wdWJsaXNoKG9yZy5lY2xpcHNlLndzdC5zZXJ2ZXIuY29yZS5JTW9kdWxlLCBvcmcuZWNsaXBzZS5jb3JlLnJ1bnRpbWUuSVByb2dyZXNzTW9uaXRvcikKICAgICAqLwogICAgcHVibGljIElTdGF0dXNbXSB1bnB1Ymxpc2goSVByb2dyZXNzTW9uaXRvciBtb25pdG9yKSB7CiAgICAgICAgdHJ5IHsKICAgICAgICAJIEZpbGUgZmlsZSA9IGNvbXB1dGVCdWlsZEZpbGUoKTsKICAgICAgICAgICAgcnVuQW50KGZpbGUudG9TdHJpbmcoKSxnZXRVbnB1Ymxpc2hUYXJnZXRzRm9yTW9kdWxlKCksZ2V0UHVibGlzaFByb3BlcnRpZXMoKSxtb25pdG9yKTsKICAgICAgICB9IGNhdGNoIChDb3JlRXhjZXB0aW9uIGUpIHsKICAgICAgICAgICAgSVN0YXR1cyBzID0gbmV3IFN0YXR1cyhJU3RhdHVzLkVSUk9SLENvcmVQbHVnaW4uUExVR0lOX0lELDAsR2VuZXJpY1NlcnZlckNvcmVNZXNzYWdlcy5lcnJvclJlbW92ZU1vZHVsZUFudHB1Ymxpc2hlcixlKTsKICAgICAgICAgICAgcmV0dXJuIG5ldyBJU3RhdHVzW10ge3N9OwogICAgICAgIH0KICAgICAgICByZXR1cm4gbnVsbDsKICAgIH0KfQo=