LyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogQ29weXJpZ2h0IChjKSAyMDAzLCAyMDA1IElCTSBDb3Jwb3JhdGlvbiBhbmQgb3RoZXJzLgogKiBBbGwgcmlnaHRzIHJlc2VydmVkLiBUaGlzIHByb2dyYW0gYW5kIHRoZSBhY2NvbXBhbnlpbmcgbWF0ZXJpYWxzCiAqIGFyZSBtYWRlIGF2YWlsYWJsZSB1bmRlciB0aGUgdGVybXMgb2YgdGhlIEVjbGlwc2UgUHVibGljIExpY2Vuc2UgdjEuMAogKiB3aGljaCBhY2NvbXBhbmllcyB0aGlzIGRpc3RyaWJ1dGlvbiwgYW5kIGlzIGF2YWlsYWJsZSBhdAogKiBodHRwOi8vd3d3LmVjbGlwc2Uub3JnL2xlZ2FsL2VwbC12MTAuaHRtbAqgKiAKICogQ29udHJpYnV0b3JzOgogKiAgICAgSUJNIENvcnBvcmF0aW9uIC0gSW5pdGlhbCBBUEkgYW5kIGltcGxlbWVudGF0aW9uCiAqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwpwYWNrYWdlIG9yZy5lY2xpcHNlLndzdC5zZXJ2ZXIudWkuaW50ZXJuYWw7CgppbXBvcnQgb3JnLmVjbGlwc2UuY29yZS5ydW50aW1lLklQcm9ncmVzc01vbml0b3I7CmltcG9ydCBvcmcuZWNsaXBzZS5qZmFjZS5kaWFsb2dzLkRpYWxvZzsKaW1wb3J0IG9yZy5lY2xpcHNlLmpmYWNlLnByZWZlcmVuY2UuUHJlZmVyZW5jZVBhZ2U7CmltcG9ydCBvcmcuZWNsaXBzZS53c3Quc2VydmVyLmNvcmUuSVNlcnZlcjsKaW1wb3J0IG9yZy5lY2xpcHNlLndzdC5zZXJ2ZXIuY29yZS5TZXJ2ZXJDb3JlOwppbXBvcnQgb3JnLmVjbGlwc2Uud3N0LnNlcnZlci5jb3JlLmludGVybmFsLlNlcnZlclByZWZlcmVuY2VzOwppbXBvcnQgb3JnLmVjbGlwc2Uuc3d0LlNXVDsKaW1wb3J0IG9yZy5lY2xpcHNlLnN3dC5ldmVudHMuU2VsZWN0aW9uQWRhcHRlcjsKaW1wb3J0IG9yZy5lY2xpcHNlLnN3dC5ldmVudHMuU2VsZWN0aW9uRXZlbnQ7CmltcG9ydCBvcmcuZWNsaXBzZS5zd3QubGF5b3V0LkdyaWREYXRhOwppbXBvcnQgb3JnLmVjbGlwc2Uuc3d0LmxheW91dC5HcmlkTGF5b3V0OwppbXBvcnQgb3JnLmVjbGlwc2Uuc3d0LndpZGdldHMuQnV0dG9uOwppbXBvcnQgb3JnLmVjbGlwc2Uuc3d0LndpZGdldHMuQ29tYm87CmltcG9ydCBvcmcuZWNsaXBzZS5zd3Qud2lkZ2V0cy5Db21wb3NpdGU7CmltcG9ydCBvcmcuZWNsaXBzZS5zd3Qud2lkZ2V0cy5Db250cm9sOwppbXBvcnQgb3JnLmVjbGlwc2Uuc3d0LndpZGdldHMuR3JvdXA7CmltcG9ydCBvcmcuZWNsaXBzZS5zd3Qud2lkZ2V0cy5MYWJlbDsKaW1wb3J0IG9yZy5lY2xpcHNlLnN3dC53aWRnZXRzLlNwaW5uZXI7CmltcG9ydCBvcmcuZWNsaXBzZS51aS5JV29ya2JlbmNoOwppbXBvcnQgb3JnLmVjbGlwc2UudWkuSVdvcmtiZW5jaFByZWZlcmVuY2VQYWdlOwppbXBvcnQgb3JnLmVjbGlwc2UudWkuUGxhdGZvcm1VSTsKaW1wb3J0IG9yZy5lY2xpcHNlLnVpLmhlbHAuSVdvcmtiZW5jaEhlbHBTeXN0ZW07Ci8qKgogKiBUaGUgcHJlZmVyZW5jZSBwYWdlIHRoYXQgaG9sZHMgc2VydmVyIHByb3BlcnRpZXMuCiAqLwpwdWJsaWMgY2xhc3MgU2VydmVyUHJlZmVyZW5jZVBhZ2UgZXh0ZW5kcyBQcmVmZXJlbmNlUGFnZSBpbXBsZW1lbnRzIElXb3JrYmVuY2hQcmVmZXJlbmNlUGFnZSB7Cglwcm90ZWN0ZWQgQnV0dG9uIHB1Ymxpc2hCZWZvcmVTdGFydDsKCXByb3RlY3RlZCBCdXR0b24gYXV0b1Jlc3RhcnQ7CgoJcHJvdGVjdGVkIEJ1dHRvbiBwcm9tcHRJcnJldmVyc2libGU7CgkKCXByb3RlY3RlZCBCdXR0b24gc2hvd09uQWN0aXZpdHk7CgkKCXByb3RlY3RlZCBieXRlIHNhdmVFZGl0b3JzOwoJCglwcm90ZWN0ZWQgQnV0dG9uIHNhdmVOZXZlcjsKCXByb3RlY3RlZCBCdXR0b24gc2F2ZVByb21wdDsKCXByb3RlY3RlZCBCdXR0b24gc2F2ZUF1dG87CgoJcHJvdGVjdGVkIFNlcnZlclByZWZlcmVuY2VzIHByZWZlcmVuY2VzOwoJcHJvdGVjdGVkIFNlcnZlclVJUHJlZmVyZW5jZXMgdWlQcmVmZXJlbmNlczsKCQoJcHJvdGVjdGVkIEJ1dHRvbiBhdXRvUHVibGlzaE9uQWN0aW9uOwoJcHJvdGVjdGVkIEJ1dHRvbiBhdXRvUHVibGlzaExvY2FsOwoJcHJvdGVjdGVkIFNwaW5uZXIgYXV0b1B1Ymxpc2hMb2NhbFRpbWU7Cglwcm90ZWN0ZWQgQnV0dG9uIGF1dG9QdWJsaXNoUmVtb3RlOwoJcHJvdGVjdGVkIFNwaW5uZXIgYXV0b1B1Ymxpc2hSZW1vdGVUaW1lOwoJCglwcm90ZWN0ZWQgQ29tYm8gbWFjaGluZVNwZWVkQ29tYm87CgoJLyoqCgkgKiBTZXJ2ZXJQcmVmZXJlbmNlc1BhZ2UgY29uc3RydWN0b3IgY29tbWVudC4KCSAqLwoJcHVibGljIFNlcnZlclByZWZlcmVuY2VQYWdlKCkgewoJCXN1cGVyKCk7CgkKCQlwcmVmZXJlbmNlcyA9IFNlcnZlclByZWZlcmVuY2VzLmdldEluc3RhbmNlKCk7CgkJdWlQcmVmZXJlbmNlcyA9IFNlcnZlclVJUGx1Z2luLmdldFByZWZlcmVuY2VzKCk7Cgl9CgkKCS8qKgoJICogQ3JlYXRlIHRoZSBwcmVmZXJlbmNlIG9wdGlvbnMuCgkgKgoJICogQHBhcmFtIHBhcmVudCBvcmcuZWNsaXBzZS5zd3Qud2lkZ2V0cy5Db21wb3NpdGUKCSAqIEByZXR1cm4gb3JnLmVjbGlwc2Uuc3d0LndpZGdldHMuQ29udHJvbAoJICovCglwcm90ZWN0ZWQgQ29udHJvbCBjcmVhdGVDb250ZW50cyhDb21wb3NpdGUgcGFyZW50KSB7CgkJaW5pdGlhbGl6ZURpYWxvZ1VuaXRzKHBhcmVudCk7CgkJSVdvcmtiZW5jaEhlbHBTeXN0ZW0gd2hzID0gUGxhdGZvcm1VSS5nZXRXb3JrYmVuY2goKS5nZXRIZWxwU3lzdGVtKCk7CgkJd2hzLnNldEhlbHAocGFyZW50LCBDb250ZXh0SWRzLlBSRUZfR0VORVJBTCk7CgkJCgkJQ29tcG9zaXRlIGNvbXBvc2l0ZSA9IG5ldyBDb21wb3NpdGUocGFyZW50LCBTV1QuTk9ORSk7CgkJR3JpZExheW91dCBsYXlvdXQgPSBuZXcgR3JpZExheW91dCgpOwoJCWxheW91dC5ob3Jpem9udGFsU3BhY2luZyA9IGNvbnZlcnRIb3Jpem9udGFsRExVc1RvUGl4ZWxzKDQpOwoJCWxheW91dC52ZXJ0aWNhbFNwYWNpbmcgPSBjb252ZXJ0VmVydGljYWxETFVzVG9QaXhlbHMoNCk7CgkJbGF5b3V0Lm1hcmdpbldpZHRoID0gMDsKCQlsYXlvdXQubWFyZ2luSGVpZ2h0ID0gMDsKCQlsYXlvdXQubnVtQ29sdW1ucyA9IDQ7CgkJY29tcG9zaXRlLnNldExheW91dChsYXlvdXQpOwoJCUdyaWREYXRhIGRhdGEgPSBuZXcgR3JpZERhdGEoR3JpZERhdGEuRklMTF9IT1JJWk9OVEFMIHwgR3JpZERhdGEuVkVSVElDQUxfQUxJR05fRklMTCk7CgkJY29tcG9zaXRlLnNldExheW91dERhdGEoZGF0YSk7CgkJCgkJcHVibGlzaEJlZm9yZVN0YXJ0ID0gbmV3IEJ1dHRvbihjb21wb3NpdGUsIFNXVC5DSEVDSyk7CgkJcHVibGlzaEJlZm9yZVN0YXJ0LnNldFRleHQoTWVzc2FnZXMucHJlZkF1dG9QdWJsaXNoKTsKCQlkYXRhID0gbmV3IEdyaWREYXRhKEdyaWREYXRhLkhPUklaT05UQUxfQUxJR05fRklMTCk7CgkJZGF0YS5ob3Jpem9udGFsU3BhbiA9IDQ7CgkJcHVibGlzaEJlZm9yZVN0YXJ0LnNldExheW91dERhdGEoZGF0YSk7CgkJcHVibGlzaEJlZm9yZVN0YXJ0LnNldFNlbGVjdGlvbihwcmVmZXJlbmNlcy5pc0F1dG9QdWJsaXNoaW5nKCkpOwoJCXdocy5zZXRIZWxwKHB1Ymxpc2hCZWZvcmVTdGFydCwgQ29udGV4dElkcy5QUkVGX0dFTkVSQUxfUFVCTElTSF9CRUZPUkVfU1RBUlQpOwoJCQoJCWF1dG9QdWJsaXNoTG9jYWwgPSBuZXcgQnV0dG9uKGNvbXBvc2l0ZSwgU1dULkNIRUNLKTsKCQlhdXRvUHVibGlzaExvY2FsLnNldFRleHQoTWVzc2FnZXMucHJlZkF1dG9QdWJsaXNoTG9jYWwpOwoJCWRhdGEgPSBuZXcgR3JpZERhdGEoR3JpZERhdGEuRklMTF9IT1JJWk9OVEFMKTsKCQlkYXRhLmhvcml6b250YWxTcGFuID0gMjsKCQlhdXRvUHVibGlzaExvY2FsLnNldExheW91dERhdGEoZGF0YSk7CgkJYXV0b1B1Ymxpc2hMb2NhbC5zZXRTZWxlY3Rpb24ocHJlZmVyZW5jZXMuZ2V0QXV0b1B1Ymxpc2hMb2NhbCgpKTsKCQkvL1dvcmtiZW5jaEhlbHAuc2V0SGVscChzYXZlUHJvbXB0LCBDb250ZXh0SWRzLik7CgkJCgkJYXV0b1B1Ymxpc2hMb2NhbFRpbWUgPSBuZXcgU3Bpbm5lcihjb21wb3NpdGUsIFNXVC5CT1JERVIpOwoJCWF1dG9QdWJsaXNoTG9jYWxUaW1lLnNldE1pbmltdW0oMCk7CgkJYXV0b1B1Ymxpc2hMb2NhbFRpbWUuc2V0TWF4aW11bSgxMjApOwoJCWF1dG9QdWJsaXNoTG9jYWxUaW1lLnNldFNlbGVjdGlvbihwcmVmZXJlbmNlcy5nZXRBdXRvUHVibGlzaExvY2FsVGltZSgpKTsKCQlhdXRvUHVibGlzaExvY2FsVGltZS5zZXRFbmFibGVkKGF1dG9QdWJsaXNoTG9jYWwuZ2V0U2VsZWN0aW9uKCkpOwoJCWRhdGEgPSBuZXcgR3JpZERhdGEoR3JpZERhdGEuSE9SSVpPTlRBTF9BTElHTl9FTkQpOwoJCWRhdGEud2lkdGhIaW50ID0gNjA7CgkJYXV0b1B1Ymxpc2hMb2NhbFRpbWUuc2V0TGF5b3V0RGF0YShkYXRhKTsKCQkKCQlMYWJlbCBsYWJlbCA9IG5ldyBMYWJlbChjb21wb3NpdGUsIFNXVC5OT05FKTsKCQlsYWJlbC5zZXRUZXh0KE1lc3NhZ2VzLnByZWZBdXRvUHVibGlzaFNlY29uZHMpOwoJCWRhdGEgPSBuZXcgR3JpZERhdGEoR3JpZERhdGEuSE9SSVpPTlRBTF9BTElHTl9FTkQpOwoJCWxhYmVsLnNldExheW91dERhdGEoZGF0YSk7CgkJCgkJYXV0b1B1Ymxpc2hMb2NhbC5hZGRTZWxlY3Rpb25MaXN0ZW5lcihuZXcgU2VsZWN0aW9uQWRhcHRlcigpIHsKCQkJcHVibGljIHZvaWQgd2lkZ2V0U2VsZWN0ZWQoU2VsZWN0aW9uRXZlbnQgZSkgewoJCQkJYXV0b1B1Ymxpc2hMb2NhbFRpbWUuc2V0RW5hYmxlZChhdXRvUHVibGlzaExvY2FsLmdldFNlbGVjdGlvbigpKTsKCQkJfQoJCX0pOwoJCQoJCWF1dG9QdWJsaXNoUmVtb3RlID0gbmV3IEJ1dHRvbihjb21wb3NpdGUsIFNXVC5DSEVDSyk7CgkJYXV0b1B1Ymxpc2hSZW1vdGUuc2V0VGV4dChNZXNzYWdlcy5wcmVmQXV0b1B1Ymxpc2hSZW1vdGUpOwoJCWRhdGEgPSBuZXcgR3JpZERhdGEoR3JpZERhdGEuSE9SSVpPTlRBTF9BTElHTl9GSUxMKTsKCQlkYXRhLmhvcml6b250YWxTcGFuID0gMjsKCQlhdXRvUHVibGlzaFJlbW90ZS5zZXRMYXlvdXREYXRhKGRhdGEpOwoJCWF1dG9QdWJsaXNoUmVtb3RlLnNldFNlbGVjdGlvbihwcmVmZXJlbmNlcy5nZXRBdXRvUHVibGlzaFJlbW90ZSgpKTsKCQkvL1dvcmtiZW5jaEhlbHAuc2V0SGVscChhdXRvUHVibGlzaFJlbW90ZSwgQ29udGV4dElkcy4pOwoJCQoJCWF1dG9QdWJsaXNoUmVtb3RlVGltZSA9IG5ldyBTcGlubmVyKGNvbXBvc2l0ZSwgU1dULkJPUkRFUik7CgkJYXV0b1B1Ymxpc2hMb2NhbFRpbWUuc2V0TWluaW11bSgwKTsKCQlhdXRvUHVibGlzaExvY2FsVGltZS5zZXRNYXhpbXVtKDEyMCk7CgkJYXV0b1B1Ymxpc2hSZW1vdGVUaW1lLnNldFNlbGVjdGlvbihwcmVmZXJlbmNlcy5nZXRBdXRvUHVibGlzaFJlbW90ZVRpbWUoKSk7CgkJYXV0b1B1Ymxpc2hSZW1vdGVUaW1lLnNldEVuYWJsZWQoYXV0b1B1Ymxpc2hSZW1vdGUuZ2V0U2VsZWN0aW9uKCkpOwoJCWRhdGEgPSBuZXcgR3JpZERhdGEoR3JpZERhdGEuSE9SSVpPTlRBTF9BTElHTl9GSUxMKTsKCQlkYXRhLndpZHRoSGludCA9IDYwOwoJCWF1dG9QdWJsaXNoUmVtb3RlVGltZS5zZXRMYXlvdXREYXRhKGRhdGEpOwoJCQoJCWF1dG9QdWJsaXNoUmVtb3RlLmFkZFNlbGVjdGlvbkxpc3RlbmVyKG5ldyBTZWxlY3Rpb25BZGFwdGVyKCkgewoJCQlwdWJsaWMgdm9pZCB3aWRnZXRTZWxlY3RlZChTZWxlY3Rpb25FdmVudCBlKSB7CgkJCQlhdXRvUHVibGlzaFJlbW90ZVRpbWUuc2V0RW5hYmxlZChhdXRvUHVibGlzaFJlbW90ZS5nZXRTZWxlY3Rpb24oKSk7CgkJCX0KCQl9KTsKCQkKCQlsYWJlbCA9IG5ldyBMYWJlbChjb21wb3NpdGUsIFNXVC5OT05FKTsKCQlsYWJlbC5zZXRUZXh0KE1lc3NhZ2VzLnByZWZBdXRvUHVibGlzaFNlY29uZHMpOwoJCWRhdGEgPSBuZXcgR3JpZERhdGEoR3JpZERhdGEuSE9SSVpPTlRBTF9BTElHTl9FTkQpOwoJCWxhYmVsLnNldExheW91dERhdGEoZGF0YSk7CgkJCgkJYXV0b1Jlc3RhcnQgPSBuZXcgQnV0dG9uKGNvbXBvc2l0ZSwgU1dULkNIRUNLKTsKCQlhdXRvUmVzdGFydC5zZXRUZXh0KE1lc3NhZ2VzLnByZWZBdXRvUmVzdGFydCk7CgkJZGF0YSA9IG5ldyBHcmlkRGF0YShHcmlkRGF0YS5IT1JJWk9OVEFMX0FMSUdOX0ZJTEwpOwoJCWRhdGEuaG9yaXpvbnRhbFNwYW4gPSA0OwoJCWF1dG9SZXN0YXJ0LnNldExheW91dERhdGEoZGF0YSk7CgkJYXV0b1Jlc3RhcnQuc2V0U2VsZWN0aW9uKHByZWZlcmVuY2VzLmlzQXV0b1Jlc3RhcnRpbmcoKSk7CgkJd2hzLnNldEhlbHAoYXV0b1Jlc3RhcnQsIENvbnRleHRJZHMuUFJFRl9HRU5FUkFMX0FVVE9fUkVTVEFSVCk7CgkJCgkJcHJvbXB0SXJyZXZlcnNpYmxlID0gbmV3IEJ1dHRvbihjb21wb3NpdGUsIFNXVC5DSEVDSyk7CgkJcHJvbXB0SXJyZXZlcnNpYmxlLnNldFRleHQoTWVzc2FnZXMucHJlZlByb21wdElycmV2ZXJzaWJsZSk7CgkJZGF0YSA9IG5ldyBHcmlkRGF0YShHcmlkRGF0YS5IT1JJWk9OVEFMX0FMSUdOX0ZJTEwpOwoJCWRhdGEuaG9yaXpvbnRhbFNwYW4gPSA0OwoJCXByb21wdElycmV2ZXJzaWJsZS5zZXRMYXlvdXREYXRhKGRhdGEpOwoJCXByb21wdElycmV2ZXJzaWJsZS5zZXRTZWxlY3Rpb24odWlQcmVmZXJlbmNlcy5nZXRQcm9tcHRCZWZvcmVJcnJldmVyc2libGVDaGFuZ2UoKSk7CgkJd2hzLnNldEhlbHAocHJvbXB0SXJyZXZlcnNpYmxlLCBDb250ZXh0SWRzLlBSRUZfR0VORVJBTF9QUk9NUFRfSVJSRVZFUlNJQkxFKTsKCQkKCQlzaG93T25BY3Rpdml0eSA9IG5ldyBCdXR0b24oY29tcG9zaXRlLCBTV1QuQ0hFQ0spOwoJCXNob3dPbkFjdGl2aXR5LnNldFRleHQoTWVzc2FnZXMucHJlZlNob3dPbkFjdGl2aXR5KTsKCQlkYXRhID0gbmV3IEdyaWREYXRhKEdyaWREYXRhLkhPUklaT05UQUxfQUxJR05fRklMTCk7CgkJZGF0YS5ob3Jpem9udGFsU3BhbiA9IDQ7CgkJc2hvd09uQWN0aXZpdHkuc2V0TGF5b3V0RGF0YShkYXRhKTsKCQlzaG93T25BY3Rpdml0eS5zZXRTZWxlY3Rpb24odWlQcmVmZXJlbmNlcy5nZXRTaG93T25BY3Rpdml0eSgpKTsKCQl3aHMuc2V0SGVscChzaG93T25BY3Rpdml0eSwgQ29udGV4dElkcy5QUkVGX0dFTkVSQUxfU0hPV19PTl9BQ1RJVklUWSk7CgkJCgkJLy8gc2F2ZSBlZGl0b3JzIGdyb3VwCgkJR3JvdXAgc2F2ZUVkaXRvckdyb3VwID0gbmV3IEdyb3VwKGNvbXBvc2l0ZSwgU1dULk5PTkUpOwoJCXNhdmVFZGl0b3JHcm91cC5zZXRUZXh0KE1lc3NhZ2VzLnByZWZTYXZlRWRpdG9yc0dyb3VwKTsKCQkKCQlsYXlvdXQgPSBuZXcgR3JpZExheW91dCgpOwoJCWxheW91dC5udW1Db2x1bW5zID0gMzsKCQlzYXZlRWRpdG9yR3JvdXAuc2V0TGF5b3V0KGxheW91dCk7CgkJZGF0YSA9IG5ldyBHcmlkRGF0YShHcmlkRGF0YS5GSUxMX0hPUklaT05UQUwpOwoJCWRhdGEuaG9yaXpvbnRhbFNwYW4gPSA0OwoJCXNhdmVFZGl0b3JHcm91cC5zZXRMYXlvdXREYXRhKGRhdGEpOwoJCQoJCXNhdmVOZXZlciA9IG5ldyBCdXR0b24oc2F2ZUVkaXRvckdyb3VwLCBTV1QuUkFESU8pOwoJCXNhdmVOZXZlci5zZXRUZXh0KE1lc3NhZ2VzLnByZWZTYXZlRWRpdG9yc05ldmVyKTsKCQlzYXZlTmV2ZXIuYWRkU2VsZWN0aW9uTGlzdGVuZXIobmV3IFNlbGVjdGlvbkFkYXB0ZXIoKSB7CgkJCXB1YmxpYyB2b2lkIHdpZGdldFNlbGVjdGVkKFNlbGVjdGlvbkV2ZW50IGV2ZW50KSB7CgkJCQlzYXZlRWRpdG9ycyA9IFNlcnZlclVJUHJlZmVyZW5jZXMuU0FWRV9FRElUT1JTX05FVkVSOwoJCQl9CgkJfSk7CgkJd2hzLnNldEhlbHAoc2F2ZU5ldmVyLCBDb250ZXh0SWRzLlBSRUZfR0VORVJBTF9TQVZFX0VESVRPUlMpOwoJCQoJCXNhdmVQcm9tcHQgPSBuZXcgQnV0dG9uKHNhdmVFZGl0b3JHcm91cCwgU1dULlJBRElPKTsKCQlzYXZlUHJvbXB0LnNldFRleHQoTWVzc2FnZXMucHJlZlNhdmVFZGl0b3JzUHJvbXB0KTsKCQlzYXZlUHJvbXB0LmFkZFNlbGVjdGlvbkxpc3RlbmVyKG5ldyBTZWxlY3Rpb25BZGFwdGVyKCkgewoJCQlwdWJsaWMgdm9pZCB3aWRnZXRTZWxlY3RlZChTZWxlY3Rpb25FdmVudCBldmVudCkgewoJCQkJc2F2ZUVkaXRvcnMgPSBTZXJ2ZXJVSVByZWZlcmVuY2VzLlNBVkVfRURJVE9SU19QUk9NUFQ7CgkJCX0KCQl9KTsKCQl3aHMuc2V0SGVscChzYXZlUHJvbXB0LCBDb250ZXh0SWRzLlBSRUZfR0VORVJBTF9TQVZFX0VESVRPUlMpOwoJCQoJCXNhdmVBdXRvID0gbmV3IEJ1dHRvbihzYXZlRWRpdG9yR3JvdXAsIFNXVC5SQURJTyk7CgkJc2F2ZUF1dG8uc2V0VGV4dChNZXNzYWdlcy5wcmVmU2F2ZUVkaXRvcnNBdXRvc2F2ZSk7CgkJc2F2ZUF1dG8uYWRkU2VsZWN0aW9uTGlzdGVuZXIobmV3IFNlbGVjdGlvbkFkYXB0ZXIoKSB7CgkJCXB1YmxpYyB2b2lkIHdpZGdldFNlbGVjdGVkKFNlbGVjdGlvbkV2ZW50IGV2ZW50KSB7CgkJCQlzYXZlRWRpdG9ycyA9IFNlcnZlclVJUHJlZmVyZW5jZXMuU0FWRV9FRElUT1JTX0FVVE87CgkJCX0KCQl9KTsKCQl3aHMuc2V0SGVscChzYXZlQXV0bywgQ29udGV4dElkcy5QUkVGX0dFTkVSQUxfU0FWRV9FRElUT1JTKTsKCQkKCQlsYWJlbCA9IG5ldyBMYWJlbChjb21wb3NpdGUsIFNXVC5OT05FKTsKCQlsYWJlbC5zZXRUZXh0KE1lc3NhZ2VzLnByZWZNYWNoaW5lU3BlZWQpOwoJCQoJCW1hY2hpbmVTcGVlZENvbWJvID0gbmV3IENvbWJvKGNvbXBvc2l0ZSwgU1dULlJFQURfT05MWSk7CgkJU3RyaW5nW10gaXRlbXMgPSBuZXcgU3RyaW5nW10gewoJCQlNZXNzYWdlcy5wcmVmTWFjaGluZVNwZWVkVmVyeVNsb3csCgkJCU1lc3NhZ2VzLnByZWZNYWNoaW5lU3BlZWRTbG93LAoJCQlNZXNzYWdlcy5wcmVmTWFjaGluZVNwZWVkQXZlcmFnZSwKCQkJTWVzc2FnZXMucHJlZk1hY2hpbmVTcGVlZEZhc3QsCgkJCU1lc3NhZ2VzLnByZWZNYWNoaW5lU3BlZWRWZXJ5RmFzdAoJCX07CgkJbWFjaGluZVNwZWVkQ29tYm8uc2V0SXRlbXMoaXRlbXMpOwoJCW1hY2hpbmVTcGVlZENvbWJvLnNlbGVjdChwcmVmZXJlbmNlcy5nZXRNYWNoaW5lU3BlZWQoKSAvIDIpOwoJCWRhdGEgPSBuZXcgR3JpZERhdGEoR3JpZERhdGEuSE9SSVpPTlRBTF9BTElHTl9GSUxMKTsKCQlkYXRhLmhvcml6b250YWxTcGFuID0gMzsKCQltYWNoaW5lU3BlZWRDb21iby5zZXRMYXlvdXREYXRhKGRhdGEpOwoJCS8vd2hzLnNldEhlbHAobWFjaGluZVNwZWVkU2xpZGVyLCBDb250ZXh0SWRzLlBSRUZfR0VORVJBTF9QUk9NUFRfSVJSRVZFUlNJQkxFKTsKCQkKCQlzZXRTYXZlRWRpdG9yU3RhdHVzKHVpUHJlZmVyZW5jZXMuZ2V0U2F2ZUVkaXRvcnMoKSk7CgkJCgkJRGlhbG9nLmFwcGx5RGlhbG9nRm9udChjb21wb3NpdGUpOwoJCgkJcmV0dXJuIGNvbXBvc2l0ZTsKCX0KCQoJcHJvdGVjdGVkIHZvaWQgc2V0U2F2ZUVkaXRvclN0YXR1cyhieXRlIHN0YXR1cykgewoJCXNhdmVFZGl0b3JzID0gc3RhdHVzOwoJCXNhdmVOZXZlci5zZXRTZWxlY3Rpb24oc2F2ZUVkaXRvcnMgPT0gU2VydmVyVUlQcmVmZXJlbmNlcy5TQVZFX0VESVRPUlNfTkVWRVIpOwoJCXNhdmVQcm9tcHQuc2V0U2VsZWN0aW9uKHNhdmVFZGl0b3JzID09IFNlcnZlclVJUHJlZmVyZW5jZXMuU0FWRV9FRElUT1JTX1BST01QVCk7CgkJc2F2ZUF1dG8uc2V0U2VsZWN0aW9uKHNhdmVFZGl0b3JzID09IFNlcnZlclVJUHJlZmVyZW5jZXMuU0FWRV9FRElUT1JTX0FVVE8pOyAKCX0KCQoJLyoqCgkgKiBJbml0aWFsaXplcyB0aGlzIHByZWZlcmVuY2UgcGFnZSB1c2luZyB0aGUgcGFzc2VkIHdvcmtiZW5jaC4KCSAqCgkgKiBAcGFyYW0gd29ya2JlbmNoIHRoZSBjdXJyZW50IHdvcmtiZW5jaAoJICovCglwdWJsaWMgdm9pZCBpbml0KElXb3JrYmVuY2ggd29ya2JlbmNoKSB7CgkJLy8gZG8gbm90aGluZwoJfQoJCgkvKioKCSAqIFBlcmZvcm1zIHNwZWNpYWwgcHJvY2Vzc2luZyB3aGVuIHRoaXMgcGFnZSdzIERlZmF1bHRzIGJ1dHRvbiBoYXMgYmVlbiBwcmVzc2VkLgoJICovCglwcm90ZWN0ZWQgdm9pZCBwZXJmb3JtRGVmYXVsdHMoKSB7CgkJYXV0b1Jlc3RhcnQuc2V0U2VsZWN0aW9uKHByZWZlcmVuY2VzLmlzRGVmYXVsdEF1dG9SZXN0YXJ0aW5nKCkpOwoJCXB1Ymxpc2hCZWZvcmVTdGFydC5zZXRTZWxlY3Rpb24ocHJlZmVyZW5jZXMuaXNEZWZhdWx0QXV0b1B1Ymxpc2hpbmcoKSk7CgkJcHJvbXB0SXJyZXZlcnNpYmxlLnNldFNlbGVjdGlvbih1aVByZWZlcmVuY2VzLmdldERlZmF1bHRQcm9tcHRCZWZvcmVJcnJldmVyc2libGVDaGFuZ2UoKSk7CgkJc2hvd09uQWN0aXZpdHkuc2V0U2VsZWN0aW9uKHVpUHJlZmVyZW5jZXMuZ2V0RGVmYXVsdFNob3dPbkFjdGl2aXR5KCkpOwoJCQoJCWF1dG9QdWJsaXNoTG9jYWwuc2V0U2VsZWN0aW9uKHByZWZlcmVuY2VzLmdldERlZmF1bHRBdXRvUHVibGlzaExvY2FsKCkpOwoJCWF1dG9QdWJsaXNoTG9jYWxUaW1lLnNldFNlbGVjdGlvbihwcmVmZXJlbmNlcy5nZXREZWZhdWx0QXV0b1B1Ymxpc2hMb2NhbFRpbWUoKSk7CgkJYXV0b1B1Ymxpc2hSZW1vdGUuc2V0U2VsZWN0aW9uKHByZWZlcmVuY2VzLmdldERlZmF1bHRBdXRvUHVibGlzaFJlbW90ZSgpKTsKCQlhdXRvUHVibGlzaFJlbW90ZVRpbWUuc2V0U2VsZWN0aW9uKHByZWZlcmVuY2VzLmdldERlZmF1bHRBdXRvUHVibGlzaFJlbW90ZVRpbWUoKSk7CgkJCgkJbWFjaGluZVNwZWVkQ29tYm8uc2VsZWN0KHByZWZlcmVuY2VzLmdldERlZmF1bHRNYWNoaW5lU3BlZWQoKSAvIDIpOwoJCQoJCXNldFNhdmVFZGl0b3JTdGF0dXModWlQcmVmZXJlbmNlcy5nZXREZWZhdWx0U2F2ZUVkaXRvcnMoKSk7CgkKCQlzdXBlci5wZXJmb3JtRGVmYXVsdHMoKTsKCX0KCgkvKioKCSAqIEBzZWUgb3JnLmVjbGlwc2UuamZhY2UucHJlZmVyZW5jZS5JUHJlZmVyZW5jZVBhZ2UjcGVyZm9ybU9rKCkKCSAqLwoJcHVibGljIGJvb2xlYW4gcGVyZm9ybU9rKCkgewoJCXByZWZlcmVuY2VzLnNldEF1dG9QdWJsaXNoaW5nKHB1Ymxpc2hCZWZvcmVTdGFydC5nZXRTZWxlY3Rpb24oKSk7CgkJcHJlZmVyZW5jZXMuc2V0QXV0b1Jlc3RhcnRpbmcoYXV0b1Jlc3RhcnQuZ2V0U2VsZWN0aW9uKCkpOwoJCXVpUHJlZmVyZW5jZXMuc2V0U2F2ZUVkaXRvcnMoc2F2ZUVkaXRvcnMpOwoJCXVpUHJlZmVyZW5jZXMuc2V0UHJvbXB0QmVmb3JlSXJyZXZlcnNpYmxlQ2hhbmdlKHByb21wdElycmV2ZXJzaWJsZS5nZXRTZWxlY3Rpb24oKSk7CgkJdWlQcmVmZXJlbmNlcy5zZXRTaG93T25BY3Rpdml0eShzaG93T25BY3Rpdml0eS5nZXRTZWxlY3Rpb24oKSk7CgkJCgkJcHJlZmVyZW5jZXMuc2V0QXV0b1B1Ymxpc2hMb2NhbChhdXRvUHVibGlzaExvY2FsLmdldFNlbGVjdGlvbigpKTsKCQlwcmVmZXJlbmNlcy5zZXRBdXRvUHVibGlzaExvY2FsVGltZShhdXRvUHVibGlzaExvY2FsVGltZS5nZXRTZWxlY3Rpb24oKSk7CgkJcHJlZmVyZW5jZXMuc2V0QXV0b1B1Ymxpc2hSZW1vdGUoYXV0b1B1Ymxpc2hSZW1vdGUuZ2V0U2VsZWN0aW9uKCkpOwoJCXByZWZlcmVuY2VzLnNldEF1dG9QdWJsaXNoUmVtb3RlVGltZShhdXRvUHVibGlzaFJlbW90ZVRpbWUuZ2V0U2VsZWN0aW9uKCkpOwoJCQoJCXByZWZlcmVuY2VzLnNldE1hY2hpbmVTcGVlZChtYWNoaW5lU3BlZWRDb21iby5nZXRTZWxlY3Rpb25JbmRleCgpICogMik7CgkKCQkvLyBhdXRvIHJlc3RhcnQgYW55IHNlcnZlcnMgdGhhdCBhcmUgcmVhZHkgZm9yIHJlc3RhcnQKCQlpZiAoYXV0b1Jlc3RhcnQuZ2V0U2VsZWN0aW9uKCkpCgkJCWF1dG9SZXN0YXJ0QWxsKCk7CgkKCQlyZXR1cm4gdHJ1ZTsKCX0KCgkvKioKCSAqIEF1dG9tYXRpY2FsbHkgcmVzdGFydCBhbnkgc2VydmVycyB0aGF0IHJlcXVpcmUgaXQuCgkgKi8KCXByb3RlY3RlZCBzdGF0aWMgdm9pZCBhdXRvUmVzdGFydEFsbCgpIHsKCQlUcmFjZS50cmFjZShUcmFjZS5GSU5FU1QsICJBdXRvIHJlc3RhcnRpbmcgYWxsIGRpcnR5IHNlcnZlcnMiKTsKCQoJCUlTZXJ2ZXJbXSBzZXJ2ZXJzID0gU2VydmVyQ29yZS5nZXRTZXJ2ZXJzKCk7CgkJaWYgKHNlcnZlcnMgIT0gbnVsbCkgewoJCQlpbnQgc2l6ZSA9IHNlcnZlcnMubGVuZ3RoOwoJCQlmb3IgKGludCBpID0gMDsgaSA8IHNpemU7IGkrKykgewoJCQkJSVNlcnZlciBzZXJ2ZXIgPSBzZXJ2ZXJzW2ldOwoJCQkJaWYgKHNlcnZlci5nZXRTZXJ2ZXJSZXN0YXJ0U3RhdGUoKSkgewoJCQkJCVN0cmluZyBtb2RlID0gc2VydmVyLmdldE1vZGUoKTsKCQkJCQlpZiAoc2VydmVyLmNhblJlc3RhcnQobW9kZSkuaXNPSygpKQoJCQkJCQl0cnkgewoJCQkJCQkJVHJhY2UudHJhY2UoVHJhY2UuRklORVNULCAiQXR0ZW1wdGluZyB0byBhdXRvIHJlc3RhcnQgIiArIHNlcnZlci5nZXROYW1lKCkpOwoJCQkJCQkJc2VydmVyLnJlc3RhcnQobW9kZSwgKElQcm9ncmVzc01vbml0b3IpbnVsbCk7CgkJCQkJCX0gY2F0Y2ggKEV4Y2VwdGlvbiBlKSB7CgkJCQkJCQlUcmFjZS50cmFjZShUcmFjZS5TRVZFUkUsICJFcnJvciByZXN0YXJ0aW5nOiAiICsgc2VydmVyLCBlKTsKCQkJCQkJfQoJCQkJfQoJCQl9CgkJfQoJfQp9