LyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogQ29weXJpZ2h0IChjKSAyMDAzLCAyMDA0IElCTSBDb3Jwb3JhdGlvbiBhbmQgb3RoZXJzLgogKiBBbGwgcmlnaHRzIHJlc2VydmVkLiBUaGlzIHByb2dyYW0gYW5kIHRoZSBhY2NvbXBhbnlpbmcgbWF0ZXJpYWxzCiAqIGFyZSBtYWRlIGF2YWlsYWJsZSB1bmRlciB0aGUgdGVybXMgb2YgdGhlIENvbW1vbiBQdWJsaWMgTGljZW5zZSB2MS4wCiAqIHdoaWNoIGFjY29tcGFuaWVzIHRoaXMgZGlzdHJpYnV0aW9uLCBhbmQgaXMgYXZhaWxhYmxlIGF0CiAqIGh0dHA6Ly93d3cuZWNsaXBzZS5vcmcvbGVnYWwvY3BsLXYxMC5odG1sCqAqCiAqIENvbnRyaWJ1dG9yczoKICogICAgSUJNIC0gSW5pdGlhbCBBUEkgYW5kIGltcGxlbWVudGF0aW9uCiAqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwpwYWNrYWdlIG9yZy5lY2xpcHNlLndzdC5zZXJ2ZXIudWkuaW50ZXJuYWw7CgppbXBvcnQgb3JnLmVjbGlwc2UuamZhY2UuZGlhbG9ncy5EaWFsb2c7CmltcG9ydCBvcmcuZWNsaXBzZS5qZmFjZS5wcmVmZXJlbmNlLlByZWZlcmVuY2VQYWdlOwppbXBvcnQgb3JnLmVjbGlwc2Uud3N0LnNlcnZlci5jb3JlLklTZXJ2ZXI7CmltcG9ydCBvcmcuZWNsaXBzZS53c3Quc2VydmVyLmNvcmUuSVNlcnZlclByZWZlcmVuY2VzOwppbXBvcnQgb3JnLmVjbGlwc2Uud3N0LnNlcnZlci5jb3JlLlNlcnZlckNvcmU7CmltcG9ydCBvcmcuZWNsaXBzZS53c3Quc2VydmVyLnVpLklTZXJ2ZXJVSVByZWZlcmVuY2VzOwppbXBvcnQgb3JnLmVjbGlwc2Uud3N0LnNlcnZlci51aS5TZXJ2ZXJVSUNvcmU7CmltcG9ydCBvcmcuZWNsaXBzZS5zd3QuU1dUOwppbXBvcnQgb3JnLmVjbGlwc2Uuc3d0LmV2ZW50cy5TZWxlY3Rpb25BZGFwdGVyOwppbXBvcnQgb3JnLmVjbGlwc2Uuc3d0LmV2ZW50cy5TZWxlY3Rpb25FdmVudDsKaW1wb3J0IG9yZy5lY2xpcHNlLnN3dC5sYXlvdXQuR3JpZERhdGE7CmltcG9ydCBvcmcuZWNsaXBzZS5zd3QubGF5b3V0LkdyaWRMYXlvdXQ7CmltcG9ydCBvcmcuZWNsaXBzZS5zd3Qud2lkZ2V0cy5CdXR0b247CmltcG9ydCBvcmcuZWNsaXBzZS5zd3Qud2lkZ2V0cy5Db21wb3NpdGU7CmltcG9ydCBvcmcuZWNsaXBzZS5zd3Qud2lkZ2V0cy5Db250cm9sOwppbXBvcnQgb3JnLmVjbGlwc2Uuc3d0LndpZGdldHMuR3JvdXA7CmltcG9ydCBvcmcuZWNsaXBzZS51aS5JV29ya2JlbmNoOwppbXBvcnQgb3JnLmVjbGlwc2UudWkuSVdvcmtiZW5jaFByZWZlcmVuY2VQYWdlOwppbXBvcnQgb3JnLmVjbGlwc2UudWkuaGVscC5Xb3JrYmVuY2hIZWxwOwovKioKICogVGhlIHByZWZlcmVuY2UgcGFnZSB0aGF0IGhvbGRzIHNlcnZlciBwcm9wZXJ0aWVzLgogKi8KcHVibGljIGNsYXNzIFNlcnZlclByZWZlcmVuY2VQYWdlIGV4dGVuZHMgUHJlZmVyZW5jZVBhZ2UgaW1wbGVtZW50cyBJV29ya2JlbmNoUHJlZmVyZW5jZVBhZ2UgewoJcHJvdGVjdGVkIEJ1dHRvbiBwdWJsaXNoQmVmb3JlU3RhcnQ7Cglwcm90ZWN0ZWQgQnV0dG9uIGF1dG9SZXN0YXJ0OwoKCXByb3RlY3RlZCBCdXR0b24gcHJvbXB0SXJyZXZlcnNpYmxlOwoJCglwcm90ZWN0ZWQgYnl0ZSBzYXZlRWRpdG9yczsKCQoJcHJvdGVjdGVkIEJ1dHRvbiBzYXZlTmV2ZXI7Cglwcm90ZWN0ZWQgQnV0dG9uIHNhdmVQcm9tcHQ7Cglwcm90ZWN0ZWQgQnV0dG9uIHNhdmVBdXRvOwoJCglwcm90ZWN0ZWQgQnV0dG9uIGNyZWF0ZUluV29ya3NwYWNlOwoKCXByb3RlY3RlZCBJU2VydmVyUHJlZmVyZW5jZXMgcHJlZmVyZW5jZXM7Cglwcm90ZWN0ZWQgSVNlcnZlclVJUHJlZmVyZW5jZXMgdWlQcmVmZXJlbmNlczsKCgkvKioKCSAqIFNlcnZlclByZWZlcmVuY2VzUGFnZSBjb25zdHJ1Y3RvciBjb21tZW50LgoJICovCglwdWJsaWMgU2VydmVyUHJlZmVyZW5jZVBhZ2UoKSB7CgkJc3VwZXIoKTsKCQoJCXByZWZlcmVuY2VzID0gU2VydmVyQ29yZS5nZXRTZXJ2ZXJQcmVmZXJlbmNlcygpOwoJCXVpUHJlZmVyZW5jZXMgPSBTZXJ2ZXJVSUNvcmUuZ2V0UHJlZmVyZW5jZXMoKTsKCX0KCQoJLyoqCgkgKiBDcmVhdGUgdGhlIHByZWZlcmVuY2Ugb3B0aW9ucy4KCSAqCgkgKiBAcGFyYW0gcGFyZW50IG9yZy5lY2xpcHNlLnN3dC53aWRnZXRzLkNvbXBvc2l0ZQoJICogQHJldHVybiBvcmcuZWNsaXBzZS5zd3Qud2lkZ2V0cy5Db250cm9sCgkgKi8KCXByb3RlY3RlZCBDb250cm9sIGNyZWF0ZUNvbnRlbnRzKENvbXBvc2l0ZSBwYXJlbnQpIHsKCQlpbml0aWFsaXplRGlhbG9nVW5pdHMocGFyZW50KTsKCQkKCQlDb21wb3NpdGUgY29tcG9zaXRlID0gbmV3IENvbXBvc2l0ZShwYXJlbnQsIFNXVC5OT05FKTsKCQlHcmlkTGF5b3V0IGxheW91dCA9IG5ldyBHcmlkTGF5b3V0KCk7CgkJbGF5b3V0Lmhvcml6b250YWxTcGFjaW5nID0gY29udmVydEhvcml6b250YWxETFVzVG9QaXhlbHMoNCk7CgkJbGF5b3V0LnZlcnRpY2FsU3BhY2luZyA9IGNvbnZlcnRWZXJ0aWNhbERMVXNUb1BpeGVscyg0KTsKCQlsYXlvdXQubWFyZ2luV2lkdGggPSAwOwoJCWxheW91dC5tYXJnaW5IZWlnaHQgPSAwOwoJCWxheW91dC5udW1Db2x1bW5zID0gMzsKCQljb21wb3NpdGUuc2V0TGF5b3V0KGxheW91dCk7CgkJR3JpZERhdGEgZGF0YSA9IG5ldyBHcmlkRGF0YShHcmlkRGF0YS5GSUxMX0hPUklaT05UQUwgfCBHcmlkRGF0YS5WRVJUSUNBTF9BTElHTl9GSUxMKTsKCQljb21wb3NpdGUuc2V0TGF5b3V0RGF0YShkYXRhKTsKCQlXb3JrYmVuY2hIZWxwLnNldEhlbHAoY29tcG9zaXRlLCBDb250ZXh0SWRzLlBSRUZfR0VORVJBTCk7CgkJCgkJcHVibGlzaEJlZm9yZVN0YXJ0ID0gbmV3IEJ1dHRvbihjb21wb3NpdGUsIFNXVC5DSEVDSyk7CgkJcHVibGlzaEJlZm9yZVN0YXJ0LnNldFRleHQoU2VydmVyVUlQbHVnaW4uZ2V0UmVzb3VyY2UoIiVwcmVmUHVibGlzaEJlZm9yZVN0YXJ0aW5nIikpOwoJCWRhdGEgPSBuZXcgR3JpZERhdGEoR3JpZERhdGEuSE9SSVpPTlRBTF9BTElHTl9GSUxMKTsKCQlkYXRhLmhvcml6b250YWxTcGFuID0gMzsKCQlwdWJsaXNoQmVmb3JlU3RhcnQuc2V0TGF5b3V0RGF0YShkYXRhKTsKCQlwdWJsaXNoQmVmb3JlU3RhcnQuc2V0U2VsZWN0aW9uKHByZWZlcmVuY2VzLmlzQXV0b1B1Ymxpc2hpbmcoKSk7CgkJV29ya2JlbmNoSGVscC5zZXRIZWxwKHB1Ymxpc2hCZWZvcmVTdGFydCwgQ29udGV4dElkcy5QUkVGX0dFTkVSQUxfUFVCTElTSF9CRUZPUkVfU1RBUlQpOwoJCQoJCWF1dG9SZXN0YXJ0ID0gbmV3IEJ1dHRvbihjb21wb3NpdGUsIFNXVC5DSEVDSyk7CgkJYXV0b1Jlc3RhcnQuc2V0VGV4dChTZXJ2ZXJVSVBsdWdpbi5nZXRSZXNvdXJjZSgiJXByZWZBdXRvUmVzdGFydCIpKTsKCQlkYXRhID0gbmV3IEdyaWREYXRhKEdyaWREYXRhLkhPUklaT05UQUxfQUxJR05fRklMTCk7CgkJZGF0YS5ob3Jpem9udGFsU3BhbiA9IDM7CgkJYXV0b1Jlc3RhcnQuc2V0TGF5b3V0RGF0YShkYXRhKTsKCQlhdXRvUmVzdGFydC5zZXRTZWxlY3Rpb24ocHJlZmVyZW5jZXMuaXNBdXRvUmVzdGFydGluZygpKTsKCQlXb3JrYmVuY2hIZWxwLnNldEhlbHAoYXV0b1Jlc3RhcnQsIENvbnRleHRJZHMuUFJFRl9HRU5FUkFMX0FVVE9fUkVTVEFSVCk7CgkJCgkJcHJvbXB0SXJyZXZlcnNpYmxlID0gbmV3IEJ1dHRvbihjb21wb3NpdGUsIFNXVC5DSEVDSyk7CgkJcHJvbXB0SXJyZXZlcnNpYmxlLnNldFRleHQoU2VydmVyVUlQbHVnaW4uZ2V0UmVzb3VyY2UoIiVwcmVmUHJvbXB0SXJyZXZlcnNpYmxlIikpOwoJCWRhdGEgPSBuZXcgR3JpZERhdGEoR3JpZERhdGEuSE9SSVpPTlRBTF9BTElHTl9GSUxMKTsKCQlkYXRhLmhvcml6b250YWxTcGFuID0gMzsKCQlwcm9tcHRJcnJldmVyc2libGUuc2V0TGF5b3V0RGF0YShkYXRhKTsKCQlwcm9tcHRJcnJldmVyc2libGUuc2V0U2VsZWN0aW9uKHVpUHJlZmVyZW5jZXMuZ2V0UHJvbXB0QmVmb3JlSXJyZXZlcnNpYmxlQ2hhbmdlKCkpOwoJCVdvcmtiZW5jaEhlbHAuc2V0SGVscChwcm9tcHRJcnJldmVyc2libGUsIENvbnRleHRJZHMuUFJFRl9HRU5FUkFMX1BST01QVF9JUlJFVkVSU0lCTEUpOwoJCQoJCWNyZWF0ZUluV29ya3NwYWNlID0gbmV3IEJ1dHRvbihjb21wb3NpdGUsIFNXVC5DSEVDSyk7CgkJY3JlYXRlSW5Xb3Jrc3BhY2Uuc2V0VGV4dChTZXJ2ZXJVSVBsdWdpbi5nZXRSZXNvdXJjZSgiJXByZWZDcmVhdGVJbldvcmtzcGFjZSIpKTsKCQlkYXRhID0gbmV3IEdyaWREYXRhKEdyaWREYXRhLkhPUklaT05UQUxfQUxJR05fRklMTCk7CgkJZGF0YS5ob3Jpem9udGFsU3BhbiA9IDM7CgkJY3JlYXRlSW5Xb3Jrc3BhY2Uuc2V0TGF5b3V0RGF0YShkYXRhKTsKCQljcmVhdGVJbldvcmtzcGFjZS5zZXRTZWxlY3Rpb24ocHJlZmVyZW5jZXMuaXNDcmVhdGVSZXNvdXJjZXNJbldvcmtzcGFjZSgpKTsKCQlXb3JrYmVuY2hIZWxwLnNldEhlbHAoY3JlYXRlSW5Xb3Jrc3BhY2UsIENvbnRleHRJZHMuUFJFRl9HRU5FUkFMX0NSRUFURV9JTl9XT1JLU1BBQ0UpOwoJCQoJCS8vIHNhdmUgZWRpdG9ycyBncm91cAoJCUdyb3VwIHNhdmVFZGl0b3JHcm91cCA9IG5ldyBHcm91cChjb21wb3NpdGUsIFNXVC5OT05FKTsKCQlzYXZlRWRpdG9yR3JvdXAuc2V0VGV4dChTZXJ2ZXJVSVBsdWdpbi5nZXRSZXNvdXJjZSgiJXByZWZTYXZlRWRpdG9yc0dyb3VwIikpOwoJCQoJCWxheW91dCA9IG5ldyBHcmlkTGF5b3V0KCk7CgkJbGF5b3V0Lm51bUNvbHVtbnMgPSAzOwoJCXNhdmVFZGl0b3JHcm91cC5zZXRMYXlvdXQobGF5b3V0KTsKCQlkYXRhID0gbmV3IEdyaWREYXRhKEdyaWREYXRhLkZJTExfSE9SSVpPTlRBTCk7CgkJZGF0YS5ob3Jpem9udGFsU3BhbiA9IDM7CgkJc2F2ZUVkaXRvckdyb3VwLnNldExheW91dERhdGEoZGF0YSk7CgkJCgkJc2F2ZU5ldmVyID0gbmV3IEJ1dHRvbihzYXZlRWRpdG9yR3JvdXAsIFNXVC5SQURJTyk7CgkJc2F2ZU5ldmVyLnNldFRleHQoU2VydmVyVUlQbHVnaW4uZ2V0UmVzb3VyY2UoIiVwcmVmU2F2ZUVkaXRvcnNOZXZlciIpKTsKCQlzYXZlTmV2ZXIuYWRkU2VsZWN0aW9uTGlzdGVuZXIobmV3IFNlbGVjdGlvbkFkYXB0ZXIoKSB7CgkJCXB1YmxpYyB2b2lkIHdpZGdldFNlbGVjdGVkKFNlbGVjdGlvbkV2ZW50IGV2ZW50KSB7CgkJCQlzYXZlRWRpdG9ycyA9IElTZXJ2ZXJVSVByZWZlcmVuY2VzLlNBVkVfRURJVE9SU19ORVZFUjsKCQkJfQoJCX0pOwoJCVdvcmtiZW5jaEhlbHAuc2V0SGVscChzYXZlTmV2ZXIsIENvbnRleHRJZHMuUFJFRl9HRU5FUkFMX1NBVkVfRURJVE9SUyk7CgkJCgkJc2F2ZVByb21wdCA9IG5ldyBCdXR0b24oc2F2ZUVkaXRvckdyb3VwLCBTV1QuUkFESU8pOwoJCXNhdmVQcm9tcHQuc2V0VGV4dChTZXJ2ZXJVSVBsdWdpbi5nZXRSZXNvdXJjZSgiJXByZWZTYXZlRWRpdG9yc1Byb21wdCIpKTsKCQlzYXZlUHJvbXB0LmFkZFNlbGVjdGlvbkxpc3RlbmVyKG5ldyBTZWxlY3Rpb25BZGFwdGVyKCkgewoJCQlwdWJsaWMgdm9pZCB3aWRnZXRTZWxlY3RlZChTZWxlY3Rpb25FdmVudCBldmVudCkgewoJCQkJc2F2ZUVkaXRvcnMgPSBJU2VydmVyVUlQcmVmZXJlbmNlcy5TQVZFX0VESVRPUlNfUFJPTVBUOwoJCQl9CgkJfSk7CgkJV29ya2JlbmNoSGVscC5zZXRIZWxwKHNhdmVQcm9tcHQsIENvbnRleHRJZHMuUFJFRl9HRU5FUkFMX1NBVkVfRURJVE9SUyk7CgkJCgkJc2F2ZUF1dG8gPSBuZXcgQnV0dG9uKHNhdmVFZGl0b3JHcm91cCwgU1dULlJBRElPKTsKCQlzYXZlQXV0by5zZXRUZXh0KFNlcnZlclVJUGx1Z2luLmdldFJlc291cmNlKCIlcHJlZlNhdmVFZGl0b3JzQXV0b3NhdmUiKSk7CgkJc2F2ZUF1dG8uYWRkU2VsZWN0aW9uTGlzdGVuZXIobmV3IFNlbGVjdGlvbkFkYXB0ZXIoKSB7CgkJCXB1YmxpYyB2b2lkIHdpZGdldFNlbGVjdGVkKFNlbGVjdGlvbkV2ZW50IGV2ZW50KSB7CgkJCQlzYXZlRWRpdG9ycyA9IElTZXJ2ZXJVSVByZWZlcmVuY2VzLlNBVkVfRURJVE9SU19BVVRPOwoJCQl9CgkJfSk7CgkJV29ya2JlbmNoSGVscC5zZXRIZWxwKHNhdmVBdXRvLCBDb250ZXh0SWRzLlBSRUZfR0VORVJBTF9TQVZFX0VESVRPUlMpOwoJCQoJCXNldFNhdmVFZGl0b3JTdGF0dXModWlQcmVmZXJlbmNlcy5nZXRTYXZlRWRpdG9ycygpKTsKCQkKCQkKCQlEaWFsb2cuYXBwbHlEaWFsb2dGb250KGNvbXBvc2l0ZSk7CgkKCQlyZXR1cm4gY29tcG9zaXRlOwoJfQoJCglwcm90ZWN0ZWQgdm9pZCBzZXRTYXZlRWRpdG9yU3RhdHVzKGJ5dGUgc3RhdHVzKSB7CgkJc2F2ZUVkaXRvcnMgPSBzdGF0dXM7CgkJc2F2ZU5ldmVyLnNldFNlbGVjdGlvbihzYXZlRWRpdG9ycyA9PSBJU2VydmVyVUlQcmVmZXJlbmNlcy5TQVZFX0VESVRPUlNfTkVWRVIpOwoJCXNhdmVQcm9tcHQuc2V0U2VsZWN0aW9uKHNhdmVFZGl0b3JzID09IElTZXJ2ZXJVSVByZWZlcmVuY2VzLlNBVkVfRURJVE9SU19QUk9NUFQpOwoJCXNhdmVBdXRvLnNldFNlbGVjdGlvbihzYXZlRWRpdG9ycyA9PSBJU2VydmVyVUlQcmVmZXJlbmNlcy5TQVZFX0VESVRPUlNfQVVUTyk7IAoJfQoJCgkvKioKCSAqIEluaXRpYWxpemVzIHRoaXMgcHJlZmVyZW5jZSBwYWdlIHVzaW5nIHRoZSBwYXNzZWQgZGVza3RvcC4KCSAqCgkgKiBAcGFyYW0gZGVza3RvcCB0aGUgY3VycmVudCBkZXNrdG9wCgkgKi8KCXB1YmxpYyB2b2lkIGluaXQoSVdvcmtiZW5jaCB3b3JrYmVuY2gpIHsKCQkvLyBkbyBub3RoaW5nCgl9CgkKCS8qKgoJICogUGVyZm9ybXMgc3BlY2lhbCBwcm9jZXNzaW5nIHdoZW4gdGhpcyBwYWdlJ3MgRGVmYXVsdHMgYnV0dG9uIGhhcyBiZWVuIHByZXNzZWQuCgkgKiA8cD4KCSAqIFRoaXMgaXMgYSBmcmFtZXdvcmsgaG9vayBtZXRob2QgZm9yIHN1YmxjYXNzZXMgdG8gZG8gc3BlY2lhbCB0aGluZ3Mgd2hlbgoJICogdGhlIERlZmF1bHRzIGJ1dHRvbiBoYXMgYmVlbiBwcmVzc2VkLgoJICogU3ViY2xhc3NlcyBtYXkgb3ZlcnJpZGUsIGJ1dCBzaG91bGQgY2FsbCA8Y29kZT5zdXBlci5wZXJmb3JtRGVmYXVsdHM8L2NvZGU+LgoJICogPC9wPgoJICovCglwcm90ZWN0ZWQgdm9pZCBwZXJmb3JtRGVmYXVsdHMoKSB7CgkJYXV0b1Jlc3RhcnQuc2V0U2VsZWN0aW9uKHByZWZlcmVuY2VzLmlzRGVmYXVsdEF1dG9SZXN0YXJ0aW5nKCkpOwoJCXB1Ymxpc2hCZWZvcmVTdGFydC5zZXRTZWxlY3Rpb24ocHJlZmVyZW5jZXMuaXNEZWZhdWx0QXV0b1B1Ymxpc2hpbmcoKSk7CgkJcHJvbXB0SXJyZXZlcnNpYmxlLnNldFNlbGVjdGlvbih1aVByZWZlcmVuY2VzLmdldERlZmF1bHRQcm9tcHRCZWZvcmVJcnJldmVyc2libGVDaGFuZ2UoKSk7CgkJY3JlYXRlSW5Xb3Jrc3BhY2Uuc2V0U2VsZWN0aW9uKHByZWZlcmVuY2VzLmlzRGVmYXVsdENyZWF0ZVJlc291cmNlc0luV29ya3NwYWNlKCkpOwoJCQoJCXNldFNhdmVFZGl0b3JTdGF0dXModWlQcmVmZXJlbmNlcy5nZXREZWZhdWx0U2F2ZUVkaXRvcnMoKSk7CgkKCQlzdXBlci5wZXJmb3JtRGVmYXVsdHMoKTsKCX0KCgkvKioKCSAqIE1ldGhvZCBkZWNsYXJlZCBvbiBJUHJlZmVyZW5jZVBhZ2UuCgkgKiBTdWJjbGFzc2VzIHNob3VsZCBvdmVycmlkZQoJICovCglwdWJsaWMgYm9vbGVhbiBwZXJmb3JtT2soKSB7CgkJcHJlZmVyZW5jZXMuc2V0QXV0b1B1Ymxpc2hpbmcocHVibGlzaEJlZm9yZVN0YXJ0LmdldFNlbGVjdGlvbigpKTsKCQlwcmVmZXJlbmNlcy5zZXRBdXRvUmVzdGFydGluZyhhdXRvUmVzdGFydC5nZXRTZWxlY3Rpb24oKSk7CgkJdWlQcmVmZXJlbmNlcy5zZXRTYXZlRWRpdG9ycyhzYXZlRWRpdG9ycyk7CgkJdWlQcmVmZXJlbmNlcy5zZXRQcm9tcHRCZWZvcmVJcnJldmVyc2libGVDaGFuZ2UocHJvbXB0SXJyZXZlcnNpYmxlLmdldFNlbGVjdGlvbigpKTsKCQlwcmVmZXJlbmNlcy5zZXRDcmVhdGVSZXNvdXJjZXNJbldvcmtzcGFjZShjcmVhdGVJbldvcmtzcGFjZS5nZXRTZWxlY3Rpb24oKSk7CgkKCQkvLyBhdXRvIHJlc3RhcnQgYW55IHNlcnZlcnMgdGhhdCBhcmUgcmVhZHkgZm9yIHJlc3RhcnQKCQlpZiAoYXV0b1Jlc3RhcnQuZ2V0U2VsZWN0aW9uKCkpCgkJCWF1dG9SZXN0YXJ0QWxsKCk7CgkKCQlyZXR1cm4gdHJ1ZTsKCX0KCgkvKioKCSAqIEF1dG9tYXRpY2FsbHkgcmVzdGFydCBhbnkgc2VydmVycyB0aGF0IHJlcXVpcmUgaXQuCgkgKi8KCXByb3RlY3RlZCBzdGF0aWMgdm9pZCBhdXRvUmVzdGFydEFsbCgpIHsKCQlUcmFjZS50cmFjZShUcmFjZS5GSU5FU1QsICJBdXRvIHJlc3RhcnRpbmcgYWxsIGRpcnR5IHNlcnZlcnMiKTsKCQoJCUlTZXJ2ZXJbXSBzZXJ2ZXJzID0gU2VydmVyQ29yZS5nZXRTZXJ2ZXJzKCk7CgkJaWYgKHNlcnZlcnMgIT0gbnVsbCkgewoJCQlpbnQgc2l6ZSA9IHNlcnZlcnMubGVuZ3RoOwoJCQlmb3IgKGludCBpID0gMDsgaSA8IHNpemU7IGkrKykgewoJCQkJSVNlcnZlciBzZXJ2ZXIgPSBzZXJ2ZXJzW2ldOwoJCQkJaWYgKHNlcnZlci5nZXRTZXJ2ZXJSZXN0YXJ0U3RhdGUoKSkgewoJCQkJCVN0cmluZyBtb2RlID0gc2VydmVyLmdldE1vZGUoKTsKCQkJCQlpZiAoc2VydmVyLmNhblJlc3RhcnQobW9kZSkpCgkJCQkJCXRyeSB7CgkJCQkJCQlUcmFjZS50cmFjZShUcmFjZS5GSU5FU1QsICJBdHRlbXB0aW5nIHRvIGF1dG8gcmVzdGFydCAiICsgc2VydmVyLmdldE5hbWUoKSk7CgkJCQkJCQlzZXJ2ZXIucmVzdGFydChtb2RlKTsKCQkJCQkJfSBjYXRjaCAoRXhjZXB0aW9uIGUpIHsKCQkJCQkJCVRyYWNlLnRyYWNlKFRyYWNlLlNFVkVSRSwgIkVycm9yIHJlc3RhcnRpbmc6ICIgKyBzZXJ2ZXIsIGUpOwoJCQkJCQl9CgkJCQl9CgkJCX0KCQl9Cgl9Cn0=