cGFja2FnZSBvcmcuZWNsaXBzZS53c3Quc2VydmVyLnVpLmludGVybmFsOwovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiBDb3B5cmlnaHQgKGMpIDIwMDMgSUJNIENvcnBvcmF0aW9uIGFuZCBvdGhlcnMuCiAqIEFsbCByaWdodHMgcmVzZXJ2ZWQuIKAgVGhpcyBwcm9ncmFtIGFuZCB0aGUgYWNjb21wYW55aW5nIG1hdGVyaWFscwogKiBhcmUgbWFkZSBhdmFpbGFibGUgdW5kZXIgdGhlIHRlcm1zIG9mIHRoZSBDb21tb24gUHVibGljIExpY2Vuc2UgdjEuMAogKiB3aGljaCBhY2NvbXBhbmllcyB0aGlzIGRpc3RyaWJ1dGlvbiwgYW5kIGlzIGF2YWlsYWJsZSBhdAogKiBodHRwOi8vd3d3LmVjbGlwc2Uub3JnL2xlZ2FsL2NwbC12MTAuaHRtbAqgKgogKiBDb250cmlidXRvcnM6CiAqICAgIElCTSAtIEluaXRpYWwgQVBJIGFuZCBpbXBsZW1lbnRhdGlvbgogKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKi8KaW1wb3J0IG9yZy5lY2xpcHNlLmNvcmUucmVzb3VyY2VzLklQcm9qZWN0OwppbXBvcnQgb3JnLmVjbGlwc2UuY29yZS5ydW50aW1lLkNvcmVFeGNlcHRpb247CmltcG9ydCBvcmcuZWNsaXBzZS5jb3JlLnJ1bnRpbWUuSUFkYXB0YWJsZTsKaW1wb3J0IG9yZy5lY2xpcHNlLmNvcmUucnVudGltZS5OdWxsUHJvZ3Jlc3NNb25pdG9yOwppbXBvcnQgb3JnLmVjbGlwc2UuamZhY2UuZGlhbG9ncy5EaWFsb2c7CmltcG9ydCBvcmcuZWNsaXBzZS53c3Quc2VydmVyLmNvcmUuKjsKaW1wb3J0IG9yZy5lY2xpcHNlLndzdC5zZXJ2ZXIuY29yZS5tb2RlbC5JUHJvamVjdE1vZHVsZTsKaW1wb3J0IG9yZy5lY2xpcHNlLndzdC5zZXJ2ZXIudWkuU2VydmVyVUlDb3JlOwppbXBvcnQgb3JnLmVjbGlwc2Uuc3d0LlNXVDsKaW1wb3J0IG9yZy5lY2xpcHNlLnN3dC5ldmVudHMuU2VsZWN0aW9uQWRhcHRlcjsKaW1wb3J0IG9yZy5lY2xpcHNlLnN3dC5ldmVudHMuU2VsZWN0aW9uRXZlbnQ7CmltcG9ydCBvcmcuZWNsaXBzZS5zd3QubGF5b3V0LkdyaWREYXRhOwppbXBvcnQgb3JnLmVjbGlwc2Uuc3d0LmxheW91dC5HcmlkTGF5b3V0OwppbXBvcnQgb3JnLmVjbGlwc2Uuc3d0LndpZGdldHMuQ29tcG9zaXRlOwppbXBvcnQgb3JnLmVjbGlwc2Uuc3d0LndpZGdldHMuQ29udHJvbDsKaW1wb3J0IG9yZy5lY2xpcHNlLnN3dC53aWRnZXRzLkxhYmVsOwppbXBvcnQgb3JnLmVjbGlwc2Uuc3d0LndpZGdldHMuVGFibGU7CmltcG9ydCBvcmcuZWNsaXBzZS5zd3Qud2lkZ2V0cy5UYWJsZUl0ZW07CmltcG9ydCBvcmcuZWNsaXBzZS51aS5kaWFsb2dzLlByb3BlcnR5UGFnZTsKCi8qKgogKiBQcm9wZXJ0eVBhZ2UgZm9yIElQcm9qZWN0cy4gSXQgc2hvd3MgdGhlIHNlcnZlciBhbmQgcnVudGltZSBwcmVmZXJlbmNlIGZvciB0aGUgcHJvamVjdC4KICovCnB1YmxpYyBjbGFzcyBQcm9qZWN0UHJvcGVydHlQYWdlIGV4dGVuZHMgUHJvcGVydHlQYWdlIHsKCXByb3RlY3RlZCBJUHJvamVjdCBwcm9qZWN0OwoJcHJvdGVjdGVkIElQcm9qZWN0TW9kdWxlIG1vZHVsZTsKCXByb3RlY3RlZCBJU2VydmVyIHNlcnZlcjsKCQoJcHJvdGVjdGVkIFJ1bnRpbWVUYXJnZXRDb21wb3NpdGUgcnRDb21wOwoKCS8qKgoJICogUHJvamVjdFByb3BlcnR5UGFnZSBjb25zdHJ1Y3RvciBjb21tZW50LgoJICovCglwdWJsaWMgUHJvamVjdFByb3BlcnR5UGFnZSgpIHsKCQlzdXBlcigpOwoJfQoKCS8qKgoJICogQ3JlYXRlIHRoZSBib2R5IG9mIHRoZSBwYWdlLgoJICoKCSAqIEBwYXJhbSBwYXJlbnQgb3JnLmVjbGlwc2Uuc3d0LndpZGdldHMuQ29tcG9zaXRlCgkgKiBAcmV0dXJuIG9yZy5lY2xpcHNlLnN3dC53aWRnZXRzLkNvbnRyb2wKCSAqLwoJcHJvdGVjdGVkIENvbnRyb2wgY3JlYXRlQ29udGVudHMoQ29tcG9zaXRlIHBhcmVudCkgewoJCXRyeSB7CgkJCUlBZGFwdGFibGUgZWxlbWVudCA9IGdldEVsZW1lbnQoKTsKCQkJaWYgKGVsZW1lbnQgaW5zdGFuY2VvZiBJUHJvamVjdCkKCQkJCXByb2plY3QgPSAoSVByb2plY3QpIGVsZW1lbnQ7CgoJCQlDb21wb3NpdGUgY29tcG9zaXRlID0gbmV3IENvbXBvc2l0ZShwYXJlbnQsIFNXVC5OT05FKTsKCQkJR3JpZExheW91dCBsYXlvdXQgPSBuZXcgR3JpZExheW91dCgpOwoJCQlsYXlvdXQubWFyZ2luSGVpZ2h0ID0gMDsKCQkJbGF5b3V0Lm1hcmdpbldpZHRoID0gMDsKCQkJbGF5b3V0Lm51bUNvbHVtbnMgPSAzOwoJCQlsYXlvdXQudmVydGljYWxTcGFjaW5nID0gMTA7CgkJCWNvbXBvc2l0ZS5zZXRMYXlvdXQobGF5b3V0KTsKCQkJY29tcG9zaXRlLnNldExheW91dERhdGEobmV3IEdyaWREYXRhKEdyaWREYXRhLkZJTExfQk9USCkpOwoJCQkKCQkJTGFiZWwgbGFiZWwgPSBuZXcgTGFiZWwoY29tcG9zaXRlLCBTV1QuV1JBUCk7CgkJCWxhYmVsLnNldFRleHQoU2VydmVyVUlQbHVnaW4uZ2V0UmVzb3VyY2UoIiVwcmVmUHJvamVjdERlc2NyaXB0aW9uIikpOwoJCQlHcmlkRGF0YSBkYXRhID0gbmV3IEdyaWREYXRhKEdyaWREYXRhLkhPUklaT05UQUxfQUxJR05fRklMTCk7CgkJCWRhdGEuaG9yaXpvbnRhbFNwYW4gPSAzOwoJCQlkYXRhLndpZHRoSGludCA9IDIwMDsKCQkJbGFiZWwuc2V0TGF5b3V0RGF0YShkYXRhKTsKCgkJCW1vZHVsZSA9IFNlcnZlclV0aWwuZ2V0TW9kdWxlUHJvamVjdChwcm9qZWN0KTsKCgkJCWlmIChtb2R1bGUgPT0gbnVsbCkgewoJCQkJbGFiZWwgPSBuZXcgTGFiZWwoY29tcG9zaXRlLCBTV1QuTk9ORSk7CgkJCQlsYWJlbC5zZXRUZXh0KFNlcnZlclVJUGx1Z2luLmdldFJlc291cmNlKCIlcHJlZlByb2plY3ROb3RNb2R1bGUiKSk7CgkJCQlkYXRhID0gbmV3IEdyaWREYXRhKEdyaWREYXRhLkhPUklaT05UQUxfQUxJR05fRklMTCk7CgkJCQlkYXRhLmhvcml6b250YWxTcGFuID0gMzsKCQkJCWxhYmVsLnNldExheW91dERhdGEoZGF0YSk7CgkJCX0gZWxzZSB7CgkJCQlJTW9kdWxlS2luZCBtayA9IFNlcnZlckNvcmUuZ2V0TW9kdWxlS2luZChtb2R1bGUuZ2V0VHlwZSgpKTsKCQkJCWlmIChtayAhPSBudWxsKSB7CgkJCQkJbGFiZWwgPSBuZXcgTGFiZWwoY29tcG9zaXRlLCBTV1QuTk9ORSk7CgkJCQkJbGFiZWwuc2V0VGV4dChTZXJ2ZXJVSVBsdWdpbi5nZXRSZXNvdXJjZSgiJXByZWZQcm9qZWN0IikpOwoJCQkJCWRhdGEgPSBuZXcgR3JpZERhdGEoR3JpZERhdGEuSE9SSVpPTlRBTF9BTElHTl9GSUxMKTsKCQkJCQlsYWJlbC5zZXRMYXlvdXREYXRhKGRhdGEpOwoJCQkJCgkJCQkJTGFiZWwgbW9kdWxlS2luZCA9IG5ldyBMYWJlbChjb21wb3NpdGUsIFNXVC5OT05FKTsKCQkJCQlkYXRhID0gbmV3IEdyaWREYXRhKEdyaWREYXRhLkhPUklaT05UQUxfQUxJR05fRklMTCk7CgkJCQkJZGF0YS5ob3Jpem9udGFsU3BhbiA9IDI7CgkJCQkJbW9kdWxlS2luZC5zZXRMYXlvdXREYXRhKGRhdGEpOwoJCQkJCW1vZHVsZUtpbmQuc2V0VGV4dChtb2R1bGUuZ2V0TmFtZSgpICsgIiAoIiArIG1rLmdldE5hbWUoKSArICIpIik7CgkJCQl9CgkJCQkKCQkJCXJ0Q29tcCA9IG5ldyBSdW50aW1lVGFyZ2V0Q29tcG9zaXRlKGNvbXBvc2l0ZSwgcHJvamVjdCk7CgkJCQkKCQkJCUlQcm9qZWN0UHJvcGVydGllcyBwcmVmcyA9IFNlcnZlckNvcmUuZ2V0UHJvamVjdFByb3BlcnRpZXMocHJvamVjdCk7CgkJCQlJU2VydmVyIHByZWZTZXJ2ZXIgPSBwcmVmcy5nZXREZWZhdWx0U2VydmVyKCk7CgkKCQkJCWxhYmVsID0gbmV3IExhYmVsKGNvbXBvc2l0ZSwgU1dULk5PTkUpOwoJCQkJbGFiZWwuc2V0VGV4dChTZXJ2ZXJVSVBsdWdpbi5nZXRSZXNvdXJjZSgiJXByZWZQcm9qZWN0RGVmYXVsdFNlcnZlciIpKTsKCQkJCWRhdGEgPSBuZXcgR3JpZERhdGEoR3JpZERhdGEuSE9SSVpPTlRBTF9BTElHTl9GSUxMIHwgR3JpZERhdGEuVkVSVElDQUxfQUxJR05fQkVHSU5OSU5HKTsKCQkJCWxhYmVsLnNldExheW91dERhdGEoZGF0YSk7CgkJCQkKCQkJCWZpbmFsIElTZXJ2ZXJbXSBzZXJ2ZXJzID0gU2VydmVyVXRpbC5nZXRTZXJ2ZXJzQnlTdXBwb3J0ZWRNb2R1bGUobW9kdWxlKTsKCQkJCWlmIChzZXJ2ZXJzID09IG51bGwgfHwgc2VydmVycy5sZW5ndGggPT0gMCkgewoJCQkJCWxhYmVsID0gbmV3IExhYmVsKGNvbXBvc2l0ZSwgU1dULldSQVApOwoJCQkJCWxhYmVsLnNldFRleHQoU2VydmVyVUlQbHVnaW4uZ2V0UmVzb3VyY2UoIiVwcmVmUHJvamVjdE5vdENvbmZpZ3VyZWQiKSk7CgkJCQkJZGF0YSA9IG5ldyBHcmlkRGF0YSgpOwoJCQkJCWRhdGEuaG9yaXpvbnRhbFNwYW4gPSAyOwoJCQkJCWxhYmVsLnNldExheW91dERhdGEoZGF0YSk7CgkJCQl9IGVsc2UgewoJCQkJCWZpbmFsIFRhYmxlIHRhYmxlID0gbmV3IFRhYmxlKGNvbXBvc2l0ZSwgU1dULkJPUkRFUiB8IFNXVC5TSU5HTEUgfCBTV1QuVl9TQ1JPTEwpOwoJCQkJCWRhdGEgPSBuZXcgR3JpZERhdGEoR3JpZERhdGEuRklMTF9IT1JJWk9OVEFMKTsKCQkJCQlkYXRhLmhvcml6b250YWxTcGFuID0gMjsKCQkJCQlkYXRhLmhlaWdodEhpbnQgPSA3MDsKCQkJCQl0YWJsZS5zZXRMYXlvdXREYXRhKGRhdGEpOwoJCQkJCQoJCQkJCS8vIGFkZCBub25lIG9wdGlvbgoJCQkJCVRhYmxlSXRlbSBpdGVtID0gbmV3IFRhYmxlSXRlbSh0YWJsZSwgU1dULk5PTkUpOwoJCQkJCWl0ZW0uc2V0VGV4dChTZXJ2ZXJVSVBsdWdpbi5nZXRSZXNvdXJjZSgiJXByZWZQcm9qZWN0Tm9TZXJ2ZXIiKSk7CgkJCQkJLy9pdGVtLnNldEltYWdlKCk7CgkJCQkJCgkJCQkJaW50IHNpemUyID0gc2VydmVycy5sZW5ndGg7CgkJCQkJaW50IGNvdW50ID0gMDsKCQkJCQlmb3IgKGludCBqID0gMDsgaiA8IHNpemUyOyBqKyspIHsKCQkJCQkJaXRlbSA9IG5ldyBUYWJsZUl0ZW0odGFibGUsIFNXVC5OT05FKTsKCQkJCQkJaXRlbS5zZXRUZXh0KFNlcnZlclVJQ29yZS5nZXRMYWJlbFByb3ZpZGVyKCkuZ2V0VGV4dChzZXJ2ZXJzW2pdKSk7CgkJCQkJCWl0ZW0uc2V0SW1hZ2UoU2VydmVyVUlDb3JlLmdldExhYmVsUHJvdmlkZXIoKS5nZXRJbWFnZShzZXJ2ZXJzW2pdKSk7CgkJCQkJCWl0ZW0uc2V0RGF0YShzZXJ2ZXJzW2pdKTsKCQkJCQkJaWYgKHNlcnZlcnNbal0uZXF1YWxzKHByZWZTZXJ2ZXIpKQoJCQkJCQkJY291bnQgPSBqICsgMTsKCQkJCQl9CgoJCQkJCXRhYmxlLnNldFNlbGVjdGlvbihjb3VudCk7CgoJCQkJCXRhYmxlLmFkZFNlbGVjdGlvbkxpc3RlbmVyKG5ldyBTZWxlY3Rpb25BZGFwdGVyKCkgewoJCQkJCQlwdWJsaWMgdm9pZCB3aWRnZXRTZWxlY3RlZChTZWxlY3Rpb25FdmVudCBldmVudCkgewoJCQkJCQkJaW50IGluZGV4ID0gdGFibGUuZ2V0U2VsZWN0aW9uSW5kZXgoKTsKCQkJCQkJCWlmIChpbmRleCA9PSAwKSB7CgkJCQkJCQkJc2VydmVyID0gbnVsbDsKCQkJCQkJCX0gZWxzZSBpZiAoaW5kZXggPiAwKSB7CgkJCQkJCQkJc2VydmVyID0gc2VydmVyc1tpbmRleC0xXTsKCQkJCQkJCX0KCQkJCQkJfQoJCQkJCX0pOwoJCQkJfQoJCQl9CgkJCQoJCQlEaWFsb2cuYXBwbHlEaWFsb2dGb250KGNvbXBvc2l0ZSk7CgoJCQlyZXR1cm4gY29tcG9zaXRlOwoJCX0gY2F0Y2ggKEV4Y2VwdGlvbiBlKSB7CgkJCVRyYWNlLnRyYWNlKCJFcnJvciBjcmVhdGluZyBwcm9qZWN0IHByb3BlcnR5IHBhZ2UiLCBlKTsKCQkJcmV0dXJuIG51bGw7CgkJfQoJfQoKCS8qKiAKCSAqIE1ldGhvZCBkZWNsYXJlZCBvbiBJUHJlZmVyZW5jZVBhZ2UuCgkgKiBTdWJjbGFzc2VzIHNob3VsZCBvdmVycmlkZQoJICovCglwdWJsaWMgYm9vbGVhbiBwZXJmb3JtT2soKSB7CgkJaWYgKG1vZHVsZSAhPSBudWxsKSB7CgkJCXRyeSB7CgkJCQlpZiAocnRDb21wLmhhc0NoYW5nZWQoKSkKCQkJCQlydENvbXAuYXBwbHkobmV3IE51bGxQcm9ncmVzc01vbml0b3IoKSk7CgkJCQkKCQkJCUlQcm9qZWN0UHJvcGVydGllcyBwcm9wcyA9IFNlcnZlckNvcmUuZ2V0UHJvamVjdFByb3BlcnRpZXMocHJvamVjdCk7CgkJCQlwcm9wcy5zZXREZWZhdWx0U2VydmVyKHNlcnZlciwgbmV3IE51bGxQcm9ncmVzc01vbml0b3IoKSk7CgkJCX0gY2F0Y2ggKENvcmVFeGNlcHRpb24gZSkgewoJCQkJVHJhY2UudHJhY2UoVHJhY2UuU0VWRVJFLCAiRXJyb3Igc2V0dGluZyBwcmVmZXJyZWQgc2VydmVyIiwgZSk7CgkJCQlFY2xpcHNlVXRpbC5vcGVuRXJyb3IoU2VydmVyVUlQbHVnaW4uZ2V0UmVzb3VyY2UoIiVlcnJvckNvdWxkTm90U2F2ZVByZWZlcmVuY2UiKSwgZS5nZXRTdGF0dXMoKSk7CgkJCQlyZXR1cm4gZmFsc2U7CgkJCX0KCQl9CgkJcmV0dXJuIHN1cGVyLnBlcmZvcm1PaygpOwoJfQp9