LyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogQ29weXJpZ2h0IChjKSAyMDA0IEV0ZXJhdGlvbiBCaWxpc2ltIEEuUy4KICogQWxsIHJpZ2h0cyByZXNlcnZlZC4goCBUaGlzIHByb2dyYW0gYW5kIHRoZSBhY2NvbXBhbnlpbmcgbWF0ZXJpYWxzCiAqIGFyZSBtYWRlIGF2YWlsYWJsZSB1bmRlciB0aGUgdGVybXMgb2YgdGhlIENvbW1vbiBQdWJsaWMgTGljZW5zZSB2MS4wCiAqIHdoaWNoIGFjY29tcGFuaWVzIHRoaXMgZGlzdHJpYnV0aW9uLCBhbmQgaXMgYXZhaWxhYmxlIGF0CiAqIGh0dHA6Ly93d3cuZWNsaXBzZS5vcmcvbGVnYWwvY3BsLXYxMC5odG1sCiAqIAogKiBDb250cmlidXRvcnM6CiAqICAgICBOYWNpIE0uIERhaSAtIGluaXRpYWwgQVBJIGFuZCBpbXBsZW1lbnRhdGlvbgogKiAKICogVEhJUyBTT0ZUV0FSRSBJUyBQUk9WSURFRCBgYEFTIElTJycgQU5EIEFOWSBFWFBSRVNTRUQgT1IgSU1QTElFRAogKiBXQVJSQU5USUVTLCBJTkNMVURJTkcsIEJVVCBOT1QgTElNSVRFRCBUTywgVEhFIElNUExJRUQgV0FSUkFOVElFUwogKiBPRiBNRVJDSEFOVEFCSUxJVFkgQU5EIEZJVE5FU1MgRk9SIEEgUEFSVElDVUxBUiBQVVJQT1NFIEFSRQogKiBESVNDTEFJTUVELiAgSU4gTk8gRVZFTlQgU0hBTEwgRVRFUkFUSU9OIEEuUy4gT1IKICogSVRTIENPTlRSSUJVVE9SUyBCRSBMSUFCTEUgRk9SIEFOWSBESVJFQ1QsIElORElSRUNULCBJTkNJREVOVEFMLAogKiBTUEVDSUFMLCBFWEVNUExBUlksIE9SIENPTlNFUVVFTlRJQUwgREFNQUdFUyAoSU5DTFVESU5HLCBCVVQgTk9UCiAqIExJTUlURUQgVE8sIFBST0NVUkVNRU5UIE9GIFNVQlNUSVRVVEUgR09PRFMgT1IgU0VSVklDRVM7IExPU1MgT0YKICogVVNFLCBEQVRBLCBPUiBQUk9GSVRTOyBPUiBCVVNJTkVTUyBJTlRFUlJVUFRJT04pIEhPV0VWRVIgQ0FVU0VEIEFORAogKiBPTiBBTlkgVEhFT1JZIE9GIExJQUJJTElUWSwgV0hFVEhFUiBJTiBDT05UUkFDVCwgU1RSSUNUIExJQUJJTElUWSwKICogT1IgVE9SVCAoSU5DTFVESU5HIE5FR0xJR0VOQ0UgT1IgT1RIRVJXSVNFKSBBUklTSU5HIElOIEFOWSBXQVkgT1VUCiAqIE9GIFRIRSBVU0UgT0YgVEhJUyBTT0ZUV0FSRSwgRVZFTiBJRiBBRFZJU0VEIE9GIFRIRSBQT1NTSUJJTElUWSBPRgogKiBTVUNIIERBTUFHRS4KICogPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT0KICoKICogVGhpcyBzb2Z0d2FyZSBjb25zaXN0cyBvZiB2b2x1bnRhcnkgY29udHJpYnV0aW9ucyBtYWRlIGJ5IG1hbnkKICogaW5kaXZpZHVhbHMgb24gYmVoYWxmIG9mIHRoZSBFdGVyYXRpb24gQmlsaXNpbSBBLlMuICBGb3IgbW9yZQogKiBpbmZvcm1hdGlvbiBvbiBldGVyYXRpb24sIHBsZWFzZSBzZWUKICogPGh0dHA6Ly93d3cuZXRlcmF0aW9uLmNvbS8+LgogKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwpwYWNrYWdlIG9yZy5lY2xpcHNlLmpzdC5zZXJ2ZXIuZ2VuZXJpYy5jb3JlLmludGVybmFsOwoKaW1wb3J0IGphdmEuaW8uRmlsZTsKaW1wb3J0IGphdmEuaW8uRmlsZU91dHB1dFN0cmVhbTsKaW1wb3J0IGphdmEuaW8uSU9FeGNlcHRpb247CmltcG9ydCBqYXZhLmlvLklucHV0U3RyZWFtOwppbXBvcnQgamF2YS5pby5PdXRwdXRTdHJlYW07CmltcG9ydCBqYXZhLm5ldC5VUkw7CmltcG9ydCBqYXZhLnV0aWwuSGFzaE1hcDsKaW1wb3J0IGphdmEudXRpbC5JdGVyYXRvcjsKaW1wb3J0IGphdmEudXRpbC5NYXA7CmltcG9ydCBqYXZhLnV0aWwuamFyLkphckVudHJ5OwppbXBvcnQgamF2YS51dGlsLmphci5KYXJGaWxlOwppbXBvcnQgb3JnLmVjbGlwc2UuYW50LmludGVybmFsLnVpLklBbnRVSUNvbnN0YW50czsKaW1wb3J0IG9yZy5lY2xpcHNlLmFudC5pbnRlcm5hbC51aS5sYXVuY2hDb25maWd1cmF0aW9ucy5JQW50TGF1bmNoQ29uZmlndXJhdGlvbkNvbnN0YW50czsKaW1wb3J0IG9yZy5lY2xpcHNlLmNvcmUucnVudGltZS5Db3JlRXhjZXB0aW9uOwppbXBvcnQgb3JnLmVjbGlwc2UuY29yZS5ydW50aW1lLklQcm9ncmVzc01vbml0b3I7CmltcG9ydCBvcmcuZWNsaXBzZS5jb3JlLnJ1bnRpbWUuSVN0YXR1czsKaW1wb3J0IG9yZy5lY2xpcHNlLmNvcmUucnVudGltZS5QbGF0Zm9ybTsKaW1wb3J0IG9yZy5lY2xpcHNlLmNvcmUucnVudGltZS5TdGF0dXM7CmltcG9ydCBvcmcuZWNsaXBzZS5kZWJ1Zy5jb3JlLkRlYnVnUGx1Z2luOwppbXBvcnQgb3JnLmVjbGlwc2UuZGVidWcuY29yZS5JTGF1bmNoQ29uZmlndXJhdGlvbjsKaW1wb3J0IG9yZy5lY2xpcHNlLmRlYnVnLmNvcmUuSUxhdW5jaENvbmZpZ3VyYXRpb25UeXBlOwppbXBvcnQgb3JnLmVjbGlwc2UuZGVidWcuY29yZS5JTGF1bmNoQ29uZmlndXJhdGlvbldvcmtpbmdDb3B5OwppbXBvcnQgb3JnLmVjbGlwc2UuZGVidWcuY29yZS5JTGF1bmNoTWFuYWdlcjsKaW1wb3J0IG9yZy5lY2xpcHNlLmRlYnVnLnVpLklEZWJ1Z1VJQ29uc3RhbnRzOwppbXBvcnQgb3JnLmVjbGlwc2UuamR0LmxhdW5jaGluZy5JSmF2YUxhdW5jaENvbmZpZ3VyYXRpb25Db25zdGFudHM7CmltcG9ydCBvcmcuZWNsaXBzZS5qc3Quc2VydmVyLmdlbmVyaWMuaW50ZXJuYWwuY29yZS51dGlsLkZpbGVVdGlsOwppbXBvcnQgb3JnLmVjbGlwc2UuanN0LnNlcnZlci5nZW5lcmljLnNlcnZlcnR5cGUuZGVmaW5pdGlvbi5Nb2R1bGU7CmltcG9ydCBvcmcuZWNsaXBzZS5qc3Quc2VydmVyLmdlbmVyaWMuc2VydmVydHlwZS5kZWZpbml0aW9uLlB1Ymxpc2hlckRhdGE7CmltcG9ydCBvcmcuZWNsaXBzZS5qc3Quc2VydmVyLmNvcmUuSUVKQk1vZHVsZTsKaW1wb3J0IG9yZy5lY2xpcHNlLmpzdC5zZXJ2ZXIuY29yZS5JRW50ZXJwcmlzZUFwcGxpY2F0aW9uOwppbXBvcnQgb3JnLmVjbGlwc2UuanN0LnNlcnZlci5jb3JlLklXZWJNb2R1bGU7CmltcG9ydCBvcmcuZWNsaXBzZS51aS5leHRlcm5hbHRvb2xzLmludGVybmFsLm1vZGVsLklFeHRlcm5hbFRvb2xDb25zdGFudHM7CmltcG9ydCBvcmcuZWNsaXBzZS53c3Quc2VydmVyLmNvcmUuSU1vZHVsZUFydGlmYWN0OwppbXBvcnQgb3JnLm9zZ2kuZnJhbWV3b3JrLkJ1bmRsZTsKLyoqCiAqIEFudCBiYXNlZCBwdWJsaXNoZXIuCiAqIEFsbCB0aGUgcHJvcGVydGllcyBkZWZpbmVkIGluIHRoZSBzZXJ2ZXIgZGVmaW5pdGlvbiBmaWxlIGFyZQogKiBwYXNzZWQgaW50byB0aGUgQU5UIGJ1aWxkIGZpbGUgYXMgcHJvcGVydGllcy4KICogSW4gYWRkaXRpb24gdG8gdGhlIHByb3BlcnRpZXMgZGVmaW5lZCBpbiB0aGUgc2VydmVyIGRlZmluaXRpb24KICogPEk+bW9kdWxlLmRpcjwvST4sIDxJPm1vZHVsZS5uYW1lLDwvST4gYW5kIDxJPnNlcnZlci5wdWJsaXNoLmRpcjwvST4gYXJlIGNvbXB1dGVkIGFuZCBwYXNzZWQgdG8gdGhlIAogKiBkZWZpbml0aW9uIGZpbGUuCiAqIDx1bD4KICogPGxpPm1vZHVsZS5kaXI6IGluY2x1ZGVzIHRoZSByb290IG9mIHRoZSBtb2R1bGUgcHJvamVjdCBmaWxlPC9saT4KICogPGxpPm1vZHVsZS5uYW1lOiB0aGUgbmFtZSBvZiB0aGUgbW9kdWxlPC9saT4KICogPGxpPnNlcnZlci5wdWJsaXNoLmRpcjogdGhlIGRpcmVjdG9yeSB0byBwdXQgdGhlIGRlcGxveW1lbnQgdW5pdHM8L2xpPgogKiA8L3VsPgogKgogKiBAYXV0aG9yIEdvcmtlbSBFcmNhbgogKi8KCnB1YmxpYyBjbGFzcyBBbnRQdWJsaXNoZXIgZXh0ZW5kcyBHZW5lcmljUHVibGlzaGVyewoJcHJpdmF0ZSBzdGF0aWMgZmluYWwgU3RyaW5nIEpBUl9QUk9UT0NPTF9QUkVGSVggPSAiamFyIjsKCgkvKioKCSAqIHB1Ymxpc2hlciBpZCBmb3IgQU5UIHB1Ymxpc2hlci4KCSAqLwoJcHVibGljIHN0YXRpYyBmaW5hbCBTdHJpbmcgUFVCTElTSEVSX0lEPSJvcmcuZWNsaXBzZS5qc3Quc2VydmVyLmdlbmVyaWMuYW50cHVibGlzaGVyIjsgLy8kTk9OLU5MUy0xJAogICAgCglwcml2YXRlIHN0YXRpYyBmaW5hbCBTdHJpbmcgUFJPUF9TRVJWRVJfUFVCTElTSF9ESVIgPSAic2VydmVyLnB1Ymxpc2guZGlyIjsvLyROT04tTkxTLTEkCglwcml2YXRlIHN0YXRpYyBmaW5hbCBTdHJpbmcgUFJPUF9NT0RVTEVfRElSID0gIm1vZHVsZS5kaXIiOy8vJE5PTi1OTFMtMSQKCXByaXZhdGUgc3RhdGljIGZpbmFsIFN0cmluZyBQUk9QX01PRFVMRV9OQU1FID0gIm1vZHVsZS5uYW1lIjsvLyROT04tTkxTLTEkCglwcml2YXRlIHN0YXRpYyBmaW5hbCBTdHJpbmcgTU9EVUxFX1BVQkxJU0hfVEFSR0VUX1BSRUZJWCA9ICJ0YXJnZXQucHVibGlzaC4iOyAvLyROT04tTkxTLTEkCglwcml2YXRlIHN0YXRpYyBmaW5hbCBTdHJpbmcgTU9EVUxFX1VOUFVCTElTSF9UQVJHRVRfUFJFRklYID0gInRhcmdldC51bnB1Ymxpc2guIjsvLyROT04tTkxTLTEkCglwcml2YXRlIHN0YXRpYyBmaW5hbCBTdHJpbmcgREFUQV9OQU1FX0JVSUxEX0ZJTEU9ImJ1aWxkLmZpbGUiOy8vJE5PTi1OTFMtMSQKCiAgICAvKiAobm9uLUphdmFkb2MpCgkgKiBAc2VlIG9yZy5lY2xpcHNlLnd0cC5zZXJ2ZXIuY29yZS5tb2RlbC5JUHVibGlzaGVyI3B1Ymxpc2gob3JnLmVjbGlwc2Uud3RwLnNlcnZlci5jb3JlLnJlc291cmNlcy5JTW9kdWxlUmVzb3VyY2VbXSwgb3JnLmVjbGlwc2UuY29yZS5ydW50aW1lLklQcm9ncmVzc01vbml0b3IpCgkgKi8KCXB1YmxpYyBJU3RhdHVzW10gcHVibGlzaChJTW9kdWxlQXJ0aWZhY3RbXSByZXNvdXJjZSwgSVByb2dyZXNzTW9uaXRvciBtb25pdG9yKXsKICAgICAgICB0cnl7CiAgICAgICAgCUZpbGUgZmlsZSA9IGNvbXB1dGVCdWlsZEZpbGUoKTsKICAgICAgICAJcnVuQW50KGZpbGUudG9TdHJpbmcoKSxnZXRQdWJsaXNoVGFyZ2V0c0Zvck1vZHVsZSgpLGdldFB1Ymxpc2hQcm9wZXJ0aWVzKCksbW9uaXRvcik7CiAgICAgICAgfWNhdGNoKENvcmVFeGNlcHRpb24gZSl7CiAgICAgICAgICAgIElTdGF0dXMgcyA9IG5ldyBTdGF0dXMoSVN0YXR1cy5FUlJPUixDb3JlUGx1Z2luLlBMVUdJTl9JRCwwLEdlbmVyaWNTZXJ2ZXJDb3JlTWVzc2FnZXMuZXJyb3JQdWJsaXNoQW50cHVibGlzaGVyLGUpOwogICAgICAgICAgICBDb3JlUGx1Z2luLmdldERlZmF1bHQoKS5nZXRMb2coKS5sb2cocyk7CiAgICAgICAgICAgIHJldHVybiBuZXcgSVN0YXR1c1tdIHtzfTsKICAgICAgICB9CgkJcmV0dXJuIG51bGw7Cgl9CgogICAgLyoqCiAgICAgKiAKICAgICAqIEByZXR1cm4KICAgICAqIEB0aHJvd3MgQ29yZUV4Y2VwdGlvbgogICAgICovCiAgICBwcml2YXRlIEZpbGUgY29tcHV0ZUJ1aWxkRmlsZSgpIHRocm93cyBDb3JlRXhjZXB0aW9uIHsKICAgICAgICBCdW5kbGUgYnVuZGxlID0gUGxhdGZvcm0uZ2V0QnVuZGxlKGdldFNlcnZlclJ1bnRpbWUoKS5nZXRTZXJ2ZXJUeXBlRGVmaW5pdGlvbigpLmdldENvbmZpZ3VyYXRpb25FbGVtZW50TmFtZXNwYWNlKCkpOwogICAgICAgIFVSTCBidW5kbGVVcmwgPWJ1bmRsZS5nZXRFbnRyeShnZXRCdWlsZEZpbGUoKSk7CiAgICAgICAgVVJMIGZpbGVVUkwgPSBGaWxlVXRpbC5yZXNvbHZlVVJMKGJ1bmRsZVVybCk7CiAgICAgICAgaWYoZmlsZVVSTC5nZXRQcm90b2NvbCgpLmVxdWFscyhKQVJfUFJPVE9DT0xfUFJFRklYKSl7CiAgICAgICAgCU91dHB1dFN0cmVhbSBvcz1udWxsOwogICAgICAgIAlJbnB1dFN0cmVhbSBpcz1udWxsOyAKICAgICAgICAJdHJ5ewogICAgICAgIAkJU3RyaW5nIGZpbGVuYW1lID1maWxlVVJMLmdldFBhdGgoKTsKICAgICAgICAJCVN0cmluZyBqYXJuYW1lPSBmaWxlVVJMLmdldEZpbGUoKS5zdWJzdHJpbmcoMCxmaWxlbmFtZS5pbmRleE9mKCchJykpOwogICAgICAgIAkJCiAgICAgICAgCQlGaWxlIGphckZpbGUgPSBuZXcgRmlsZShuZXcgVVJMKGphcm5hbWUpLmdldEZpbGUoKSk7CiAgICAgICAgCQlKYXJGaWxlIGphciA9IG5ldyBKYXJGaWxlKGphckZpbGUpOwogICAgICAgIAkJRmlsZSB0bXBGaWxlID0gRmlsZVV0aWwuY3JlYXRlVGVtcEZpbGUoZ2V0QnVpbGRGaWxlKCksQ29yZVBsdWdpbi5nZXREZWZhdWx0KCkuZ2V0U3RhdGVMb2NhdGlvbigpLnRvT1NTdHJpbmcoKSk7CiAgICAgICAgCQlvcyA9IG5ldyBGaWxlT3V0cHV0U3RyZWFtKHRtcEZpbGUpOwogICAgICAgIAkJU3RyaW5nIGVudHJ5bmFtZT0gZ2V0QnVpbGRGaWxlKCk7CiAgICAgICAgCQlpZiAoZW50cnluYW1lLnN0YXJ0c1dpdGgoIi8iKSkvLyROT04tTkxTLTEkCiAgICAgICAgCQkJZW50cnluYW1lPSBlbnRyeW5hbWUuc3Vic3RyaW5nKDEpOwogICAgICAgIAkJSmFyRW50cnkgZW50cnkgPSBqYXIuZ2V0SmFyRW50cnkoZW50cnluYW1lKTsKICAgICAgICAJCWlzID1qYXIuZ2V0SW5wdXRTdHJlYW0oZW50cnkpOwogICAgICAgIAkJRmlsZVV0aWwuY29weShpcyxvcyk7CiAgICAgICAgCQlyZXR1cm4gdG1wRmlsZTsKICAgICAgICAgCX0KICAgICAgICAJY2F0Y2ggKElPRXhjZXB0aW9uIGUpIHsKICAgICAgICAJCUlTdGF0dXMgcyA9IG5ldyBTdGF0dXMoSVN0YXR1cy5FUlJPUixDb3JlUGx1Z2luLlBMVUdJTl9JRCwwLCJlcnJvciBjcmVhdGluZyB0ZW1wb3JhcnkgYnVpbGQgZmlsZSIsZSk7Ly8kTk9OLU5MUy0xJAogICAgICAgICAgICAgICAgQ29yZVBsdWdpbi5nZXREZWZhdWx0KCkuZ2V0TG9nKCkubG9nKHMpOwoJCQkJdGhyb3cgbmV3IENvcmVFeGNlcHRpb24ocyk7CgkJCX0KICAgICAgICAJZmluYWxseXsKICAgICAgICAJCXRyeSB7CiAgICAgICAgCQkJaWYoaXMhPW51bGwpCiAgICAgICAgCQkJCWlzLmNsb3NlKCk7CgkJCQkJaWYob3MhPW51bGwpCgkJCQkJCW9zLmNsb3NlKCk7CgkJCQl9IGNhdGNoIChJT0V4Y2VwdGlvbiBlKSB7CgkJCQkJLy9pZ25vcmUKCQkJCX0KICAgICAgICAJfQogICAgICAgIH0KICAgICAgICBlbHNlewogICAgICAgIAlyZXR1cm4gRmlsZVV0aWwucmVzb2x2ZUZpbGUoZmlsZVVSTCk7CiAgICAgICAgfSAJCiAgICB9CiAgIAogICAgLyoqCiAgICAgKiBAcmV0dXJuCiAgICAgKi8KICAgIHByaXZhdGUgU3RyaW5nIGdldFB1Ymxpc2hUYXJnZXRzRm9yTW9kdWxlKCkgewogICAgCXJldHVybiBkb0dldFRhcmdldHMoTU9EVUxFX1BVQkxJU0hfVEFSR0VUX1BSRUZJWCtnZXRNb2R1bGVUeXBlSWQoKSk7CiAgICB9CgogICAgLyoqCiAgICAgKiBAcmV0dXJuCiAgICAgKi8KICAgIHByaXZhdGUgU3RyaW5nIGdldFVucHVibGlzaFRhcmdldHNGb3JNb2R1bGUoKSB7CiAgICAgICAgcmV0dXJuIGRvR2V0VGFyZ2V0cyhNT0RVTEVfVU5QVUJMSVNIX1RBUkdFVF9QUkVGSVgrZ2V0TW9kdWxlVHlwZUlkKCkpOwogICAgfQogICAgCiAgICAvKioKICAgICAqIEBwYXJhbSBkYXRhbmFtZQogICAgICogQHJldHVybgogICAgICovCiAgICBwcml2YXRlIFN0cmluZyBkb0dldFRhcmdldHMoU3RyaW5nIGRhdGFuYW1lKSB7CiAgICAJU3RyaW5nQnVmZmVyIGJ1ZmZlciA9IG5ldyBTdHJpbmdCdWZmZXIoKTsKICAgIAlJdGVyYXRvciBpdGVyYXRvciA9IGdldFNlcnZlclJ1bnRpbWUoKS5nZXRTZXJ2ZXJUeXBlRGVmaW5pdGlvbigpLmdldFB1Ymxpc2hlcihQVUJMSVNIRVJfSUQpLmdldFB1Ymxpc2hlcmRhdGEoKS5pdGVyYXRvcigpOwogICAgICAgIHdoaWxlKGl0ZXJhdG9yLmhhc05leHQoKSl7CiAgICAgICAgICAgIFB1Ymxpc2hlckRhdGEgZGF0YSA9IChQdWJsaXNoZXJEYXRhKWl0ZXJhdG9yLm5leHQoKTsKICAgICAgICAgICAgaWYoZGF0YW5hbWUuZXF1YWxzKGRhdGEuZ2V0RGF0YW5hbWUoKSkpIHsKICAgICAgICAgICAgICAgIGlmKGJ1ZmZlci5sZW5ndGgoKT4wKQogICAgICAgICAgICAgICAgCWJ1ZmZlci5hcHBlbmQoIiwiKTsvLyROT04tTkxTLTEkCiAgICAgICAgICAgIAlidWZmZXIuYXBwZW5kKGRhdGEuZ2V0RGF0YXZhbHVlKCkpOwogICAgICAgICAgICB9ICAgCiAgICAgICAgfQogICAgICAgIHJldHVybiBidWZmZXIudG9TdHJpbmcoKTsKICAgIH0KCiAgICBwcml2YXRlIFN0cmluZyBnZXRNb2R1bGVUeXBlSWQoKXsKICAgICAgICByZXR1cm4gZ2V0TW9kdWxlKClbMF0uZ2V0TW9kdWxlVHlwZSgpLmdldElkKCk7CiAgICB9CiAgICAKCXByaXZhdGUgU3RyaW5nIGdldEJ1aWxkRmlsZSgpCiAgICB7CiAgICAgICAgSXRlcmF0b3IgaXRlcmF0b3IgPSBnZXRTZXJ2ZXJSdW50aW1lKCkuZ2V0U2VydmVyVHlwZURlZmluaXRpb24oKS5nZXRQdWJsaXNoZXIoUFVCTElTSEVSX0lEKS5nZXRQdWJsaXNoZXJkYXRhKCkuaXRlcmF0b3IoKTsKICAgICAgICB3aGlsZShpdGVyYXRvci5oYXNOZXh0KCkpCiAgICAgICAgewogICAgICAgICAgICBQdWJsaXNoZXJEYXRhIGRhdGEgPSAoUHVibGlzaGVyRGF0YSlpdGVyYXRvci5uZXh0KCk7CiAgICAgICAgICAgIGlmKERBVEFfTkFNRV9CVUlMRF9GSUxFLmVxdWFscyhkYXRhLmdldERhdGFuYW1lKCkpKQogICAgICAgICAgICAgICAgcmV0dXJuIGdldFNlcnZlclJ1bnRpbWUoKS5nZXRTZXJ2ZXJUeXBlRGVmaW5pdGlvbigpLmdldFJlc29sdmVyKCkucmVzb2x2ZVByb3BlcnRpZXMoZGF0YS5nZXREYXRhdmFsdWUoKSk7CiAgICAgICAgfQogICAgICAgIHJldHVybiBudWxsOwogICAgfQoJCglwcml2YXRlIE1hcCBnZXRQdWJsaXNoUHJvcGVydGllcygpCgl7CiAgICAgICAgTWFwIHByb3BzID0gbmV3IEhhc2hNYXAoKTsKICAgICAgICAvLyBwYXNzIGFsbCBwcm9wZXJ0aWVzIHRvIGJ1aWxkIGZpbGUuCiAgICAgICAgTWFwIHNlcnZlclByb3BlcnRpZXMgPSBnZXRTZXJ2ZXIoKS5nZXRTZXJ2ZXJJbnN0YW5jZVByb3BlcnRpZXNJbXBsKCk7CiAgICAgICAgTWFwIHByb3BlcnRpZXMgPSBnZXRTZXJ2ZXJSdW50aW1lKCkuZ2V0U2VydmVySW5zdGFuY2VQcm9wZXJ0aWVzKCk7CiAgICAgICAgcHJvcGVydGllcy5wdXRBbGwoc2VydmVyUHJvcGVydGllcyk7CiAgICAgICAgSXRlcmF0b3IgcHJvcGVydHlJdGVyYXRvciA9IHByb3BlcnRpZXMua2V5U2V0KCkuaXRlcmF0b3IoKTsKICAgICAgICB3aGlsZShwcm9wZXJ0eUl0ZXJhdG9yLmhhc05leHQoKSkKICAgICAgICB7CiAgICAgICAgICAgIFN0cmluZyBwcm9wZXJ0eSA9IChTdHJpbmcpcHJvcGVydHlJdGVyYXRvci5uZXh0KCk7CiAgICAgICAgICAgIFN0cmluZyB2YWx1ZSA9IChTdHJpbmcpcHJvcGVydGllcy5nZXQocHJvcGVydHkpOwogICAgICAgICAgICBpZih2YWx1ZSE9bnVsbCAmJiB2YWx1ZS50cmltKCkubGVuZ3RoKCk+MCkKICAgICAgICAgICAgCXByb3BzLnB1dChwcm9wZXJ0eSxwcm9wZXJ0aWVzLmdldChwcm9wZXJ0eSkpOwogICAgICAgIH0KICAgICAgICBNb2R1bGUgbW9kdWxlID0gIGdldFNlcnZlclJ1bnRpbWUoKS5nZXRTZXJ2ZXJUeXBlRGVmaW5pdGlvbigpLmdldE1vZHVsZShnZXRNb2R1bGVUeXBlSWQoKSk7CgkJU3RyaW5nIG1vZERpciA9IG1vZHVsZS5nZXRQdWJsaXNoRGlyKCk7CgkJbW9kRGlyID0gZ2V0U2VydmVyUnVudGltZSgpLmdldFNlcnZlclR5cGVEZWZpbml0aW9uKCkuZ2V0UmVzb2x2ZXIoKS5yZXNvbHZlUHJvcGVydGllcyhtb2REaXIpOwoKCQlJV2ViTW9kdWxlIHdlYk1vZHVsZSA9IChJV2ViTW9kdWxlKWdldE1vZHVsZSgpWzBdLmxvYWRBZGFwdGVyKElXZWJNb2R1bGUuY2xhc3MsbnVsbCk7CiAgICAgICAgSUVKQk1vZHVsZSBlamJNb2R1bGUgPSAoSUVKQk1vZHVsZSlnZXRNb2R1bGUoKVswXS5sb2FkQWRhcHRlcihJRUpCTW9kdWxlLmNsYXNzLG51bGwpOwoJCUlFbnRlcnByaXNlQXBwbGljYXRpb24gZWFyTW9kdWxlID0gKElFbnRlcnByaXNlQXBwbGljYXRpb24pZ2V0TW9kdWxlKClbMF0ubG9hZEFkYXB0ZXIoSUVudGVycHJpc2VBcHBsaWNhdGlvbi5jbGFzcyxudWxsKTsKICAgICAgICAKICAgICAgICBTdHJpbmcgbW9kdWxlTmFtZT0idW5rbm93bm1vZHVsZSI7Ly8kTk9OLU5MUy0xJAogICAgICAgIFN0cmluZyBtb2R1bGVEaXI9IiI7Ly8kTk9OLU5MUy0xJAogICAgICAgIGlmKHdlYk1vZHVsZSE9bnVsbCl7ICAgIAogICAgICAgICAgICBtb2R1bGVOYW1lID0gdGhpcy5ndWVzc01vZHVsZU5hbWUod2ViTW9kdWxlKTsKICAgICAgICAgICAgbW9kdWxlRGlyID0gd2ViTW9kdWxlLmdldExvY2F0aW9uKCkudG9TdHJpbmcoKTsKICAgICAgICB9CiAgICAgICAgaWYoZWpiTW9kdWxlIT1udWxsKXsgIAogICAgICAgICAgICBtb2R1bGVOYW1lID0gZ2V0TW9kdWxlKClbMF0uZ2V0TmFtZSgpOwogICAgICAgICAgICBtb2R1bGVEaXI9IGVqYk1vZHVsZS5nZXRMb2NhdGlvbigpLnRvU3RyaW5nKCk7CiAgICAgICAgfQogICAgICAgIGlmKGVhck1vZHVsZSE9bnVsbCkKICAgICAgICB7CiAgICAgICAgCW1vZHVsZU5hbWUgPSBnZXRNb2R1bGUoKVswXS5nZXROYW1lKCk7CiAgICAgICAgCW1vZHVsZURpciA9IGVhck1vZHVsZS5nZXRMb2NhdGlvbigpLnRvU3RyaW5nKCk7CiAgICAgICAgfQoJCXByb3BzLnB1dChQUk9QX01PRFVMRV9OQU1FLG1vZHVsZU5hbWUpOwoJCXByb3BzLnB1dChQUk9QX01PRFVMRV9ESVIsbW9kdWxlRGlyKTsKCQlwcm9wcy5wdXQoUFJPUF9TRVJWRVJfUFVCTElTSF9ESVIsbW9kRGlyKTsKCQlyZXR1cm4gcHJvcHM7Cgl9CgoJcHJpdmF0ZSBTdHJpbmcgZ3Vlc3NNb2R1bGVOYW1lKElXZWJNb2R1bGUgd2ViTW9kdWxlKSB7CgkJU3RyaW5nIG1vZHVsZU5hbWUgPSBnZXRNb2R1bGUoKVswXS5nZXROYW1lKCk7IAoJCVN0cmluZyBjb250ZXh0Um9vdCA9IHdlYk1vZHVsZS5nZXRDb250ZXh0Um9vdCgpOwoJCWlmKGNvbnRleHRSb290LmNoYXJBdCgwKSA9PSAnLycpCgkJCW1vZHVsZU5hbWUgPSBjb250ZXh0Um9vdC5zdWJzdHJpbmcoMSk7CgkJcmV0dXJuIG1vZHVsZU5hbWU7Cgl9CgoJcHJpdmF0ZSB2b2lkIHJ1bkFudChTdHJpbmcgYnVpbGRGaWxlLFN0cmluZyB0YXJnZXRzLE1hcCBwcm9wZXJ0aWVzICxJUHJvZ3Jlc3NNb25pdG9yIG1vbml0b3IpdGhyb3dzIENvcmVFeGNlcHRpb257CgkJSUxhdW5jaE1hbmFnZXIgbGF1bmNoTWFuYWdlciA9IERlYnVnUGx1Z2luLmdldERlZmF1bHQoKS5nZXRMYXVuY2hNYW5hZ2VyKCk7CgkJSUxhdW5jaENvbmZpZ3VyYXRpb25UeXBlIHR5cGUgPSBsYXVuY2hNYW5hZ2VyLmdldExhdW5jaENvbmZpZ3VyYXRpb25UeXBlKElBbnRMYXVuY2hDb25maWd1cmF0aW9uQ29uc3RhbnRzLklEX0FOVF9MQVVOQ0hfQ09ORklHVVJBVElPTl9UWVBFKTsKCgkJSUxhdW5jaENvbmZpZ3VyYXRpb25Xb3JraW5nQ29weSB3Yz0gdHlwZS5uZXdJbnN0YW5jZShudWxsLHByb3BlcnRpZXMuZ2V0KFBST1BfTU9EVUxFX05BTUUpKyIgbW9kdWxlIHB1Ymxpc2hlciIpOwoJCXdjLnNldENvbnRhaW5lcihudWxsKTsKCQl3Yy5zZXRBdHRyaWJ1dGUoSUV4dGVybmFsVG9vbENvbnN0YW50cy5BVFRSX0xPQ0FUSU9OLCBidWlsZEZpbGUpOwoJCXdjLnNldEF0dHJpYnV0ZShJSmF2YUxhdW5jaENvbmZpZ3VyYXRpb25Db25zdGFudHMuQVRUUl9DTEFTU1BBVEhfUFJPVklERVIsIm9yZy5lY2xpcHNlLmFudC51aS5BbnRDbGFzc3BhdGhQcm92aWRlciIpOwoJCXdjLnNldEF0dHJpYnV0ZShJQW50TGF1bmNoQ29uZmlndXJhdGlvbkNvbnN0YW50cy5BVFRSX0FOVF9UQVJHRVRTLHRhcmdldHMpOwoJCXdjLnNldEF0dHJpYnV0ZShJQW50TGF1bmNoQ29uZmlndXJhdGlvbkNvbnN0YW50cy5BVFRSX0FOVF9QUk9QRVJUSUVTLHByb3BlcnRpZXMpOwoJCXdjLnNldEF0dHJpYnV0ZShJRGVidWdVSUNvbnN0YW50cy5BVFRSX0xBVU5DSF9JTl9CQUNLR1JPVU5ELGZhbHNlKTsKCQl3Yy5zZXRBdHRyaWJ1dGUoSURlYnVnVUlDb25zdGFudHMuQVRUUl9DQVBUVVJFX0lOX0NPTlNPTEUsdHJ1ZSk7CgkJd2Muc2V0QXR0cmlidXRlKElEZWJ1Z1VJQ29uc3RhbnRzLkFUVFJfUFJJVkFURSx0cnVlKTsKCQkKCQl3Yy5zZXRBdHRyaWJ1dGUoSUphdmFMYXVuY2hDb25maWd1cmF0aW9uQ29uc3RhbnRzLkFUVFJfU09VUkNFX1BBVEhfUFJPVklERVIsICJvcmcuZWNsaXBzZS5hbnQudWkuQW50Q2xhc3NwYXRoUHJvdmlkZXIiKTsgCgkJd2Muc2V0QXR0cmlidXRlKElKYXZhTGF1bmNoQ29uZmlndXJhdGlvbkNvbnN0YW50cy5BVFRSX1ZNX0lOU1RBTExfTkFNRSxnZXRTZXJ2ZXJSdW50aW1lKCkuZ2V0Vk1JbnN0YWxsKCkuZ2V0TmFtZSgpKTsKCQl3Yy5zZXRBdHRyaWJ1dGUoSUphdmFMYXVuY2hDb25maWd1cmF0aW9uQ29uc3RhbnRzLkFUVFJfVk1fSU5TVEFMTF9UWVBFLGdldFNlcnZlclJ1bnRpbWUoKS5nZXRWTUluc3RhbGwoKS5nZXRWTUluc3RhbGxUeXBlKCkuZ2V0SWQoKSk7CgkJd2Muc2V0QXR0cmlidXRlKElKYXZhTGF1bmNoQ29uZmlndXJhdGlvbkNvbnN0YW50cy5BVFRSX01BSU5fVFlQRV9OQU1FLCAib3JnLmVjbGlwc2UuYW50LmludGVybmFsLnVpLmFudHN1cHBvcnQuSW50ZXJuYWxBbnRSdW5uZXIiKTsKCQl3Yy5zZXRBdHRyaWJ1dGUoRGVidWdQbHVnaW4uQVRUUl9QUk9DRVNTX0ZBQ1RPUllfSUQsIElBbnRVSUNvbnN0YW50cy5SRU1PVEVfQU5UX1BST0NFU1NfRkFDVE9SWV9JRCk7CgkJCgkJc2V0dXBBbnRMYXVuY2hDb25maWd1cmF0aW9uKHdjKTsKCQkKCQkKCQlJTGF1bmNoQ29uZmlndXJhdGlvbiBsYXVuY2hDb25maWcgPSB3Yy5kb1NhdmUoKTsKICAgICAgICBsYXVuY2hDb25maWcubGF1bmNoKCJydW4iLG1vbml0b3IpOwoJfQoKCgogICAgLyogKG5vbi1KYXZhZG9jKQogICAgICovCiAgICBwcm90ZWN0ZWQgdm9pZCBzZXR1cEFudExhdW5jaENvbmZpZ3VyYXRpb24oSUxhdW5jaENvbmZpZ3VyYXRpb25Xb3JraW5nQ29weSB3YykgewkJCgl9CgoJLyogKG5vbi1KYXZhZG9jKQogICAgICogQHNlZSBvcmcuZWNsaXBzZS5qc3Quc2VydmVyLmdlbmVyaWMuaW50ZXJuYWwuY29yZS5HZW5lcmljUHVibGlzaGVyI3VucHVibGlzaChvcmcuZWNsaXBzZS53c3Quc2VydmVyLmNvcmUuSU1vZHVsZSwgb3JnLmVjbGlwc2UuY29yZS5ydW50aW1lLklQcm9ncmVzc01vbml0b3IpCiAgICAgKi8KICAgIHB1YmxpYyBJU3RhdHVzW10gdW5wdWJsaXNoKElQcm9ncmVzc01vbml0b3IgbW9uaXRvcikgewogICAgICAgIHRyeSB7CiAgICAgICAgCSBGaWxlIGZpbGUgPSBjb21wdXRlQnVpbGRGaWxlKCk7CiAgICAgICAgICAgIHJ1bkFudChmaWxlLnRvU3RyaW5nKCksZ2V0VW5wdWJsaXNoVGFyZ2V0c0Zvck1vZHVsZSgpLGdldFB1Ymxpc2hQcm9wZXJ0aWVzKCksbW9uaXRvcik7CiAgICAgICAgfSBjYXRjaCAoQ29yZUV4Y2VwdGlvbiBlKSB7CiAgICAgICAgICAgIElTdGF0dXMgcyA9IG5ldyBTdGF0dXMoSVN0YXR1cy5FUlJPUixDb3JlUGx1Z2luLlBMVUdJTl9JRCwwLEdlbmVyaWNTZXJ2ZXJDb3JlTWVzc2FnZXMuZXJyb3JSZW1vdmVNb2R1bGVBbnRwdWJsaXNoZXIsZSk7CiAgICAgICAgICAgIHJldHVybiBuZXcgSVN0YXR1c1tdIHtzfTsKICAgICAgICB9CiAgICAgICAgcmV0dXJuIG51bGw7CiAgICB9Cn0K