LyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogQ29weXJpZ2h0IChjKSAyMDA1IElCTSBDb3Jwb3JhdGlvbiBhbmQgb3RoZXJzLgogKiBBbGwgcmlnaHRzIHJlc2VydmVkLiBUaGlzIHByb2dyYW0gYW5kIHRoZSBhY2NvbXBhbnlpbmcgbWF0ZXJpYWxzCiAqIGFyZSBtYWRlIGF2YWlsYWJsZSB1bmRlciB0aGUgdGVybXMgb2YgdGhlIEVjbGlwc2UgUHVibGljIExpY2Vuc2UgdjEuMAogKiB3aGljaCBhY2NvbXBhbmllcyB0aGlzIGRpc3RyaWJ1dGlvbiwgYW5kIGlzIGF2YWlsYWJsZSBhdAogKiBodHRwOi8vd3d3LmVjbGlwc2Uub3JnL2xlZ2FsL2VwbC12MTAuaHRtbAqgKiAKICogQ29udHJpYnV0b3JzOgogKiAgICAgSUJNIENvcnBvcmF0aW9uIC0gSW5pdGlhbCBBUEkgYW5kIGltcGxlbWVudGF0aW9uCiAqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwpwYWNrYWdlIG9yZy5lY2xpcHNlLndzdC5zZXJ2ZXIuY29yZS5pbnRlcm5hbDsKCmltcG9ydCBqYXZhLmlvLkZpbGU7CmltcG9ydCBqYXZhLnV0aWwuQXJyYXlMaXN0OwppbXBvcnQgamF2YS51dGlsLkhhc2hNYXA7CmltcG9ydCBqYXZhLnV0aWwuSXRlcmF0b3I7CmltcG9ydCBqYXZhLnV0aWwuTGlzdDsKaW1wb3J0IGphdmEudXRpbC5NYXA7CmltcG9ydCBqYXZhLnV0aWwuU3RyaW5nVG9rZW5pemVyOwoKaW1wb3J0IG9yZy5lY2xpcHNlLmNvcmUucnVudGltZS5Db3JlRXhjZXB0aW9uOwppbXBvcnQgb3JnLmVjbGlwc2UuY29yZS5ydW50aW1lLklQYXRoOwppbXBvcnQgb3JnLmVjbGlwc2Uud3N0LnNlcnZlci5jb3JlLklNb2R1bGU7CmltcG9ydCBvcmcuZWNsaXBzZS53c3Quc2VydmVyLmNvcmUuU2VydmVyVXRpbDsKaW1wb3J0IG9yZy5lY2xpcHNlLndzdC5zZXJ2ZXIuY29yZS5tb2RlbC5JTW9kdWxlRmlsZTsKaW1wb3J0IG9yZy5lY2xpcHNlLndzdC5zZXJ2ZXIuY29yZS5tb2RlbC5JTW9kdWxlRm9sZGVyOwppbXBvcnQgb3JnLmVjbGlwc2Uud3N0LnNlcnZlci5jb3JlLm1vZGVsLklNb2R1bGVSZXNvdXJjZTsKaW1wb3J0IG9yZy5lY2xpcHNlLndzdC5zZXJ2ZXIuY29yZS5tb2RlbC5JTW9kdWxlUmVzb3VyY2VEZWx0YTsKaW1wb3J0IG9yZy5lY2xpcHNlLndzdC5zZXJ2ZXIuY29yZS5tb2RlbC5Nb2R1bGVEZWxlZ2F0ZTsKaW1wb3J0IG9yZy5lY2xpcHNlLndzdC5zZXJ2ZXIuY29yZS5tb2RlbC5TZXJ2ZXJCZWhhdmlvdXJEZWxlZ2F0ZTsKLyoqCiAqIEhlbHBlciB0byBvYnRhaW4gYW5kIHN0b3JlIHRoZSBwdWJsaXNoaW5nIGluZm9ybWF0aW9uICh3aGF0IGZpbGVzCiAqIHdlcmUgcHVibGlzaGVkIGFuZCB3aGVuKSBmb3IgYSBzaW5nbGUgc2VydmVyLgogKi8KcHVibGljIGNsYXNzIFNlcnZlclB1Ymxpc2hJbmZvIHsKCXByb3RlY3RlZCBJUGF0aCBwYXRoOwoKCS8vIG1hcCBvZiBtb2R1bGUgaWRzIHRvIE1vZHVsZVB1Ymxpc2hJbmZvCglwcm90ZWN0ZWQgTWFwIG1vZHVsZVB1Ymxpc2hJbmZvOwoKCS8qKgoJICogU2VydmVyUHVibGlzaEluZm8gY29uc3RydWN0b3IgY29tbWVudC4KCSAqLwoJcHJvdGVjdGVkIFNlcnZlclB1Ymxpc2hJbmZvKElQYXRoIHBhdGgpIHsKCQlzdXBlcigpOwoJCQoJCXRoaXMucGF0aCA9IHBhdGg7CgkJbW9kdWxlUHVibGlzaEluZm8gPSBuZXcgSGFzaE1hcCgpOwoJCWxvYWQoKTsKCX0KCglwcml2YXRlIFN0cmluZyBnZXRLZXkoSU1vZHVsZVtdIG1vZHVsZSkgewoJCVN0cmluZ0J1ZmZlciBzYiA9IG5ldyBTdHJpbmdCdWZmZXIoKTsKCQkKCQlpZiAobW9kdWxlICE9IG51bGwpIHsKCQkJaW50IHNpemUgPSBtb2R1bGUubGVuZ3RoOwoJCQlmb3IgKGludCBpID0gMDsgaSA8IHNpemU7IGkrKykgewoJCQkJaWYgKGkgIT0gMCkKCQkJCQlzYi5hcHBlbmQoIiMiKTsKCQkJCXNiLmFwcGVuZChtb2R1bGVbaV0uZ2V0SWQoKSk7CgkJCX0KCQl9CgkJCgkJcmV0dXJuIHNiLnRvU3RyaW5nKCk7Cgl9CgoJcHJpdmF0ZSBTdHJpbmcgZ2V0S2V5KFN0cmluZyBtb2R1bGVJZCkgewoJCXJldHVybiBtb2R1bGVJZDsKCX0KCQoJcHJpdmF0ZSBJTW9kdWxlW10gZ2V0TW9kdWxlKFN0cmluZyBtb2R1bGVJZCkgewoJCWlmIChtb2R1bGVJZCA9PSBudWxsIHx8IG1vZHVsZUlkLmxlbmd0aCgpID09IDApCgkJCXJldHVybiBuZXcgSU1vZHVsZVswXTsKCQkKCQlMaXN0IGxpc3QgPSBuZXcgQXJyYXlMaXN0KCk7CgkJU3RyaW5nVG9rZW5pemVyIHN0ID0gbmV3IFN0cmluZ1Rva2VuaXplcihtb2R1bGVJZCwgIiMiKTsKCQl3aGlsZSAoc3QuaGFzTW9yZVRva2VucygpKSB7CgkJCVN0cmluZyBtaWQgPSBzdC5uZXh0VG9rZW4oKTsKCQkJaWYgKG1pZCAhPSBudWxsICYmIG1pZC5sZW5ndGgoKSA+IDApIHsKCQkJCUlNb2R1bGUgbSA9IFNlcnZlclV0aWwuZ2V0TW9kdWxlKG1pZCk7CgkJCQlpZiAobSA9PSBudWxsKQoJCQkJCXJldHVybiBudWxsOwoJCQkJbGlzdC5hZGQobSk7CgkJCX0KCQl9CgkJCgkJSU1vZHVsZVtdIG1vZHVsZXMgPSBuZXcgSU1vZHVsZVtsaXN0LnNpemUoKV07CgkJbGlzdC50b0FycmF5KG1vZHVsZXMpOwoJCXJldHVybiBtb2R1bGVzOwoJfQoKCXB1YmxpYyBib29sZWFuIGhhc01vZHVsZVB1Ymxpc2hJbmZvKElNb2R1bGVbXSBtb2R1bGUpIHsKCQlTdHJpbmcga2V5ID0gZ2V0S2V5KG1vZHVsZSk7CgkJcmV0dXJuIG1vZHVsZVB1Ymxpc2hJbmZvLmNvbnRhaW5zS2V5KGtleSk7Cgl9CgoJLypwdWJsaWMgdm9pZCByZW1vdmVNb2R1bGVQdWJsaXNoSW5mbyhJTW9kdWxlW10gbW9kdWxlKSB7CgkJU3RyaW5nIGtleSA9IGdldEtleShtb2R1bGUpOwoJCW1vZHVsZVB1Ymxpc2hJbmZvLnJlbW92ZShrZXkpOwoJCQoJCXNhdmUoKTsKCX0qLwoKCS8qKgoJICogCgkgKiBOb3RlOiBzYXZlKCkgbXVzdCBiZSBjYWxsZWQgbWFudWFsbHkgYWZ0ZXIgbWFraW5nIHRoaXMgY2FsbC4KCSAqIAoJICogQHBhcmFtIG1vZHVsZUxpc3QKCSAqLwoJcHVibGljIHZvaWQgcmVtb3ZlRGVsZXRlZE1vZHVsZVB1Ymxpc2hJbmZvKExpc3QgbW9kdWxlTGlzdCkgewoJCWludCBzaXplID0gbW9kdWxlTGlzdC5zaXplKCk7CgkJTGlzdCByZW1vdmVkID0gbmV3IEFycmF5TGlzdCgpOwoJCQoJCUl0ZXJhdG9yIGl0ZXJhdG9yID0gbW9kdWxlUHVibGlzaEluZm8ua2V5U2V0KCkuaXRlcmF0b3IoKTsKCQl3aGlsZSAoaXRlcmF0b3IuaGFzTmV4dCgpKSB7CgkJCVN0cmluZyBrZXkgPSAoU3RyaW5nKSBpdGVyYXRvci5uZXh0KCk7CgkJCQoJCQlib29sZWFuIGZvdW5kID0gZmFsc2U7CgkJCWZvciAoaW50IGkgPSAwOyBpIDwgc2l6ZTsgaSsrKSB7CgkJCQlJTW9kdWxlW10gbW9kdWxlID0gKElNb2R1bGVbXSkgbW9kdWxlTGlzdC5nZXQoaSk7CgkJCQlTdHJpbmcga2V5MiA9IGdldEtleShtb2R1bGUpOwoJCQkJaWYgKGtleSAhPSBudWxsICYmIGtleS5lcXVhbHMoa2V5MikpCgkJCQkJZm91bmQgPSB0cnVlOwoJCQl9CgkJCWlmICghZm91bmQpCgkJCQlyZW1vdmVkLmFkZChrZXkpOwoJCX0KCQkKCQlpdGVyYXRvciA9IHJlbW92ZWQuaXRlcmF0b3IoKTsKCQl3aGlsZSAoaXRlcmF0b3IuaGFzTmV4dCgpKSB7CgkJCVN0cmluZyBrZXkgPSAoU3RyaW5nKSBpdGVyYXRvci5uZXh0KCk7CgkJCW1vZHVsZVB1Ymxpc2hJbmZvLnJlbW92ZShrZXkpOwoJCX0KCX0KCgkvKioKCSAqIFJldHVybiB0aGUgcHVibGlzaCBzdGF0ZS4KCSAqLwoJcHJvdGVjdGVkIE1vZHVsZVB1Ymxpc2hJbmZvIGdldE1vZHVsZVB1Ymxpc2hJbmZvKElNb2R1bGVbXSBtb2R1bGUpIHsKCQlTdHJpbmcga2V5ID0gZ2V0S2V5KG1vZHVsZSk7CgoJCS8vIGNoZWNrIGlmIGl0IG5vdyBleGlzdHMKCQlpZiAobW9kdWxlUHVibGlzaEluZm8uY29udGFpbnNLZXkoa2V5KSkKCQkJcmV0dXJuIChNb2R1bGVQdWJsaXNoSW5mbykgbW9kdWxlUHVibGlzaEluZm8uZ2V0KGtleSk7CgkKCQkvLyBoYXZlIHRvIGNyZWF0ZSBhIG5ldyBvbmUKCQlJTW9kdWxlIG1vZCA9IG1vZHVsZVttb2R1bGUubGVuZ3RoIC0gMV07CgkJTW9kdWxlUHVibGlzaEluZm8gbXBpID0gbmV3IE1vZHVsZVB1Ymxpc2hJbmZvKGdldEtleShtb2R1bGUpLCBtb2QuZ2V0TmFtZSgpLCBtb2QuZ2V0TW9kdWxlVHlwZSgpKTsKCQltb2R1bGVQdWJsaXNoSW5mby5wdXQoa2V5LCBtcGkpOwoJCXJldHVybiBtcGk7Cgl9CgoJcHVibGljIHZvaWQgYWRkUmVtb3ZlZE1vZHVsZXMoTGlzdCBtb2R1bGVMaXN0LCBMaXN0IGtpbmRMaXN0KSB7CgkJaW50IHNpemUgPSBtb2R1bGVMaXN0LnNpemUoKTsKCQlMaXN0IHJlbW92ZWQgPSBuZXcgQXJyYXlMaXN0KCk7CgkJSXRlcmF0b3IgaXRlcmF0b3IgPSBtb2R1bGVQdWJsaXNoSW5mby5rZXlTZXQoKS5pdGVyYXRvcigpOwoJCXdoaWxlIChpdGVyYXRvci5oYXNOZXh0KCkpIHsKCQkJU3RyaW5nIGtleSA9IChTdHJpbmcpIGl0ZXJhdG9yLm5leHQoKTsKCQkKCQkJYm9vbGVhbiBmb3VuZCA9IGZhbHNlOwoJCQlmb3IgKGludCBpID0gMDsgaSA8IHNpemU7IGkrKykgewoJCQkJSU1vZHVsZVtdIG1vZHVsZSA9IChJTW9kdWxlW10pIG1vZHVsZUxpc3QuZ2V0KGkpOwoJCQkJU3RyaW5nIGtleTIgPSBnZXRLZXkobW9kdWxlKTsKCQkJCWlmIChrZXkgIT0gbnVsbCAmJiBrZXkuZXF1YWxzKGtleTIpKQoJCQkJCWZvdW5kID0gdHJ1ZTsKCQkJfQoJCQlpZiAoIWZvdW5kKSB7CgkJCQlNb2R1bGVQdWJsaXNoSW5mbyBtcGkgPSAoTW9kdWxlUHVibGlzaEluZm8pIG1vZHVsZVB1Ymxpc2hJbmZvLmdldChrZXkpOwoJCQkJcmVtb3ZlZC5hZGQobXBpKTsKCQkJfQoJCX0KCQkKCQlpdGVyYXRvciA9IHJlbW92ZWQuaXRlcmF0b3IoKTsKCQl3aGlsZSAoaXRlcmF0b3IuaGFzTmV4dCgpKSB7CgkJCU1vZHVsZVB1Ymxpc2hJbmZvIG1waSA9IChNb2R1bGVQdWJsaXNoSW5mbykgaXRlcmF0b3IubmV4dCgpOwoJCQlJTW9kdWxlW10gbW9kdWxlMiA9IGdldE1vZHVsZShtcGkuZ2V0TW9kdWxlSWQoKSk7CgkJCWlmIChtb2R1bGUyID09IG51bGwgfHwgbW9kdWxlMi5sZW5ndGggPT0gMCkgewoJCQkJU3RyaW5nIG1vZHVsZUlkID0gbXBpLmdldE1vZHVsZUlkKCk7CgkJCQlpZiAobW9kdWxlSWQgIT0gbnVsbCkgewoJCQkJCWludCBpbmRleCA9IG1vZHVsZUlkLmxhc3RJbmRleE9mKCIjIik7CgkJCQkJbW9kdWxlMiA9IG5ldyBJTW9kdWxlW10geyBuZXcgRGVsZXRlZE1vZHVsZShtb2R1bGVJZC5zdWJzdHJpbmcoaW5kZXggKyAxKSwgbXBpLmdldE5hbWUoKSwgbXBpLmdldE1vZHVsZVR5cGUoKSkgfTsKCQkJCX0KCQkJfQoJCQlpZiAobW9kdWxlMiAhPSBudWxsICYmIG1vZHVsZTIubGVuZ3RoID4gMCkgewoJCQkJbW9kdWxlTGlzdC5hZGQobW9kdWxlMik7CgkJCQlraW5kTGlzdC5hZGQobmV3IEludGVnZXIoU2VydmVyQmVoYXZpb3VyRGVsZWdhdGUuUkVNT1ZFRCkpOwoJCQl9CgkJfQoJfQoKCS8qKgoJICogCgkgKi8KCXB1YmxpYyB2b2lkIGxvYWQoKSB7CgkJU3RyaW5nIGZpbGVuYW1lID0gcGF0aC50b09TU3RyaW5nKCk7CgkJaWYgKCEobmV3IEZpbGUoZmlsZW5hbWUpLmV4aXN0cygpKSkKCQkJcmV0dXJuOwoJCgkJVHJhY2UudHJhY2UoVHJhY2UuRklORVNULCAiTG9hZGluZyBwdWJsaXNoIGluZm8gZnJvbSAiICsgZmlsZW5hbWUpOwoKCQl0cnkgewoJCQlJTWVtZW50byBtZW1lbnRvMiA9IFhNTE1lbWVudG8ubG9hZE1lbWVudG8oZmlsZW5hbWUpOwoJCQlJTWVtZW50b1tdIGNoaWxkcmVuID0gbWVtZW50bzIuZ2V0Q2hpbGRyZW4oIm1vZHVsZSIpOwoJCgkJCWludCBzaXplID0gY2hpbGRyZW4ubGVuZ3RoOwoJCQlmb3IgKGludCBpID0gMDsgaSA8IHNpemU7IGkrKykgewoJCQkJTW9kdWxlUHVibGlzaEluZm8gbXBpID0gbmV3IE1vZHVsZVB1Ymxpc2hJbmZvKGNoaWxkcmVuW2ldKTsKCQkJCW1vZHVsZVB1Ymxpc2hJbmZvLnB1dChnZXRLZXkobXBpLmdldE1vZHVsZUlkKCkpLCBtcGkpOwoJCQl9CgkJfSBjYXRjaCAoRXhjZXB0aW9uIGUpIHsKCQkJVHJhY2UudHJhY2UoVHJhY2UuV0FSTklORywgIkNvdWxkIG5vdCBsb2FkIHB1Ymxpc2ggaW5mb3JtYXRpb246ICIgKyBlLmdldE1lc3NhZ2UoKSk7CgkJfQoJfQoKCS8qKgoJICogCgkgKi8KCXB1YmxpYyB2b2lkIHNhdmUoKSB7CgkJU3RyaW5nIGZpbGVuYW1lID0gcGF0aC50b09TU3RyaW5nKCk7CgkJVHJhY2UudHJhY2UoVHJhY2UuRklORVNULCAiU2F2aW5nIHB1Ymxpc2ggaW5mbyB0byAiICsgZmlsZW5hbWUpOwoJCgkJdHJ5IHsKCQkJWE1MTWVtZW50byBtZW1lbnRvID0gWE1MTWVtZW50by5jcmVhdGVXcml0ZVJvb3QoInNlcnZlciIpOwoKCQkJSXRlcmF0b3IgaXRlcmF0b3IgPSBtb2R1bGVQdWJsaXNoSW5mby5rZXlTZXQoKS5pdGVyYXRvcigpOwoJCQl3aGlsZSAoaXRlcmF0b3IuaGFzTmV4dCgpKSB7CgkJCQlTdHJpbmcgY29udHJvbFJlZiA9IChTdHJpbmcpIGl0ZXJhdG9yLm5leHQoKTsKCQkJCU1vZHVsZVB1Ymxpc2hJbmZvIG1waSA9IChNb2R1bGVQdWJsaXNoSW5mbykgbW9kdWxlUHVibGlzaEluZm8uZ2V0KGNvbnRyb2xSZWYpOwoJCQkJSU1lbWVudG8gY2hpbGQgPSBtZW1lbnRvLmNyZWF0ZUNoaWxkKCJtb2R1bGUiKTsKCQkJCW1waS5zYXZlKGNoaWxkKTsKCQkJfQoJCQltZW1lbnRvLnNhdmVUb0ZpbGUoZmlsZW5hbWUpOwoJCX0gY2F0Y2ggKEV4Y2VwdGlvbiBlKSB7CgkJCVRyYWNlLnRyYWNlKFRyYWNlLlNFVkVSRSwgIkNvdWxkIG5vdCBzYXZlIHB1Ymxpc2ggaW5mb3JtYXRpb24iLCBlKTsKCQl9Cgl9CgoJLyoqCgkgKiAKCSAqIE5vdGU6IHNhdmUoKSBtdXN0IGJlIGNhbGxlZCBtYW51YWxseSBhZnRlciBtYWtpbmcgdGhpcyBjYWxsLgoJICogQHBhcmFtIG1vZHVsZQoJICovCglwdWJsaWMgdm9pZCBmaWxsKElNb2R1bGVbXSBtb2R1bGUpIHsKCQlNb2R1bGVQdWJsaXNoSW5mbyBtcGkgPSBnZXRNb2R1bGVQdWJsaXNoSW5mbyhtb2R1bGUpOwoJCW1waS5maWxsKG1vZHVsZSk7Cgl9CgoJcHJvdGVjdGVkIElNb2R1bGVSZXNvdXJjZURlbHRhW10gZ2V0RGVsdGEoSU1vZHVsZVtdIG1vZHVsZSkgewoJCWlmIChtb2R1bGUgPT0gbnVsbCkKCQkJcmV0dXJuIG5ldyBJTW9kdWxlUmVzb3VyY2VEZWx0YVswXTsKCQkKCQlyZXR1cm4gZ2V0TW9kdWxlUHVibGlzaEluZm8obW9kdWxlKS5nZXREZWx0YShtb2R1bGUpOwoJfQoKCXByb3RlY3RlZCBJTW9kdWxlUmVzb3VyY2VbXSBnZXRSZXNvdXJjZXMoSU1vZHVsZVtdIG1vZHVsZSkgewoJCWlmIChtb2R1bGUgPT0gbnVsbCkKCQkJcmV0dXJuIG5ldyBJTW9kdWxlUmVzb3VyY2VbMF07CgkJCgkJcmV0dXJuIGdldE1vZHVsZVB1Ymxpc2hJbmZvKG1vZHVsZSkuZ2V0TW9kdWxlUmVzb3VyY2VzKG1vZHVsZSk7Cgl9CgoJcHJvdGVjdGVkIHN0YXRpYyBJTW9kdWxlUmVzb3VyY2VEZWx0YVtdIGdldERlbHRhKElNb2R1bGVSZXNvdXJjZVtdIG9yaWdpbmFsLCBJTW9kdWxlUmVzb3VyY2VbXSBjdXJyZW50KSB7CgkJaWYgKG9yaWdpbmFsID09IG51bGwgfHwgY3VycmVudCA9PSBudWxsKQoJCQlyZXR1cm4gbmV3IElNb2R1bGVSZXNvdXJjZURlbHRhWzBdOwoJCgkJTGlzdCBsaXN0ID0gbmV3IEFycmF5TGlzdCgpOwoJCWludCBzaXplID0gb3JpZ2luYWwubGVuZ3RoOwoJCWludCBzaXplMiA9IGN1cnJlbnQubGVuZ3RoOwoJCQoJCU1hcCBvcmlnaW5hbE1hcCA9IG5ldyBIYXNoTWFwKHNpemUpOwoJCWZvciAoaW50IGkgPSAwOyBpIDwgc2l6ZTsgaSsrKQoJCQlvcmlnaW5hbE1hcC5wdXQob3JpZ2luYWxbaV0sIG9yaWdpbmFsW2ldKTsKCgkJLy8gYWRkZWQgYW5kIGNoYW5nZWQgcmVzb3VyY2VzCgkJZm9yIChpbnQgaSA9IDA7IGkgPCBzaXplMjsgaSsrKSB7CgkJCUlNb2R1bGVSZXNvdXJjZSBvbGQgPSAoSU1vZHVsZVJlc291cmNlKSBvcmlnaW5hbE1hcC5yZW1vdmUoY3VycmVudFtpXSk7CgkJCWlmIChvbGQgPT0gbnVsbCkgewoJCQkJTW9kdWxlUmVzb3VyY2VEZWx0YSBkZWx0YSA9IG5ldyBNb2R1bGVSZXNvdXJjZURlbHRhKGN1cnJlbnRbaV0sIElNb2R1bGVSZXNvdXJjZURlbHRhLkFEREVEKTsKCQkJCWlmIChjdXJyZW50W2ldIGluc3RhbmNlb2YgSU1vZHVsZUZvbGRlcikgewoJCQkJCUlNb2R1bGVGb2xkZXIgY3VycmVudEZvbGRlciA9IChJTW9kdWxlRm9sZGVyKSBjdXJyZW50W2ldOyAKCQkJCQlkZWx0YS5zZXRDaGlsZHJlbihnZXREZWx0YVRyZWUoY3VycmVudEZvbGRlci5tZW1iZXJzKCksIElNb2R1bGVSZXNvdXJjZURlbHRhLkFEREVEKSk7CgkJCQl9CgkJCQlsaXN0LmFkZChkZWx0YSk7CgkJCX0gZWxzZSB7CgkJCQlpZiAoY3VycmVudFtpXSBpbnN0YW5jZW9mIElNb2R1bGVGaWxlKSB7CgkJCQkJLy8gaW5jbHVkZSBmaWxlcyBvbmx5IGlmIHRoZSBtb2RpZmljYXRpb24gc3RhbXAgaGFzIGNoYW5nZWQKCQkJCQlJTW9kdWxlRmlsZSBtZjEgPSAoSU1vZHVsZUZpbGUpIG9sZDsKCQkJCQlJTW9kdWxlRmlsZSBtZjIgPSAoSU1vZHVsZUZpbGUpIGN1cnJlbnRbaV07CgkJCQkJaWYgKG1mMS5nZXRNb2RpZmljYXRpb25TdGFtcCgpICE9IG1mMi5nZXRNb2RpZmljYXRpb25TdGFtcCgpKSB7CgkJCQkJCWxpc3QuYWRkKG5ldyBNb2R1bGVSZXNvdXJjZURlbHRhKGN1cnJlbnRbaV0sIElNb2R1bGVSZXNvdXJjZURlbHRhLkNIQU5HRUQpKTsKCQkJCQl9CgkJCQl9IGVsc2UgewoJCQkJCS8vIGluY2x1ZGUgZm9sZGVycyBvbmx5IGlmIHRoZWlyIGNvbnRlbnRzIGhhdmUgY2hhbmdlZAoJCQkJCUlNb2R1bGVGb2xkZXIgbWYxID0gKElNb2R1bGVGb2xkZXIpIG9sZDsKCQkJCQlJTW9kdWxlRm9sZGVyIG1mMiA9IChJTW9kdWxlRm9sZGVyKSBjdXJyZW50W2ldOwoJCQkJCUlNb2R1bGVSZXNvdXJjZURlbHRhW10gbXJkYyA9IGdldERlbHRhKG1mMS5tZW1iZXJzKCksIG1mMi5tZW1iZXJzKCkpOwoJCQkJCWlmIChtcmRjLmxlbmd0aCA+IDApIHsKCQkJCQkJTW9kdWxlUmVzb3VyY2VEZWx0YSBtcmQgPSBuZXcgTW9kdWxlUmVzb3VyY2VEZWx0YShjdXJyZW50W2ldLCBJTW9kdWxlUmVzb3VyY2VEZWx0YS5OT19DSEFOR0UpOwoJCQkJCQltcmQuc2V0Q2hpbGRyZW4obXJkYyk7CgkJCQkJCWxpc3QuYWRkKG1yZCk7CgkJCQkJfQoJCQkJfQoJCQl9CgkJfQoJCQoJCS8vIHJlbW92ZWQgcmVzb3VyY2VzCgkJZm9yIChpbnQgaSA9IDA7IGkgPCBzaXplOyBpKyspIHsKCQkJaWYgKG9yaWdpbmFsTWFwLmNvbnRhaW5zS2V5KG9yaWdpbmFsW2ldKSkgewoJCQkJTW9kdWxlUmVzb3VyY2VEZWx0YSBkZWx0YSA9IG5ldyBNb2R1bGVSZXNvdXJjZURlbHRhKG9yaWdpbmFsW2ldLCBJTW9kdWxlUmVzb3VyY2VEZWx0YS5SRU1PVkVEKTsKCQkJCWlmIChvcmlnaW5hbFtpXSBpbnN0YW5jZW9mIElNb2R1bGVGb2xkZXIpIHsKCQkJCQlJTW9kdWxlRm9sZGVyIHJlbW92ZWRGb2xkZXIgPSAoSU1vZHVsZUZvbGRlcikgb3JpZ2luYWxbaV07IAoJCQkJCWRlbHRhLnNldENoaWxkcmVuKGdldERlbHRhVHJlZShyZW1vdmVkRm9sZGVyLm1lbWJlcnMoKSwgSU1vZHVsZVJlc291cmNlRGVsdGEuUkVNT1ZFRCkpOwoJCQkJfQoJCQkJbGlzdC5hZGQoZGVsdGEpOwoJCQl9CgkJfQoJCQoJCXJldHVybiAoSU1vZHVsZVJlc291cmNlRGVsdGFbXSkgbGlzdC50b0FycmF5KG5ldyBJTW9kdWxlUmVzb3VyY2VEZWx0YVtsaXN0LnNpemUoKV0pOwoJfQoKCXByb3RlY3RlZCBib29sZWFuIGhhc0RlbHRhKElNb2R1bGVbXSBtb2R1bGUpIHsKCQlpZiAobW9kdWxlID09IG51bGwpCgkJCXJldHVybiBmYWxzZTsKCQkKCQlyZXR1cm4gZ2V0TW9kdWxlUHVibGlzaEluZm8obW9kdWxlKS5oYXNEZWx0YShtb2R1bGUpOwoJfQoKCXByb3RlY3RlZCBzdGF0aWMgYm9vbGVhbiBoYXNEZWx0YShJTW9kdWxlUmVzb3VyY2VbXSBvcmlnaW5hbCwgSU1vZHVsZVJlc291cmNlW10gY3VycmVudCkgewoJCWlmIChvcmlnaW5hbCA9PSBudWxsIHx8IGN1cnJlbnQgPT0gbnVsbCkKCQkJcmV0dXJuIGZhbHNlOwoJCQoJCWludCBzaXplID0gb3JpZ2luYWwubGVuZ3RoOwoJCWludCBzaXplMiA9IGN1cnJlbnQubGVuZ3RoOwoJCQoJCU1hcCBvcmlnaW5hbE1hcCA9IG5ldyBIYXNoTWFwKHNpemUpOwoJCWZvciAoaW50IGkgPSAwOyBpIDwgc2l6ZTsgaSsrKQoJCQlvcmlnaW5hbE1hcC5wdXQob3JpZ2luYWxbaV0sIG9yaWdpbmFsW2ldKTsKCQkKCQkvLyBhZGRlZCBhbmQgY2hhbmdlZCByZXNvdXJjZXMKCQlmb3IgKGludCBpID0gMDsgaSA8IHNpemUyOyBpKyspIHsKCQkJSU1vZHVsZVJlc291cmNlIG9sZCA9IChJTW9kdWxlUmVzb3VyY2UpIG9yaWdpbmFsTWFwLnJlbW92ZShjdXJyZW50W2ldKTsKCQkJaWYgKG9sZCA9PSBudWxsKQoJCQkJcmV0dXJuIHRydWU7CgkJCQoJCQlpZiAoY3VycmVudFtpXSBpbnN0YW5jZW9mIElNb2R1bGVGaWxlKSB7CgkJCQkvLyBpbmNsdWRlIGZpbGVzIG9ubHkgaWYgdGhlIG1vZGlmaWNhdGlvbiBzdGFtcCBoYXMgY2hhbmdlZAoJCQkJSU1vZHVsZUZpbGUgbWYxID0gKElNb2R1bGVGaWxlKSBvbGQ7CgkJCQlJTW9kdWxlRmlsZSBtZjIgPSAoSU1vZHVsZUZpbGUpIGN1cnJlbnRbaV07CgkJCQlpZiAobWYxLmdldE1vZGlmaWNhdGlvblN0YW1wKCkgIT0gbWYyLmdldE1vZGlmaWNhdGlvblN0YW1wKCkpCgkJCQkJcmV0dXJuIHRydWU7CgkJCX0gZWxzZSB7CgkJCQkvLyBpbmNsdWRlIGZvbGRlcnMgb25seSBpZiB0aGVpciBjb250ZW50cyBoYXZlIGNoYW5nZWQKCQkJCUlNb2R1bGVGb2xkZXIgbWYxID0gKElNb2R1bGVGb2xkZXIpIG9sZDsKCQkJCUlNb2R1bGVGb2xkZXIgbWYyID0gKElNb2R1bGVGb2xkZXIpIGN1cnJlbnRbaV07CgkJCQlpZiAoaGFzRGVsdGEobWYxLm1lbWJlcnMoKSwgbWYyLm1lbWJlcnMoKSkpCgkJCQkJcmV0dXJuIHRydWU7CgkJCX0KCQl9CgkJCgkJLy8gcmVtb3ZlZCByZXNvdXJjZXMKCQlyZXR1cm4gIW9yaWdpbmFsTWFwLmlzRW1wdHkoKTsKCX0KCgkvKioKCSAqIENyZWF0ZSBhIHJlc291cmNlIGRlbHRhIGZvciBhbiBlbnRpcmUgdHJlZS4KCSAqLwoJcHJpdmF0ZSBzdGF0aWMgSU1vZHVsZVJlc291cmNlRGVsdGFbXSBnZXREZWx0YVRyZWUoSU1vZHVsZVJlc291cmNlW10gcmVzb3VyY2VzLCBpbnQga2luZCkgewoJCWlmIChyZXNvdXJjZXMgPT0gbnVsbCkKCQkJcmV0dXJuIG5ldyBJTW9kdWxlUmVzb3VyY2VEZWx0YVswXTsKCQoJCUxpc3QgbGlzdCA9IG5ldyBBcnJheUxpc3QoKTsKCQkKCQkvLyBsb29rIGZvciBkdXBsaWNhdGVzCgkJaW50IHNpemUgPSByZXNvdXJjZXMubGVuZ3RoOwoJCWZvciAoaW50IGkgPSAwOyBpIDwgc2l6ZTsgaSsrKSB7CgkJCU1vZHVsZVJlc291cmNlRGVsdGEgbXJkID0gbmV3IE1vZHVsZVJlc291cmNlRGVsdGEocmVzb3VyY2VzW2ldLCBraW5kKTsKCQkJaWYgKHJlc291cmNlc1tpXSBpbnN0YW5jZW9mIElNb2R1bGVGb2xkZXIpIHsKCQkJCUlNb2R1bGVGb2xkZXIgbWYgPSAoSU1vZHVsZUZvbGRlcikgcmVzb3VyY2VzW2ldOwoJCQkJbXJkLnNldENoaWxkcmVuKGdldERlbHRhVHJlZShtZi5tZW1iZXJzKCksIGtpbmQpKTsKCQkJfQoJCQlsaXN0LmFkZChtcmQpOwoJCX0KCQkKCQlJTW9kdWxlUmVzb3VyY2VEZWx0YVtdIGRlbHRhID0gbmV3IElNb2R1bGVSZXNvdXJjZURlbHRhW2xpc3Quc2l6ZSgpXTsKCQlsaXN0LnRvQXJyYXkoZGVsdGEpOwoJCXJldHVybiBkZWx0YTsKCX0KCgkvKioKCSAqIFJldHVybnMgdHJ1ZSBpZiB0aGUgbGlzdCBvZiBtb2R1bGVzIGJlaW5nIHB1Ymxpc2hlZCBkb2VzIG5vdCBtYXRjaCB0aGUgcHJldmlvdXMKCSAqIGxpc3Qgb2YgcHVibGlzaGVkIG1vZHVsZXMuCgkgKiAKCSAqIFRPRE86IFRoaXMgbWV0aG9kIHNob3VsZCBjb21wYXJlIHRoZSBtb2R1bGVzLiBGb3Igbm93LCBjb21wYXJpbmcgdGhlIHNpemUgaXMgZmluZS4KCSAqIAoJICogQHBhcmFtIG1vZHVsZXMgYSBsaXN0IG9mIG1vZHVsZXMKCSAqIEByZXR1cm4gPGNvZGU+dHJ1ZTwvY29kZT4gaWYgdGhlIHN0cnVjdHVyZSBvZiBwdWJsaXNoZWQgbW9kdWxlcyBoYXMgY2hhbmdlZCwgb3IKCSAqICAgIDxjb2RlPmZhbHNlPC9jb2RlPiBvdGhlcndpc2UKCSAqLwoJcHJvdGVjdGVkIGJvb2xlYW4gaGFzU3RydWN0dXJlQ2hhbmdlZChMaXN0IG1vZHVsZXMpIHsKCQlyZXR1cm4gbW9kdWxlcy5zaXplKCkgIT0gbW9kdWxlUHVibGlzaEluZm8ua2V5U2V0KCkuc2l6ZSgpOwoJfQoKCS8qKgoJICogRmlsbCB0aGUgbW9kdWxlIGNhY2hlLgoJICovCglwcm90ZWN0ZWQgdm9pZCBzdGFydENhY2hpbmcoKSB7CgkJSXRlcmF0b3IgaXRlcmF0b3IgPSBtb2R1bGVQdWJsaXNoSW5mby52YWx1ZXMoKS5pdGVyYXRvcigpOwoJCXdoaWxlIChpdGVyYXRvci5oYXNOZXh0KCkpIHsKCQkJTW9kdWxlUHVibGlzaEluZm8gbXBpID0gKE1vZHVsZVB1Ymxpc2hJbmZvKSBpdGVyYXRvci5uZXh0KCk7CgkJCW1waS5zdGFydENhY2hpbmcoKTsKCQl9Cgl9CgoJLyoqCgkgKiBDbGVhcnMgYWxsIGNhY2hlcyBvZiBjdXJyZW50IG1vZHVsZSByZXNvdXJjZXMgYW5kIGRlbHRhcy4KCSAqLwoJcHVibGljIHZvaWQgY2xlYXJDYWNoZSgpIHsKCQlJdGVyYXRvciBpdGVyYXRvciA9IG1vZHVsZVB1Ymxpc2hJbmZvLnZhbHVlcygpLml0ZXJhdG9yKCk7CgkJd2hpbGUgKGl0ZXJhdG9yLmhhc05leHQoKSkgewoJCQlNb2R1bGVQdWJsaXNoSW5mbyBtcGkgPSAoTW9kdWxlUHVibGlzaEluZm8pIGl0ZXJhdG9yLm5leHQoKTsKCQkJbXBpLmNsZWFyQ2FjaGUoKTsKCQl9Cgl9Cn0=