LyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogQ29weXJpZ2h0IChjKSAyMDA0IEV0ZXJhdGlvbiBCaWxpc2ltIEEuUy4KICogQWxsIHJpZ2h0cyByZXNlcnZlZC4goCBUaGlzIHByb2dyYW0gYW5kIHRoZSBhY2NvbXBhbnlpbmcgbWF0ZXJpYWxzCiAqIGFyZSBtYWRlIGF2YWlsYWJsZSB1bmRlciB0aGUgdGVybXMgb2YgdGhlIENvbW1vbiBQdWJsaWMgTGljZW5zZSB2MS4wCiAqIHdoaWNoIGFjY29tcGFuaWVzIHRoaXMgZGlzdHJpYnV0aW9uLCBhbmQgaXMgYXZhaWxhYmxlIGF0CiAqIGh0dHA6Ly93d3cuZWNsaXBzZS5vcmcvbGVnYWwvY3BsLXYxMC5odG1sCiAqIAogKiBDb250cmlidXRvcnM6CiAqICAgICBOYWNpIE0uIERhaSAtIGluaXRpYWwgQVBJIGFuZCBpbXBsZW1lbnRhdGlvbgogKiAKICogVEhJUyBTT0ZUV0FSRSBJUyBQUk9WSURFRCBgYEFTIElTJycgQU5EIEFOWSBFWFBSRVNTRUQgT1IgSU1QTElFRAogKiBXQVJSQU5USUVTLCBJTkNMVURJTkcsIEJVVCBOT1QgTElNSVRFRCBUTywgVEhFIElNUExJRUQgV0FSUkFOVElFUwogKiBPRiBNRVJDSEFOVEFCSUxJVFkgQU5EIEZJVE5FU1MgRk9SIEEgUEFSVElDVUxBUiBQVVJQT1NFIEFSRQogKiBESVNDTEFJTUVELiAgSU4gTk8gRVZFTlQgU0hBTEwgRVRFUkFUSU9OIEEuUy4gT1IKICogSVRTIENPTlRSSUJVVE9SUyBCRSBMSUFCTEUgRk9SIEFOWSBESVJFQ1QsIElORElSRUNULCBJTkNJREVOVEFMLAogKiBTUEVDSUFMLCBFWEVNUExBUlksIE9SIENPTlNFUVVFTlRJQUwgREFNQUdFUyAoSU5DTFVESU5HLCBCVVQgTk9UCiAqIExJTUlURUQgVE8sIFBST0NVUkVNRU5UIE9GIFNVQlNUSVRVVEUgR09PRFMgT1IgU0VSVklDRVM7IExPU1MgT0YKICogVVNFLCBEQVRBLCBPUiBQUk9GSVRTOyBPUiBCVVNJTkVTUyBJTlRFUlJVUFRJT04pIEhPV0VWRVIgQ0FVU0VEIEFORAogKiBPTiBBTlkgVEhFT1JZIE9GIExJQUJJTElUWSwgV0hFVEhFUiBJTiBDT05UUkFDVCwgU1RSSUNUIExJQUJJTElUWSwKICogT1IgVE9SVCAoSU5DTFVESU5HIE5FR0xJR0VOQ0UgT1IgT1RIRVJXSVNFKSBBUklTSU5HIElOIEFOWSBXQVkgT1VUCiAqIE9GIFRIRSBVU0UgT0YgVEhJUyBTT0ZUV0FSRSwgRVZFTiBJRiBBRFZJU0VEIE9GIFRIRSBQT1NTSUJJTElUWSBPRgogKiBTVUNIIERBTUFHRS4KICogPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT0KICoKICogVGhpcyBzb2Z0d2FyZSBjb25zaXN0cyBvZiB2b2x1bnRhcnkgY29udHJpYnV0aW9ucyBtYWRlIGJ5IG1hbnkKICogaW5kaXZpZHVhbHMgb24gYmVoYWxmIG9mIHRoZSBFdGVyYXRpb24gQmlsaXNpbSBBLlMuICBGb3IgbW9yZQogKiBpbmZvcm1hdGlvbiBvbiBldGVyYXRpb24sIHBsZWFzZSBzZWUKICogPGh0dHA6Ly93d3cuZXRlcmF0aW9uLmNvbS8+LgogKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwpwYWNrYWdlIG9yZy5lY2xpcHNlLmpzdC5zZXJ2ZXIuZ2VuZXJpYy5jb3JlLmludGVybmFsOwoKaW1wb3J0IGphdmEuaW8uRmlsZTsKaW1wb3J0IGphdmEuaW8uRmlsZU91dHB1dFN0cmVhbTsKaW1wb3J0IGphdmEuaW8uSU9FeGNlcHRpb247CmltcG9ydCBqYXZhLmlvLklucHV0U3RyZWFtOwppbXBvcnQgamF2YS5pby5PdXRwdXRTdHJlYW07CmltcG9ydCBqYXZhLm5ldC5VUkw7CmltcG9ydCBqYXZhLnV0aWwuSGFzaE1hcDsKaW1wb3J0IGphdmEudXRpbC5JdGVyYXRvcjsKaW1wb3J0IGphdmEudXRpbC5NYXA7CmltcG9ydCBqYXZhLnV0aWwuamFyLkphckVudHJ5OwppbXBvcnQgamF2YS51dGlsLmphci5KYXJGaWxlOwppbXBvcnQgb3JnLmVjbGlwc2UuYW50LmludGVybmFsLnVpLklBbnRVSUNvbnN0YW50czsKaW1wb3J0IG9yZy5lY2xpcHNlLmFudC5pbnRlcm5hbC51aS5sYXVuY2hDb25maWd1cmF0aW9ucy5JQW50TGF1bmNoQ29uZmlndXJhdGlvbkNvbnN0YW50czsKaW1wb3J0IG9yZy5lY2xpcHNlLmNvcmUucnVudGltZS5Db3JlRXhjZXB0aW9uOwppbXBvcnQgb3JnLmVjbGlwc2UuY29yZS5ydW50aW1lLklQcm9ncmVzc01vbml0b3I7CmltcG9ydCBvcmcuZWNsaXBzZS5jb3JlLnJ1bnRpbWUuSVN0YXR1czsKaW1wb3J0IG9yZy5lY2xpcHNlLmNvcmUucnVudGltZS5QbGF0Zm9ybTsKaW1wb3J0IG9yZy5lY2xpcHNlLmNvcmUucnVudGltZS5TdGF0dXM7CmltcG9ydCBvcmcuZWNsaXBzZS5kZWJ1Zy5jb3JlLkRlYnVnUGx1Z2luOwppbXBvcnQgb3JnLmVjbGlwc2UuZGVidWcuY29yZS5JTGF1bmNoQ29uZmlndXJhdGlvbjsKaW1wb3J0IG9yZy5lY2xpcHNlLmRlYnVnLmNvcmUuSUxhdW5jaENvbmZpZ3VyYXRpb25UeXBlOwppbXBvcnQgb3JnLmVjbGlwc2UuZGVidWcuY29yZS5JTGF1bmNoQ29uZmlndXJhdGlvbldvcmtpbmdDb3B5OwppbXBvcnQgb3JnLmVjbGlwc2UuZGVidWcuY29yZS5JTGF1bmNoTWFuYWdlcjsKaW1wb3J0IG9yZy5lY2xpcHNlLmRlYnVnLnVpLklEZWJ1Z1VJQ29uc3RhbnRzOwppbXBvcnQgb3JnLmVjbGlwc2UuamR0LmxhdW5jaGluZy5JSmF2YUxhdW5jaENvbmZpZ3VyYXRpb25Db25zdGFudHM7CmltcG9ydCBvcmcuZWNsaXBzZS5qc3Quc2VydmVyLmdlbmVyaWMuaW50ZXJuYWwuY29yZS51dGlsLkZpbGVVdGlsOwppbXBvcnQgb3JnLmVjbGlwc2UuanN0LnNlcnZlci5nZW5lcmljLnNlcnZlcnR5cGUuZGVmaW5pdGlvbi5Nb2R1bGU7CmltcG9ydCBvcmcuZWNsaXBzZS5qc3Quc2VydmVyLmdlbmVyaWMuc2VydmVydHlwZS5kZWZpbml0aW9uLlB1Ymxpc2hlckRhdGE7CmltcG9ydCBvcmcuZWNsaXBzZS5qc3Quc2VydmVyLmNvcmUuSUVKQk1vZHVsZTsKaW1wb3J0IG9yZy5lY2xpcHNlLmpzdC5zZXJ2ZXIuY29yZS5JRW50ZXJwcmlzZUFwcGxpY2F0aW9uOwppbXBvcnQgb3JnLmVjbGlwc2UuanN0LnNlcnZlci5jb3JlLklXZWJNb2R1bGU7CmltcG9ydCBvcmcuZWNsaXBzZS51aS5leHRlcm5hbHRvb2xzLmludGVybmFsLm1vZGVsLklFeHRlcm5hbFRvb2xDb25zdGFudHM7CmltcG9ydCBvcmcuZWNsaXBzZS53c3Quc2VydmVyLmNvcmUuSU1vZHVsZUFydGlmYWN0OwppbXBvcnQgb3JnLm9zZ2kuZnJhbWV3b3JrLkJ1bmRsZTsKLyoqCiAqIEFudCBiYXNlZCBwdWJsaXNoZXIuCiAqIEFsbCB0aGUgcHJvcGVydGllcyBkZWZpbmVkIGluIHRoZSBzZXJ2ZXIgZGVmaW5pdGlvbiBmaWxlIGFyZQogKiBwYXNzZWQgaW50byB0aGUgQU5UIGJ1aWxkIGZpbGUgYXMgcHJvcGVydGllcy4KICogSW4gYWRkaXRpb24gdG8gdGhlIHByb3BlcnRpZXMgZGVmaW5lZCBpbiB0aGUgc2VydmVyIGRlZmluaXRpb24KICogPEk+bW9kdWxlLmRpcjwvST4sIDxJPm1vZHVsZS5uYW1lLDwvST4gYW5kIDxJPnNlcnZlci5wdWJsaXNoLmRpcjwvST4gYXJlIGNvbXB1dGVkIGFuZCBwYXNzZWQgdG8gdGhlIAogKiBkZWZpbml0aW9uIGZpbGUuCiAqIDx1bD4KICogPGxpPm1vZHVsZS5kaXI6IGluY2x1ZGVzIHRoZSByb290IG9mIHRoZSBtb2R1bGUgcHJvamVjdCBmaWxlPC9saT4KICogPGxpPm1vZHVsZS5uYW1lOiB0aGUgbmFtZSBvZiB0aGUgbW9kdWxlPC9saT4KICogPGxpPnNlcnZlci5wdWJsaXNoLmRpcjogdGhlIGRpcmVjdG9yeSB0byBwdXQgdGhlIGRlcGxveW1lbnQgdW5pdHM8L2xpPgogKiA8bGk+cHJvamVjdC53b3JraW5nLmRpcjogdGhlIHdvcmtpbmcgZGlyIG9mIHRoZSBwcm9qZWN0IHRoYXQgZGVwbG95ZWQgbW9kdWxlIGlzIGluPC9saT4KICogPC91bD4KICoKICogQGF1dGhvciBHb3JrZW0gRXJjYW4KICovCgpwdWJsaWMgY2xhc3MgQW50UHVibGlzaGVyIGV4dGVuZHMgR2VuZXJpY1B1Ymxpc2hlcnsKCXByaXZhdGUgc3RhdGljIGZpbmFsIFN0cmluZyBKQVJfUFJPVE9DT0xfUFJFRklYID0gImphciI7CgoJLyoqCgkgKiBwdWJsaXNoZXIgaWQgZm9yIEFOVCBwdWJsaXNoZXIuCgkgKi8KCXB1YmxpYyBzdGF0aWMgZmluYWwgU3RyaW5nIFBVQkxJU0hFUl9JRD0ib3JnLmVjbGlwc2UuanN0LnNlcnZlci5nZW5lcmljLmFudHB1Ymxpc2hlciI7IC8vJE5PTi1OTFMtMSQKICAgIAoJcHJpdmF0ZSBzdGF0aWMgZmluYWwgU3RyaW5nIFBST1BfU0VSVkVSX1BVQkxJU0hfRElSID0gInNlcnZlci5wdWJsaXNoLmRpciI7Ly8kTk9OLU5MUy0xJAoJcHJpdmF0ZSBzdGF0aWMgZmluYWwgU3RyaW5nIFBST1BfUFJPSkVDVF9XT1JLSU5HX0RJUj0gInByb2plY3Qud29ya2luZy5kaXIiOy8vJE5PTi1OTFMtMSQKCXByaXZhdGUgc3RhdGljIGZpbmFsIFN0cmluZyBQUk9QX01PRFVMRV9ESVIgPSAibW9kdWxlLmRpciI7Ly8kTk9OLU5MUy0xJAoJcHJpdmF0ZSBzdGF0aWMgZmluYWwgU3RyaW5nIFBST1BfTU9EVUxFX05BTUUgPSAibW9kdWxlLm5hbWUiOy8vJE5PTi1OTFMtMSQKCXByaXZhdGUgc3RhdGljIGZpbmFsIFN0cmluZyBNT0RVTEVfUFVCTElTSF9UQVJHRVRfUFJFRklYID0gInRhcmdldC5wdWJsaXNoLiI7IC8vJE5PTi1OTFMtMSQKCXByaXZhdGUgc3RhdGljIGZpbmFsIFN0cmluZyBNT0RVTEVfVU5QVUJMSVNIX1RBUkdFVF9QUkVGSVggPSAidGFyZ2V0LnVucHVibGlzaC4iOy8vJE5PTi1OTFMtMSQKCXByaXZhdGUgc3RhdGljIGZpbmFsIFN0cmluZyBEQVRBX05BTUVfQlVJTERfRklMRT0iYnVpbGQuZmlsZSI7Ly8kTk9OLU5MUy0xJAoJCgogICAgLyogKG5vbi1KYXZhZG9jKQoJICogQHNlZSBvcmcuZWNsaXBzZS53dHAuc2VydmVyLmNvcmUubW9kZWwuSVB1Ymxpc2hlciNwdWJsaXNoKG9yZy5lY2xpcHNlLnd0cC5zZXJ2ZXIuY29yZS5yZXNvdXJjZXMuSU1vZHVsZVJlc291cmNlW10sIG9yZy5lY2xpcHNlLmNvcmUucnVudGltZS5JUHJvZ3Jlc3NNb25pdG9yKQoJICovCglwdWJsaWMgSVN0YXR1c1tdIHB1Ymxpc2goSU1vZHVsZUFydGlmYWN0W10gcmVzb3VyY2UsIElQcm9ncmVzc01vbml0b3IgbW9uaXRvcil7CgkJaWYoZ2V0TW9kdWxlKCkubGVuZ3RoPjEpLy8gb25seSByZXNwb25kIHRvIHJvb3QgbW9kdWxlIGNhbGxzLiAKCQkJcmV0dXJuIG51bGw7CgkJdHJ5ewogICAgICAgIAlGaWxlIGZpbGUgPSBjb21wdXRlQnVpbGRGaWxlKCk7CiAgICAgICAgCXJ1bkFudChmaWxlLnRvU3RyaW5nKCksZ2V0UHVibGlzaFRhcmdldHNGb3JNb2R1bGUoKSxnZXRQdWJsaXNoUHJvcGVydGllcygpLG1vbml0b3IpOwogICAgICAgIH1jYXRjaChDb3JlRXhjZXB0aW9uIGUpewogICAgICAgICAgICBJU3RhdHVzIHMgPSBuZXcgU3RhdHVzKElTdGF0dXMuRVJST1IsQ29yZVBsdWdpbi5QTFVHSU5fSUQsMCxHZW5lcmljU2VydmVyQ29yZU1lc3NhZ2VzLmVycm9yUHVibGlzaEFudHB1Ymxpc2hlcixlKTsKICAgICAgICAgICAgQ29yZVBsdWdpbi5nZXREZWZhdWx0KCkuZ2V0TG9nKCkubG9nKHMpOwogICAgICAgICAgICByZXR1cm4gbmV3IElTdGF0dXNbXSB7c307CiAgICAgICAgfQoJCXJldHVybiBudWxsOwoJfQoKICAgIC8qKgogICAgICogCiAgICAgKiBAcmV0dXJuCiAgICAgKiBAdGhyb3dzIENvcmVFeGNlcHRpb24KICAgICAqLwogICAgcHJpdmF0ZSBGaWxlIGNvbXB1dGVCdWlsZEZpbGUoKSB0aHJvd3MgQ29yZUV4Y2VwdGlvbiB7CiAgICAgICAgQnVuZGxlIGJ1bmRsZSA9IFBsYXRmb3JtLmdldEJ1bmRsZShnZXRTZXJ2ZXJSdW50aW1lKCkuZ2V0U2VydmVyVHlwZURlZmluaXRpb24oKS5nZXRDb25maWd1cmF0aW9uRWxlbWVudE5hbWVzcGFjZSgpKTsKICAgICAgICBVUkwgYnVuZGxlVXJsID1idW5kbGUuZ2V0RW50cnkoZ2V0QnVpbGRGaWxlKCkpOwogICAgICAgIFVSTCBmaWxlVVJMID0gRmlsZVV0aWwucmVzb2x2ZVVSTChidW5kbGVVcmwpOwogICAgICAgIGlmKGZpbGVVUkwuZ2V0UHJvdG9jb2woKS5lcXVhbHMoSkFSX1BST1RPQ09MX1BSRUZJWCkpewogICAgICAgIAlPdXRwdXRTdHJlYW0gb3M9bnVsbDsKICAgICAgICAJSW5wdXRTdHJlYW0gaXM9bnVsbDsgCiAgICAgICAgCXRyeXsKICAgICAgICAJCVN0cmluZyBmaWxlbmFtZSA9ZmlsZVVSTC5nZXRQYXRoKCk7CiAgICAgICAgCQlTdHJpbmcgamFybmFtZT0gZmlsZVVSTC5nZXRGaWxlKCkuc3Vic3RyaW5nKDAsZmlsZW5hbWUuaW5kZXhPZignIScpKTsKICAgICAgICAJCQogICAgICAgIAkJRmlsZSBqYXJGaWxlID0gbmV3IEZpbGUobmV3IFVSTChqYXJuYW1lKS5nZXRGaWxlKCkpOwogICAgICAgIAkJSmFyRmlsZSBqYXIgPSBuZXcgSmFyRmlsZShqYXJGaWxlKTsKICAgICAgICAJCUZpbGUgdG1wRmlsZSA9IEZpbGVVdGlsLmNyZWF0ZVRlbXBGaWxlKGdldEJ1aWxkRmlsZSgpLENvcmVQbHVnaW4uZ2V0RGVmYXVsdCgpLmdldFN0YXRlTG9jYXRpb24oKS50b09TU3RyaW5nKCkpOwogICAgICAgIAkJb3MgPSBuZXcgRmlsZU91dHB1dFN0cmVhbSh0bXBGaWxlKTsKICAgICAgICAJCVN0cmluZyBlbnRyeW5hbWU9IGdldEJ1aWxkRmlsZSgpOwogICAgICAgIAkJaWYgKGVudHJ5bmFtZS5zdGFydHNXaXRoKCIvIikpLy8kTk9OLU5MUy0xJAogICAgICAgIAkJCWVudHJ5bmFtZT0gZW50cnluYW1lLnN1YnN0cmluZygxKTsKICAgICAgICAJCUphckVudHJ5IGVudHJ5ID0gamFyLmdldEphckVudHJ5KGVudHJ5bmFtZSk7CiAgICAgICAgCQlpcyA9amFyLmdldElucHV0U3RyZWFtKGVudHJ5KTsKICAgICAgICAJCUZpbGVVdGlsLmNvcHkoaXMsb3MpOwogICAgICAgIAkJcmV0dXJuIHRtcEZpbGU7CiAgICAgICAgIAl9CiAgICAgICAgCWNhdGNoIChJT0V4Y2VwdGlvbiBlKSB7CiAgICAgICAgCQlJU3RhdHVzIHMgPSBuZXcgU3RhdHVzKElTdGF0dXMuRVJST1IsQ29yZVBsdWdpbi5QTFVHSU5fSUQsMCwiZXJyb3IgY3JlYXRpbmcgdGVtcG9yYXJ5IGJ1aWxkIGZpbGUiLGUpOy8vJE5PTi1OTFMtMSQKICAgICAgICAgICAgICAgIENvcmVQbHVnaW4uZ2V0RGVmYXVsdCgpLmdldExvZygpLmxvZyhzKTsKCQkJCXRocm93IG5ldyBDb3JlRXhjZXB0aW9uKHMpOwoJCQl9CiAgICAgICAgCWZpbmFsbHl7CiAgICAgICAgCQl0cnkgewogICAgICAgIAkJCWlmKGlzIT1udWxsKQogICAgICAgIAkJCQlpcy5jbG9zZSgpOwoJCQkJCWlmKG9zIT1udWxsKQoJCQkJCQlvcy5jbG9zZSgpOwoJCQkJfSBjYXRjaCAoSU9FeGNlcHRpb24gZSkgewoJCQkJCS8vaWdub3JlCgkJCQl9CiAgICAgICAgCX0KICAgICAgICB9CiAgICAgICAgZWxzZXsKICAgICAgICAJcmV0dXJuIEZpbGVVdGlsLnJlc29sdmVGaWxlKGZpbGVVUkwpOwogICAgICAgIH0gCQogICAgfQogICAKICAgIC8qKgogICAgICogQHJldHVybgogICAgICovCiAgICBwcml2YXRlIFN0cmluZyBnZXRQdWJsaXNoVGFyZ2V0c0Zvck1vZHVsZSgpIHsKICAgIAlyZXR1cm4gZG9HZXRUYXJnZXRzKE1PRFVMRV9QVUJMSVNIX1RBUkdFVF9QUkVGSVgrZ2V0TW9kdWxlVHlwZUlkKCkpOwogICAgfQoKICAgIC8qKgogICAgICogQHJldHVybgogICAgICovCiAgICBwcml2YXRlIFN0cmluZyBnZXRVbnB1Ymxpc2hUYXJnZXRzRm9yTW9kdWxlKCkgewogICAgICAgIHJldHVybiBkb0dldFRhcmdldHMoTU9EVUxFX1VOUFVCTElTSF9UQVJHRVRfUFJFRklYK2dldE1vZHVsZVR5cGVJZCgpKTsKICAgIH0KICAgIAogICAgLyoqCiAgICAgKiBAcGFyYW0gZGF0YW5hbWUKICAgICAqIEByZXR1cm4KICAgICAqLwogICAgcHJpdmF0ZSBTdHJpbmcgZG9HZXRUYXJnZXRzKFN0cmluZyBkYXRhbmFtZSkgewogICAgCVN0cmluZ0J1ZmZlciBidWZmZXIgPSBuZXcgU3RyaW5nQnVmZmVyKCk7CiAgICAJSXRlcmF0b3IgaXRlcmF0b3IgPSBnZXRTZXJ2ZXJSdW50aW1lKCkuZ2V0U2VydmVyVHlwZURlZmluaXRpb24oKS5nZXRQdWJsaXNoZXIoUFVCTElTSEVSX0lEKS5nZXRQdWJsaXNoZXJkYXRhKCkuaXRlcmF0b3IoKTsKICAgICAgICB3aGlsZShpdGVyYXRvci5oYXNOZXh0KCkpewogICAgICAgICAgICBQdWJsaXNoZXJEYXRhIGRhdGEgPSAoUHVibGlzaGVyRGF0YSlpdGVyYXRvci5uZXh0KCk7CiAgICAgICAgICAgIGlmKGRhdGFuYW1lLmVxdWFscyhkYXRhLmdldERhdGFuYW1lKCkpKSB7CiAgICAgICAgICAgICAgICBpZihidWZmZXIubGVuZ3RoKCk+MCkKICAgICAgICAgICAgICAgIAlidWZmZXIuYXBwZW5kKCIsIik7Ly8kTk9OLU5MUy0xJAogICAgICAgICAgICAJYnVmZmVyLmFwcGVuZChkYXRhLmdldERhdGF2YWx1ZSgpKTsKICAgICAgICAgICAgfSAgIAogICAgICAgIH0KICAgICAgICByZXR1cm4gYnVmZmVyLnRvU3RyaW5nKCk7CiAgICB9CgogICAgcHJpdmF0ZSBTdHJpbmcgZ2V0TW9kdWxlVHlwZUlkKCl7CiAgICAgICAgcmV0dXJuIGdldE1vZHVsZSgpWzBdLmdldE1vZHVsZVR5cGUoKS5nZXRJZCgpOwogICAgfQogICAgCglwcml2YXRlIFN0cmluZyBnZXRCdWlsZEZpbGUoKQogICAgewogICAgICAgIEl0ZXJhdG9yIGl0ZXJhdG9yID0gZ2V0U2VydmVyUnVudGltZSgpLmdldFNlcnZlclR5cGVEZWZpbml0aW9uKCkuZ2V0UHVibGlzaGVyKFBVQkxJU0hFUl9JRCkuZ2V0UHVibGlzaGVyZGF0YSgpLml0ZXJhdG9yKCk7CiAgICAgICAgd2hpbGUoaXRlcmF0b3IuaGFzTmV4dCgpKQogICAgICAgIHsKICAgICAgICAgICAgUHVibGlzaGVyRGF0YSBkYXRhID0gKFB1Ymxpc2hlckRhdGEpaXRlcmF0b3IubmV4dCgpOwogICAgICAgICAgICBpZihEQVRBX05BTUVfQlVJTERfRklMRS5lcXVhbHMoZGF0YS5nZXREYXRhbmFtZSgpKSkKICAgICAgICAgICAgICAgIHJldHVybiBnZXRTZXJ2ZXJSdW50aW1lKCkuZ2V0U2VydmVyVHlwZURlZmluaXRpb24oKS5nZXRSZXNvbHZlcigpLnJlc29sdmVQcm9wZXJ0aWVzKGRhdGEuZ2V0RGF0YXZhbHVlKCkpOwogICAgICAgIH0KICAgICAgICByZXR1cm4gbnVsbDsKICAgIH0KCQoJcHJpdmF0ZSBNYXAgZ2V0UHVibGlzaFByb3BlcnRpZXMoKQoJewogICAgICAgIE1hcCBwcm9wcyA9IG5ldyBIYXNoTWFwKCk7CiAgICAgICAgLy8gcGFzcyBhbGwgcHJvcGVydGllcyB0byBidWlsZCBmaWxlLgogICAgICAgIE1hcCBzZXJ2ZXJQcm9wZXJ0aWVzID0gZ2V0U2VydmVyKCkuZ2V0U2VydmVySW5zdGFuY2VQcm9wZXJ0aWVzSW1wbCgpOwogICAgICAgIE1hcCBwcm9wZXJ0aWVzID0gZ2V0U2VydmVyUnVudGltZSgpLmdldFNlcnZlckluc3RhbmNlUHJvcGVydGllcygpOwogICAgICAgIHByb3BlcnRpZXMucHV0QWxsKHNlcnZlclByb3BlcnRpZXMpOwogICAgICAgIEl0ZXJhdG9yIHByb3BlcnR5SXRlcmF0b3IgPSBwcm9wZXJ0aWVzLmtleVNldCgpLml0ZXJhdG9yKCk7CiAgICAgICAgd2hpbGUocHJvcGVydHlJdGVyYXRvci5oYXNOZXh0KCkpCiAgICAgICAgewogICAgICAgICAgICBTdHJpbmcgcHJvcGVydHkgPSAoU3RyaW5nKXByb3BlcnR5SXRlcmF0b3IubmV4dCgpOwogICAgICAgICAgICBTdHJpbmcgdmFsdWUgPSAoU3RyaW5nKXByb3BlcnRpZXMuZ2V0KHByb3BlcnR5KTsKICAgICAgICAgICAgaWYodmFsdWUhPW51bGwgJiYgdmFsdWUudHJpbSgpLmxlbmd0aCgpPjApCiAgICAgICAgICAgIAlwcm9wcy5wdXQocHJvcGVydHkscHJvcGVydGllcy5nZXQocHJvcGVydHkpKTsKICAgICAgICB9CiAgICAgICAgTW9kdWxlIG1vZHVsZSA9ICBnZXRTZXJ2ZXJSdW50aW1lKCkuZ2V0U2VydmVyVHlwZURlZmluaXRpb24oKS5nZXRNb2R1bGUoZ2V0TW9kdWxlVHlwZUlkKCkpOwoJCVN0cmluZyBtb2REaXIgPSBtb2R1bGUuZ2V0UHVibGlzaERpcigpOwoJCW1vZERpciA9IGdldFNlcnZlclJ1bnRpbWUoKS5nZXRTZXJ2ZXJUeXBlRGVmaW5pdGlvbigpLmdldFJlc29sdmVyKCkucmVzb2x2ZVByb3BlcnRpZXMobW9kRGlyKTsKCgkJSVdlYk1vZHVsZSB3ZWJNb2R1bGUgPSAoSVdlYk1vZHVsZSlnZXRNb2R1bGUoKVswXS5sb2FkQWRhcHRlcihJV2ViTW9kdWxlLmNsYXNzLG51bGwpOwogICAgICAgIElFSkJNb2R1bGUgZWpiTW9kdWxlID0gKElFSkJNb2R1bGUpZ2V0TW9kdWxlKClbMF0ubG9hZEFkYXB0ZXIoSUVKQk1vZHVsZS5jbGFzcyxudWxsKTsKCQlJRW50ZXJwcmlzZUFwcGxpY2F0aW9uIGVhck1vZHVsZSA9IChJRW50ZXJwcmlzZUFwcGxpY2F0aW9uKWdldE1vZHVsZSgpWzBdLmxvYWRBZGFwdGVyKElFbnRlcnByaXNlQXBwbGljYXRpb24uY2xhc3MsbnVsbCk7CiAgICAgICAgCiAgICAgICAgU3RyaW5nIG1vZHVsZU5hbWU9InVua25vd25tb2R1bGUiOy8vJE5PTi1OTFMtMSQKICAgICAgICBTdHJpbmcgbW9kdWxlRGlyPSIiOy8vJE5PTi1OTFMtMSQKICAgICAgICBpZih3ZWJNb2R1bGUhPW51bGwpeyAgICAKICAgICAgICAgICAgbW9kdWxlTmFtZSA9IHRoaXMuZ3Vlc3NNb2R1bGVOYW1lKHdlYk1vZHVsZSk7CiAgICAgICAgICAgIG1vZHVsZURpciA9IHdlYk1vZHVsZS5nZXRMb2NhdGlvbigpLnRvU3RyaW5nKCk7CiAgICAgICAgfQogICAgICAgIGlmKGVqYk1vZHVsZSE9bnVsbCl7ICAKICAgICAgICAgICAgbW9kdWxlTmFtZSA9IGdldE1vZHVsZSgpWzBdLmdldE5hbWUoKTsKICAgICAgICAgICAgbW9kdWxlRGlyPSBlamJNb2R1bGUuZ2V0TG9jYXRpb24oKS50b1N0cmluZygpOwogICAgICAgIH0KICAgICAgICBpZihlYXJNb2R1bGUhPW51bGwpCiAgICAgICAgewogICAgICAgIAltb2R1bGVOYW1lID0gZ2V0TW9kdWxlKClbMF0uZ2V0TmFtZSgpOwogICAgICAgIAltb2R1bGVEaXIgPSBlYXJNb2R1bGUuZ2V0TG9jYXRpb24oKS50b1N0cmluZygpOwogICAgICAgIH0KICAgICAgICBTdHJpbmcgcGx1Z2luSWQgPSBnZXRTZXJ2ZXJSdW50aW1lKCkuZ2V0U2VydmVyVHlwZURlZmluaXRpb24oKS5nZXRDb25maWd1cmF0aW9uRWxlbWVudE5hbWVzcGFjZSgpOwogICAgICAgIHByb3BzLnB1dChQUk9QX1BST0pFQ1RfV09SS0lOR19ESVIsZ2V0TW9kdWxlKClbMF0uZ2V0UHJvamVjdCgpLmdldFdvcmtpbmdMb2NhdGlvbihwbHVnaW5JZCkudG9TdHJpbmcoKSk7CgkJcHJvcHMucHV0KFBST1BfTU9EVUxFX05BTUUsbW9kdWxlTmFtZSk7CgkJcHJvcHMucHV0KFBST1BfTU9EVUxFX0RJUixtb2R1bGVEaXIpOwoJCXByb3BzLnB1dChQUk9QX1NFUlZFUl9QVUJMSVNIX0RJUixtb2REaXIpOwoJCXJldHVybiBwcm9wczsKCX0KCglwcml2YXRlIFN0cmluZyBndWVzc01vZHVsZU5hbWUoSVdlYk1vZHVsZSB3ZWJNb2R1bGUpIHsKCQlTdHJpbmcgbW9kdWxlTmFtZSA9IGdldE1vZHVsZSgpWzBdLmdldE5hbWUoKTsgCgkJU3RyaW5nIGNvbnRleHRSb290ID0gd2ViTW9kdWxlLmdldENvbnRleHRSb290KCk7CgkJaWYoY29udGV4dFJvb3QuY2hhckF0KDApID09ICcvJykKCQkJbW9kdWxlTmFtZSA9IGNvbnRleHRSb290LnN1YnN0cmluZygxKTsKCQlyZXR1cm4gbW9kdWxlTmFtZTsKCX0KCglwcml2YXRlIHZvaWQgcnVuQW50KFN0cmluZyBidWlsZEZpbGUsU3RyaW5nIHRhcmdldHMsTWFwIHByb3BlcnRpZXMgLElQcm9ncmVzc01vbml0b3IgbW9uaXRvcil0aHJvd3MgQ29yZUV4Y2VwdGlvbnsKCQlJTGF1bmNoTWFuYWdlciBsYXVuY2hNYW5hZ2VyID0gRGVidWdQbHVnaW4uZ2V0RGVmYXVsdCgpLmdldExhdW5jaE1hbmFnZXIoKTsKCQlJTGF1bmNoQ29uZmlndXJhdGlvblR5cGUgdHlwZSA9IGxhdW5jaE1hbmFnZXIuZ2V0TGF1bmNoQ29uZmlndXJhdGlvblR5cGUoSUFudExhdW5jaENvbmZpZ3VyYXRpb25Db25zdGFudHMuSURfQU5UX0xBVU5DSF9DT05GSUdVUkFUSU9OX1RZUEUpOwoKCQlJTGF1bmNoQ29uZmlndXJhdGlvbldvcmtpbmdDb3B5IHdjPSB0eXBlLm5ld0luc3RhbmNlKG51bGwscHJvcGVydGllcy5nZXQoUFJPUF9NT0RVTEVfTkFNRSkrIiBtb2R1bGUgcHVibGlzaGVyIik7CgkJd2Muc2V0Q29udGFpbmVyKG51bGwpOwoJCXdjLnNldEF0dHJpYnV0ZShJRXh0ZXJuYWxUb29sQ29uc3RhbnRzLkFUVFJfTE9DQVRJT04sIGJ1aWxkRmlsZSk7CgkJd2Muc2V0QXR0cmlidXRlKElKYXZhTGF1bmNoQ29uZmlndXJhdGlvbkNvbnN0YW50cy5BVFRSX0NMQVNTUEFUSF9QUk9WSURFUiwib3JnLmVjbGlwc2UuYW50LnVpLkFudENsYXNzcGF0aFByb3ZpZGVyIik7CgkJd2Muc2V0QXR0cmlidXRlKElBbnRMYXVuY2hDb25maWd1cmF0aW9uQ29uc3RhbnRzLkFUVFJfQU5UX1RBUkdFVFMsdGFyZ2V0cyk7CgkJd2Muc2V0QXR0cmlidXRlKElBbnRMYXVuY2hDb25maWd1cmF0aW9uQ29uc3RhbnRzLkFUVFJfQU5UX1BST1BFUlRJRVMscHJvcGVydGllcyk7CgkJd2Muc2V0QXR0cmlidXRlKElEZWJ1Z1VJQ29uc3RhbnRzLkFUVFJfTEFVTkNIX0lOX0JBQ0tHUk9VTkQsZmFsc2UpOwoJCXdjLnNldEF0dHJpYnV0ZShJRGVidWdVSUNvbnN0YW50cy5BVFRSX0NBUFRVUkVfSU5fQ09OU09MRSx0cnVlKTsKCQl3Yy5zZXRBdHRyaWJ1dGUoSURlYnVnVUlDb25zdGFudHMuQVRUUl9QUklWQVRFLHRydWUpOwoJCQoJCXdjLnNldEF0dHJpYnV0ZShJSmF2YUxhdW5jaENvbmZpZ3VyYXRpb25Db25zdGFudHMuQVRUUl9TT1VSQ0VfUEFUSF9QUk9WSURFUiwgIm9yZy5lY2xpcHNlLmFudC51aS5BbnRDbGFzc3BhdGhQcm92aWRlciIpOyAKCQl3Yy5zZXRBdHRyaWJ1dGUoSUphdmFMYXVuY2hDb25maWd1cmF0aW9uQ29uc3RhbnRzLkFUVFJfVk1fSU5TVEFMTF9OQU1FLGdldFNlcnZlclJ1bnRpbWUoKS5nZXRWTUluc3RhbGwoKS5nZXROYW1lKCkpOwoJCXdjLnNldEF0dHJpYnV0ZShJSmF2YUxhdW5jaENvbmZpZ3VyYXRpb25Db25zdGFudHMuQVRUUl9WTV9JTlNUQUxMX1RZUEUsZ2V0U2VydmVyUnVudGltZSgpLmdldFZNSW5zdGFsbCgpLmdldFZNSW5zdGFsbFR5cGUoKS5nZXRJZCgpKTsKCQl3Yy5zZXRBdHRyaWJ1dGUoSUphdmFMYXVuY2hDb25maWd1cmF0aW9uQ29uc3RhbnRzLkFUVFJfTUFJTl9UWVBFX05BTUUsICJvcmcuZWNsaXBzZS5hbnQuaW50ZXJuYWwudWkuYW50c3VwcG9ydC5JbnRlcm5hbEFudFJ1bm5lciIpOwoJCXdjLnNldEF0dHJpYnV0ZShEZWJ1Z1BsdWdpbi5BVFRSX1BST0NFU1NfRkFDVE9SWV9JRCwgSUFudFVJQ29uc3RhbnRzLlJFTU9URV9BTlRfUFJPQ0VTU19GQUNUT1JZX0lEKTsKCQkKCQlzZXR1cEFudExhdW5jaENvbmZpZ3VyYXRpb24od2MpOwoJCQoJCQoJCUlMYXVuY2hDb25maWd1cmF0aW9uIGxhdW5jaENvbmZpZyA9IHdjLmRvU2F2ZSgpOwogICAgICAgIGxhdW5jaENvbmZpZy5sYXVuY2goInJ1biIsbW9uaXRvcik7Cgl9CgoKCiAgICAvKiAobm9uLUphdmFkb2MpCiAgICAgKi8KICAgIHByb3RlY3RlZCB2b2lkIHNldHVwQW50TGF1bmNoQ29uZmlndXJhdGlvbihJTGF1bmNoQ29uZmlndXJhdGlvbldvcmtpbmdDb3B5IHdjKSB7CQkKCX0KCgkvKiAobm9uLUphdmFkb2MpCiAgICAgKiBAc2VlIG9yZy5lY2xpcHNlLmpzdC5zZXJ2ZXIuZ2VuZXJpYy5pbnRlcm5hbC5jb3JlLkdlbmVyaWNQdWJsaXNoZXIjdW5wdWJsaXNoKG9yZy5lY2xpcHNlLndzdC5zZXJ2ZXIuY29yZS5JTW9kdWxlLCBvcmcuZWNsaXBzZS5jb3JlLnJ1bnRpbWUuSVByb2dyZXNzTW9uaXRvcikKICAgICAqLwogICAgcHVibGljIElTdGF0dXNbXSB1bnB1Ymxpc2goSVByb2dyZXNzTW9uaXRvciBtb25pdG9yKSB7CgogICAgCWlmKGdldE1vZHVsZSgpLmxlbmd0aD4xKS8vIG9ubHkgcmVzcG9uZCB0byByb290IG1vZHVsZSBjYWxscy4gCgkJCXJldHVybiBudWxsOwogICAgCXRyeSB7CiAgICAgICAgCSBGaWxlIGZpbGUgPSBjb21wdXRlQnVpbGRGaWxlKCk7CiAgICAgICAgICAgIHJ1bkFudChmaWxlLnRvU3RyaW5nKCksZ2V0VW5wdWJsaXNoVGFyZ2V0c0Zvck1vZHVsZSgpLGdldFB1Ymxpc2hQcm9wZXJ0aWVzKCksbW9uaXRvcik7CiAgICAgICAgfSBjYXRjaCAoQ29yZUV4Y2VwdGlvbiBlKSB7CiAgICAgICAgICAgIElTdGF0dXMgcyA9IG5ldyBTdGF0dXMoSVN0YXR1cy5FUlJPUixDb3JlUGx1Z2luLlBMVUdJTl9JRCwwLEdlbmVyaWNTZXJ2ZXJDb3JlTWVzc2FnZXMuZXJyb3JSZW1vdmVNb2R1bGVBbnRwdWJsaXNoZXIsZSk7CiAgICAgICAgICAgIHJldHVybiBuZXcgSVN0YXR1c1tdIHtzfTsKICAgICAgICB9CiAgICAgICAgcmV0dXJuIG51bGw7CiAgICB9Cn0K