LyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogQ29weXJpZ2h0IChjKSAyMDA1IElCTSBDb3Jwb3JhdGlvbiBhbmQgb3RoZXJzLgogKiBBbGwgcmlnaHRzIHJlc2VydmVkLiBUaGlzIHByb2dyYW0gYW5kIHRoZSBhY2NvbXBhbnlpbmcgbWF0ZXJpYWxzCiAqIGFyZSBtYWRlIGF2YWlsYWJsZSB1bmRlciB0aGUgdGVybXMgb2YgdGhlIEVjbGlwc2UgUHVibGljIExpY2Vuc2UgdjEuMAogKiB3aGljaCBhY2NvbXBhbmllcyB0aGlzIGRpc3RyaWJ1dGlvbiwgYW5kIGlzIGF2YWlsYWJsZSBhdAogKiBodHRwOi8vd3d3LmVjbGlwc2Uub3JnL2xlZ2FsL2VwbC12MTAuaHRtbAqgKiAKICogQ29udHJpYnV0b3JzOgogKiAgICAgSUJNIENvcnBvcmF0aW9uIC0gSW5pdGlhbCBBUEkgYW5kIGltcGxlbWVudGF0aW9uCiAqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwpwYWNrYWdlIG9yZy5lY2xpcHNlLndzdC5zZXJ2ZXIuY29yZS5pbnRlcm5hbDsKCmltcG9ydCBqYXZhLmlvLkZpbGU7CmltcG9ydCBqYXZhLnV0aWwuKjsKCmltcG9ydCBvcmcuZWNsaXBzZS5jb3JlLnJlc291cmNlcy5JRmlsZTsKaW1wb3J0IG9yZy5lY2xpcHNlLmNvcmUucmVzb3VyY2VzLklQcm9qZWN0OwppbXBvcnQgb3JnLmVjbGlwc2UuY29yZS5yZXNvdXJjZXMuUmVzb3VyY2VzUGx1Z2luOwppbXBvcnQgb3JnLmVjbGlwc2UuY29yZS5ydW50aW1lLkNvcmVFeGNlcHRpb247CmltcG9ydCBvcmcuZWNsaXBzZS5jb3JlLnJ1bnRpbWUuSVBhdGg7CmltcG9ydCBvcmcuZWNsaXBzZS53c3Quc2VydmVyLmNvcmUuSU1vZHVsZTsKaW1wb3J0IG9yZy5lY2xpcHNlLndzdC5zZXJ2ZXIuY29yZS5TZXJ2ZXJVdGlsOwppbXBvcnQgb3JnLmVjbGlwc2Uud3N0LnNlcnZlci5jb3JlLm1vZGVsLio7Ci8qKgogKiBIZWxwZXIgdG8gb2J0YWluIGFuZCBzdG9yZSB0aGUgcHVibGlzaGluZyBpbmZvcm1hdGlvbiAod2hhdCBmaWxlcwogKiB3ZXJlIHB1Ymxpc2hlZCBhbmQgd2hlbikgZm9yIGEgc2luZ2xlIHNlcnZlci4KICovCnB1YmxpYyBjbGFzcyBTZXJ2ZXJQdWJsaXNoSW5mbyB7Cglwcm90ZWN0ZWQgSVBhdGggcGF0aDsKCgkvLyBtYXAgb2YgbW9kdWxlIGlkcyB0byBNb2R1bGVQdWJsaXNoSW5mbwoJcHJvdGVjdGVkIE1hcCBtb2R1bGVQdWJsaXNoSW5mbzsKCgkvKioKCSAqIFNlcnZlclB1Ymxpc2hJbmZvIGNvbnN0cnVjdG9yIGNvbW1lbnQuCgkgKi8KCXByb3RlY3RlZCBTZXJ2ZXJQdWJsaXNoSW5mbyhJUGF0aCBwYXRoKSB7CgkJc3VwZXIoKTsKCQkKCQl0aGlzLnBhdGggPSBwYXRoOwoJCW1vZHVsZVB1Ymxpc2hJbmZvID0gbmV3IEhhc2hNYXAoKTsKCQlsb2FkKCk7Cgl9CgoJcHJpdmF0ZSBTdHJpbmcgZ2V0S2V5KElNb2R1bGVbXSBtb2R1bGUpIHsKCQlTdHJpbmdCdWZmZXIgc2IgPSBuZXcgU3RyaW5nQnVmZmVyKCk7CgkJCgkJaWYgKG1vZHVsZSAhPSBudWxsKSB7CgkJCWludCBzaXplID0gbW9kdWxlLmxlbmd0aDsKCQkJZm9yIChpbnQgaSA9IDA7IGkgPCBzaXplOyBpKyspIHsKCQkJCWlmIChpICE9IDApCgkJCQkJc2IuYXBwZW5kKCIjIik7CgkJCQlzYi5hcHBlbmQobW9kdWxlW2ldLmdldElkKCkpOwoJCQl9CgkJfQoJCQoJCXJldHVybiBzYi50b1N0cmluZygpOwoJfQoKCXByaXZhdGUgU3RyaW5nIGdldEtleShTdHJpbmcgbW9kdWxlSWQpIHsKCQlyZXR1cm4gbW9kdWxlSWQ7Cgl9CgkKCXByaXZhdGUgSU1vZHVsZVtdIGdldE1vZHVsZShTdHJpbmcgbW9kdWxlSWQpIHsKCQlpZiAobW9kdWxlSWQgPT0gbnVsbCB8fCBtb2R1bGVJZC5sZW5ndGgoKSA9PSAwKQoJCQlyZXR1cm4gbmV3IElNb2R1bGVbMF07CgkJCgkJTGlzdCBsaXN0ID0gbmV3IEFycmF5TGlzdCgpOwoJCVN0cmluZ1Rva2VuaXplciBzdCA9IG5ldyBTdHJpbmdUb2tlbml6ZXIobW9kdWxlSWQsICIjIik7CgkJd2hpbGUgKHN0Lmhhc01vcmVUb2tlbnMoKSkgewoJCQlTdHJpbmcgbWlkID0gc3QubmV4dFRva2VuKCk7CgkJCWlmIChtaWQgIT0gbnVsbCAmJiBtaWQubGVuZ3RoKCkgPiAwKSB7CgkJCQlJTW9kdWxlIG0gPSBTZXJ2ZXJVdGlsLmdldE1vZHVsZShtaWQpOwoJCQkJaWYgKG0gPT0gbnVsbCkKCQkJCQlyZXR1cm4gbnVsbDsKCQkJCWxpc3QuYWRkKG0pOwoJCQl9CgkJfQoJCQoJCUlNb2R1bGVbXSBtb2R1bGVzID0gbmV3IElNb2R1bGVbbGlzdC5zaXplKCldOwoJCWxpc3QudG9BcnJheShtb2R1bGVzKTsKCQlyZXR1cm4gbW9kdWxlczsKCX0KCglwdWJsaWMgYm9vbGVhbiBoYXNNb2R1bGVQdWJsaXNoSW5mbyhJTW9kdWxlW10gbW9kdWxlKSB7CgkJU3RyaW5nIGtleSA9IGdldEtleShtb2R1bGUpOwoJCXJldHVybiBtb2R1bGVQdWJsaXNoSW5mby5jb250YWluc0tleShrZXkpOwoJfQoKCS8qcHVibGljIHZvaWQgcmVtb3ZlTW9kdWxlUHVibGlzaEluZm8oSU1vZHVsZVtdIG1vZHVsZSkgewoJCVN0cmluZyBrZXkgPSBnZXRLZXkobW9kdWxlKTsKCQltb2R1bGVQdWJsaXNoSW5mby5yZW1vdmUoa2V5KTsKCQkKCQlzYXZlKCk7Cgl9Ki8KCglwdWJsaWMgdm9pZCByZW1vdmVEZWxldGVkTW9kdWxlUHVibGlzaEluZm8oTGlzdCBtb2R1bGVMaXN0KSB7CgkJaW50IHNpemUgPSBtb2R1bGVMaXN0LnNpemUoKTsKCQlMaXN0IHJlbW92ZWQgPSBuZXcgQXJyYXlMaXN0KCk7CgkJCgkJSXRlcmF0b3IgaXRlcmF0b3IgPSBtb2R1bGVQdWJsaXNoSW5mby5rZXlTZXQoKS5pdGVyYXRvcigpOwoJCXdoaWxlIChpdGVyYXRvci5oYXNOZXh0KCkpIHsKCQkJU3RyaW5nIGtleSA9IChTdHJpbmcpIGl0ZXJhdG9yLm5leHQoKTsKCQkJCgkJCWJvb2xlYW4gZm91bmQgPSBmYWxzZTsKCQkJZm9yIChpbnQgaSA9IDA7IGkgPCBzaXplOyBpKyspIHsKCQkJCUlNb2R1bGVbXSBtb2R1bGUgPSAoSU1vZHVsZVtdKSBtb2R1bGVMaXN0LmdldChpKTsKCQkJCVN0cmluZyBrZXkyID0gZ2V0S2V5KG1vZHVsZSk7CgkJCQlpZiAoa2V5ICE9IG51bGwgJiYga2V5LmVxdWFscyhrZXkyKSkKCQkJCQlmb3VuZCA9IHRydWU7CgkJCX0KCQkJaWYgKCFmb3VuZCkKCQkJCXJlbW92ZWQuYWRkKGtleSk7CgkJfQoJCQoJCWl0ZXJhdG9yID0gcmVtb3ZlZC5pdGVyYXRvcigpOwoJCXdoaWxlIChpdGVyYXRvci5oYXNOZXh0KCkpIHsKCQkJU3RyaW5nIGtleSA9IChTdHJpbmcpIGl0ZXJhdG9yLm5leHQoKTsKCQkJbW9kdWxlUHVibGlzaEluZm8ucmVtb3ZlKGtleSk7CgkJfQoJCQoJCXNhdmUoKTsKCX0KCgkvKioKCSAqIFJldHVybiB0aGUgcHVibGlzaCBzdGF0ZS4KCSAqLwoJcHJvdGVjdGVkIE1vZHVsZVB1Ymxpc2hJbmZvIGdldE1vZHVsZVB1Ymxpc2hJbmZvKElNb2R1bGVbXSBtb2R1bGUpIHsKCQlTdHJpbmcga2V5ID0gZ2V0S2V5KG1vZHVsZSk7CgoJCS8vIGNoZWNrIGlmIGl0IG5vdyBleGlzdHMKCQlpZiAobW9kdWxlUHVibGlzaEluZm8uY29udGFpbnNLZXkoa2V5KSkKCQkJcmV0dXJuIChNb2R1bGVQdWJsaXNoSW5mbykgbW9kdWxlUHVibGlzaEluZm8uZ2V0KGtleSk7CgkKCQkvLyBoYXZlIHRvIGNyZWF0ZSBhIG5ldyBvbmUKCQlNb2R1bGVQdWJsaXNoSW5mbyBtcGkgPSBuZXcgTW9kdWxlUHVibGlzaEluZm8oZ2V0S2V5KG1vZHVsZSksIG1vZHVsZVttb2R1bGUubGVuZ3RoIC0gMV0uZ2V0TmFtZSgpKTsKCQltb2R1bGVQdWJsaXNoSW5mby5wdXQoa2V5LCBtcGkpOwoJCXJldHVybiBtcGk7Cgl9CgoJcHVibGljIHZvaWQgYWRkUmVtb3ZlZE1vZHVsZXMoTGlzdCBtb2R1bGVMaXN0LCBMaXN0IGtpbmRMaXN0KSB7CgkJaW50IHNpemUgPSBtb2R1bGVMaXN0LnNpemUoKTsKCQlMaXN0IHJlbW92ZWQgPSBuZXcgQXJyYXlMaXN0KCk7CgkJSXRlcmF0b3IgaXRlcmF0b3IgPSBtb2R1bGVQdWJsaXNoSW5mby5rZXlTZXQoKS5pdGVyYXRvcigpOwoJCXdoaWxlIChpdGVyYXRvci5oYXNOZXh0KCkpIHsKCQkJU3RyaW5nIGtleSA9IChTdHJpbmcpIGl0ZXJhdG9yLm5leHQoKTsKCQkKCQkJYm9vbGVhbiBmb3VuZCA9IGZhbHNlOwoJCQlmb3IgKGludCBpID0gMDsgaSA8IHNpemU7IGkrKykgewoJCQkJSU1vZHVsZVtdIG1vZHVsZSA9IChJTW9kdWxlW10pIG1vZHVsZUxpc3QuZ2V0KGkpOwoJCQkJU3RyaW5nIGtleTIgPSBnZXRLZXkobW9kdWxlKTsKCQkJCWlmIChrZXkgIT0gbnVsbCAmJiBrZXkuZXF1YWxzKGtleTIpKQoJCQkJCWZvdW5kID0gdHJ1ZTsKCQkJfQoJCQlpZiAoIWZvdW5kKSB7CgkJCQlNb2R1bGVQdWJsaXNoSW5mbyBtcGkgPSAoTW9kdWxlUHVibGlzaEluZm8pIG1vZHVsZVB1Ymxpc2hJbmZvLmdldChrZXkpOwoJCQkJcmVtb3ZlZC5hZGQobXBpKTsKCQkJfQoJCX0KCQkKCQlpdGVyYXRvciA9IHJlbW92ZWQuaXRlcmF0b3IoKTsKCQl3aGlsZSAoaXRlcmF0b3IuaGFzTmV4dCgpKSB7CgkJCU1vZHVsZVB1Ymxpc2hJbmZvIG1waSA9IChNb2R1bGVQdWJsaXNoSW5mbykgaXRlcmF0b3IubmV4dCgpOwoJCQlJTW9kdWxlW10gbW9kdWxlMiA9IGdldE1vZHVsZShtcGkuZ2V0TW9kdWxlSWQoKSk7CgkJCWlmIChtb2R1bGUyID09IG51bGwgfHwgbW9kdWxlMi5sZW5ndGggPT0gMCkgewoJCQkJU3RyaW5nIG1vZHVsZUlkID0gbXBpLmdldE1vZHVsZUlkKCk7CgkJCQlpZiAobW9kdWxlSWQgIT0gbnVsbCkgewoJCQkJCWludCBpbmRleCA9IG1vZHVsZUlkLmxhc3RJbmRleE9mKCIjIik7CgkJCQkJbW9kdWxlMiA9IG5ldyBJTW9kdWxlW10geyBuZXcgRGVsZXRlZE1vZHVsZShtb2R1bGVJZC5zdWJzdHJpbmcoaW5kZXggKyAxKSwgbXBpLmdldE5hbWUoKSkgfTsKCQkJCX0KCQkJfQoJCQlpZiAobW9kdWxlMiAhPSBudWxsICYmIG1vZHVsZTIubGVuZ3RoID4gMCkgewoJCQkJbW9kdWxlTGlzdC5hZGQobW9kdWxlMik7CgkJCQlraW5kTGlzdC5hZGQobmV3IEludGVnZXIoU2VydmVyQmVoYXZpb3VyRGVsZWdhdGUuUkVNT1ZFRCkpOwoJCQl9CgkJfQoJfQoKCS8qKgoJICogCgkgKi8KCXB1YmxpYyB2b2lkIGxvYWQoKSB7CgkJU3RyaW5nIGZpbGVuYW1lID0gcGF0aC50b09TU3RyaW5nKCk7CgkJaWYgKCEobmV3IEZpbGUoZmlsZW5hbWUpLmV4aXN0cygpKSkKCQkJcmV0dXJuOwoJCgkJVHJhY2UudHJhY2UoVHJhY2UuRklORVNULCAiTG9hZGluZyBwdWJsaXNoIGluZm8gZnJvbSAiICsgZmlsZW5hbWUpOwoKCQl0cnkgewoJCQlJTWVtZW50byBtZW1lbnRvMiA9IFhNTE1lbWVudG8ubG9hZE1lbWVudG8oZmlsZW5hbWUpOwoJCQlJTWVtZW50b1tdIGNoaWxkcmVuID0gbWVtZW50bzIuZ2V0Q2hpbGRyZW4oIm1vZHVsZSIpOwoJCgkJCWludCBzaXplID0gY2hpbGRyZW4ubGVuZ3RoOwoJCQlmb3IgKGludCBpID0gMDsgaSA8IHNpemU7IGkrKykgewoJCQkJTW9kdWxlUHVibGlzaEluZm8gbXBpID0gbmV3IE1vZHVsZVB1Ymxpc2hJbmZvKGNoaWxkcmVuW2ldKTsKCQkJCW1vZHVsZVB1Ymxpc2hJbmZvLnB1dChnZXRLZXkobXBpLmdldE1vZHVsZUlkKCkpLCBtcGkpOwoJCQl9CgkJfSBjYXRjaCAoRXhjZXB0aW9uIGUpIHsKCQkJVHJhY2UudHJhY2UoVHJhY2UuV0FSTklORywgIkNvdWxkIG5vdCBsb2FkIHB1Ymxpc2ggaW5mb3JtYXRpb246ICIgKyBlLmdldE1lc3NhZ2UoKSk7CgkJfQoJfQoKCS8qKgoJICogCgkgKi8KCXB1YmxpYyB2b2lkIHNhdmUoKSB7CgkJU3RyaW5nIGZpbGVuYW1lID0gcGF0aC50b09TU3RyaW5nKCk7CgkJVHJhY2UudHJhY2UoVHJhY2UuRklORVNULCAiU2F2aW5nIHB1Ymxpc2ggaW5mbyB0byAiICsgZmlsZW5hbWUpOwoJCgkJdHJ5IHsKCQkJWE1MTWVtZW50byBtZW1lbnRvID0gWE1MTWVtZW50by5jcmVhdGVXcml0ZVJvb3QoInNlcnZlciIpOwoKCQkJSXRlcmF0b3IgaXRlcmF0b3IgPSBtb2R1bGVQdWJsaXNoSW5mby5rZXlTZXQoKS5pdGVyYXRvcigpOwoJCQl3aGlsZSAoaXRlcmF0b3IuaGFzTmV4dCgpKSB7CgkJCQlTdHJpbmcgY29udHJvbFJlZiA9IChTdHJpbmcpIGl0ZXJhdG9yLm5leHQoKTsKCQkJCU1vZHVsZVB1Ymxpc2hJbmZvIG1waSA9IChNb2R1bGVQdWJsaXNoSW5mbykgbW9kdWxlUHVibGlzaEluZm8uZ2V0KGNvbnRyb2xSZWYpOwoJCQkJSU1lbWVudG8gY2hpbGQgPSBtZW1lbnRvLmNyZWF0ZUNoaWxkKCJtb2R1bGUiKTsKCQkJCW1waS5zYXZlKGNoaWxkKTsKCQkJfQoJCQltZW1lbnRvLnNhdmVUb0ZpbGUoZmlsZW5hbWUpOwoJCX0gY2F0Y2ggKEV4Y2VwdGlvbiBlKSB7CgkJCVRyYWNlLnRyYWNlKFRyYWNlLlNFVkVSRSwgIkNvdWxkIG5vdCBzYXZlIHB1Ymxpc2ggaW5mb3JtYXRpb24iLCBlKTsKCQl9Cgl9CgkKCXB1YmxpYyB2b2lkIGZpbGwoSU1vZHVsZVtdIG1vZHVsZSkgewoJCU1vZHVsZVB1Ymxpc2hJbmZvIG1waSA9IGdldE1vZHVsZVB1Ymxpc2hJbmZvKG1vZHVsZSk7CgkJaW50IHNpemUgPSBtb2R1bGUubGVuZ3RoOwoJCU1vZHVsZURlbGVnYXRlIHBtID0gKE1vZHVsZURlbGVnYXRlKSBtb2R1bGVbc2l6ZSAtIDFdLmxvYWRBZGFwdGVyKE1vZHVsZURlbGVnYXRlLmNsYXNzLCBudWxsKTsKCQl0cnkgewoJCQlpZiAocG0gIT0gbnVsbCkKCQkJCW1waS5zZXRSZXNvdXJjZXMocG0ubWVtYmVycygpKTsKCQl9IGNhdGNoIChDb3JlRXhjZXB0aW9uIGNlKSB7CgkJCS8vIGlnbm9yZQoJCX0KCQlzYXZlKCk7Cgl9CgoJcHJvdGVjdGVkIElNb2R1bGVSZXNvdXJjZURlbHRhW10gZ2V0RGVsdGEoSU1vZHVsZVtdIG1vZHVsZSkgewoJCWlmIChtb2R1bGUgPT0gbnVsbCkKCQkJcmV0dXJuIG5ldyBJTW9kdWxlUmVzb3VyY2VEZWx0YVswXTsKCQkKCQlNb2R1bGVQdWJsaXNoSW5mbyBtcGkgPSBnZXRNb2R1bGVQdWJsaXNoSW5mbyhtb2R1bGUpOwoJCWludCBzaXplID0gbW9kdWxlLmxlbmd0aDsKCQlNb2R1bGVEZWxlZ2F0ZSBwbSA9IChNb2R1bGVEZWxlZ2F0ZSkgbW9kdWxlW3NpemUgLSAxXS5sb2FkQWRhcHRlcihNb2R1bGVEZWxlZ2F0ZS5jbGFzcywgbnVsbCk7CgkJSU1vZHVsZVJlc291cmNlW10gcmVzb3VyY2VzID0gbnVsbDsKCQl0cnkgewoJCQlpZiAocG0gIT0gbnVsbCkKCQkJCXJlc291cmNlcyA9IHBtLm1lbWJlcnMoKTsKCQl9IGNhdGNoIChDb3JlRXhjZXB0aW9uIGNlKSB7CgkJCS8vIGlnbm9yZQoJCX0KCQlpZiAocmVzb3VyY2VzID09IG51bGwpCgkJCXJlc291cmNlcyA9IG5ldyBJTW9kdWxlUmVzb3VyY2VbMF07CgkJcmV0dXJuIGdldERlbHRhKG1waS5nZXRSZXNvdXJjZXMoKSwgcmVzb3VyY2VzKTsKCX0KCglwcm90ZWN0ZWQgSU1vZHVsZVJlc291cmNlRGVsdGFbXSBnZXREZWx0YShJTW9kdWxlUmVzb3VyY2VbXSBvcmlnaW5hbCwgSU1vZHVsZVJlc291cmNlW10gY3VycmVudCkgewoJCWlmIChvcmlnaW5hbCA9PSBudWxsIHx8IGN1cnJlbnQgPT0gbnVsbCkKCQkJcmV0dXJuIG5ldyBJTW9kdWxlUmVzb3VyY2VEZWx0YVswXTsKCQoJCUxpc3QgbGlzdCA9IG5ldyBBcnJheUxpc3QoKTsKCQlpbnQgc2l6ZSA9IG9yaWdpbmFsLmxlbmd0aDsKCQlpbnQgc2l6ZTIgPSBjdXJyZW50Lmxlbmd0aDsKCQkKCQlNYXAgb3JpZ2luYWxNYXAgPSBuZXcgSGFzaE1hcChzaXplKTsKCQlmb3IgKGludCBpID0gMDsgaSA8IHNpemU7IGkrKykKCQkJb3JpZ2luYWxNYXAucHV0KG9yaWdpbmFsW2ldLCBvcmlnaW5hbFtpXSk7CgoJCS8vIGFkZGVkIGFuZCBjaGFuZ2VkIHJlc291cmNlcwoJCWZvciAoaW50IGkgPSAwOyBpIDwgc2l6ZTI7IGkrKykgewoJCQlJTW9kdWxlUmVzb3VyY2Ugb2xkID0gKElNb2R1bGVSZXNvdXJjZSkgb3JpZ2luYWxNYXAucmVtb3ZlKGN1cnJlbnRbaV0pOwoJCQlpZiAob2xkID09IG51bGwpIHsKCQkJCU1vZHVsZVJlc291cmNlRGVsdGEgZGVsdGEgPSBuZXcgTW9kdWxlUmVzb3VyY2VEZWx0YShjdXJyZW50W2ldLCBJTW9kdWxlUmVzb3VyY2VEZWx0YS5BRERFRCk7CgkJCQlpZiAoY3VycmVudFtpXSBpbnN0YW5jZW9mIElNb2R1bGVGb2xkZXIpIHsKCQkJCQlJTW9kdWxlRm9sZGVyIGN1cnJlbnRGb2xkZXIgPSAoSU1vZHVsZUZvbGRlcikgY3VycmVudFtpXTsgCgkJCQkJZGVsdGEuc2V0Q2hpbGRyZW4oZ2V0RGVsdGFUcmVlKGN1cnJlbnRGb2xkZXIubWVtYmVycygpLCBJTW9kdWxlUmVzb3VyY2VEZWx0YS5BRERFRCkpOwoJCQkJfQoJCQkJbGlzdC5hZGQoZGVsdGEpOwoJCQl9IGVsc2UgewoJCQkJaWYgKGN1cnJlbnRbaV0gaW5zdGFuY2VvZiBJTW9kdWxlRmlsZSkgewoJCQkJCS8vIGluY2x1ZGUgZmlsZXMgb25seSBpZiB0aGUgbW9kaWZpY2F0aW9uIHN0YW1wIGhhcyBjaGFuZ2VkCgkJCQkJSU1vZHVsZUZpbGUgbWYxID0gKElNb2R1bGVGaWxlKSBvbGQ7CgkJCQkJSU1vZHVsZUZpbGUgbWYyID0gKElNb2R1bGVGaWxlKSBjdXJyZW50W2ldOwoJCQkJCWlmIChtZjEuZ2V0TW9kaWZpY2F0aW9uU3RhbXAoKSAhPSBtZjIuZ2V0TW9kaWZpY2F0aW9uU3RhbXAoKSkgewoJCQkJCQlsaXN0LmFkZChuZXcgTW9kdWxlUmVzb3VyY2VEZWx0YShjdXJyZW50W2ldLCBJTW9kdWxlUmVzb3VyY2VEZWx0YS5DSEFOR0VEKSk7CgkJCQkJfQoJCQkJfSBlbHNlIHsKCQkJCQkvLyBpbmNsdWRlIGZvbGRlcnMgb25seSBpZiB0aGVpciBjb250ZW50cyBoYXZlIGNoYW5nZWQKCQkJCQlJTW9kdWxlRm9sZGVyIG1mMSA9IChJTW9kdWxlRm9sZGVyKSBvbGQ7CgkJCQkJSU1vZHVsZUZvbGRlciBtZjIgPSAoSU1vZHVsZUZvbGRlcikgY3VycmVudFtpXTsKCQkJCQlJTW9kdWxlUmVzb3VyY2VEZWx0YVtdIG1yZGMgPSBnZXREZWx0YShtZjEubWVtYmVycygpLCBtZjIubWVtYmVycygpKTsKCQkJCQlpZiAobXJkYy5sZW5ndGggPiAwKSB7CgkJCQkJCU1vZHVsZVJlc291cmNlRGVsdGEgbXJkID0gbmV3IE1vZHVsZVJlc291cmNlRGVsdGEoY3VycmVudFtpXSwgSU1vZHVsZVJlc291cmNlRGVsdGEuTk9fQ0hBTkdFKTsKCQkJCQkJbXJkLnNldENoaWxkcmVuKG1yZGMpOwoJCQkJCQlsaXN0LmFkZChtcmQpOwoJCQkJCX0KCQkJCX0KCQkJfQoJCX0KCQkKCQkvLyByZW1vdmVkIHJlc291cmNlcwoJCWZvciAoaW50IGkgPSAwOyBpIDwgc2l6ZTsgaSsrKSB7CgkJCWlmIChvcmlnaW5hbE1hcC5jb250YWluc0tleShvcmlnaW5hbFtpXSkpIHsKCQkJCU1vZHVsZVJlc291cmNlRGVsdGEgZGVsdGEgPSBuZXcgTW9kdWxlUmVzb3VyY2VEZWx0YShvcmlnaW5hbFtpXSwgSU1vZHVsZVJlc291cmNlRGVsdGEuUkVNT1ZFRCk7CgkJCQlpZiAob3JpZ2luYWxbaV0gaW5zdGFuY2VvZiBJTW9kdWxlRm9sZGVyKSB7CgkJCQkJSU1vZHVsZUZvbGRlciByZW1vdmVkRm9sZGVyID0gKElNb2R1bGVGb2xkZXIpIG9yaWdpbmFsW2ldOyAKCQkJCQlkZWx0YS5zZXRDaGlsZHJlbihnZXREZWx0YVRyZWUocmVtb3ZlZEZvbGRlci5tZW1iZXJzKCksIElNb2R1bGVSZXNvdXJjZURlbHRhLlJFTU9WRUQpKTsKCQkJCX0KCQkJCWxpc3QuYWRkKGRlbHRhKTsKCQkJfQoJCX0KCQkKCQlyZXR1cm4gKElNb2R1bGVSZXNvdXJjZURlbHRhW10pIGxpc3QudG9BcnJheShuZXcgSU1vZHVsZVJlc291cmNlRGVsdGFbbGlzdC5zaXplKCldKTsKCX0KCglwcm90ZWN0ZWQgYm9vbGVhbiBoYXNEZWx0YShJTW9kdWxlW10gbW9kdWxlKSB7CgkJaWYgKG1vZHVsZSA9PSBudWxsKQoJCQlyZXR1cm4gZmFsc2U7CgkJCgkJTW9kdWxlUHVibGlzaEluZm8gbXBpID0gZ2V0TW9kdWxlUHVibGlzaEluZm8obW9kdWxlKTsKCQlpbnQgc2l6ZSA9IG1vZHVsZS5sZW5ndGg7CgkJTW9kdWxlRGVsZWdhdGUgcG0gPSAoTW9kdWxlRGVsZWdhdGUpIG1vZHVsZVtzaXplIC0gMV0ubG9hZEFkYXB0ZXIoTW9kdWxlRGVsZWdhdGUuY2xhc3MsIG51bGwpOwoJCUlNb2R1bGVSZXNvdXJjZVtdIHJlc291cmNlcyA9IG51bGw7CgkJdHJ5IHsKCQkJaWYgKHBtICE9IG51bGwpCgkJCQlyZXNvdXJjZXMgPSBwbS5tZW1iZXJzKCk7CgkJfSBjYXRjaCAoQ29yZUV4Y2VwdGlvbiBjZSkgewoJCQkvLyBpZ25vcmUKCQl9CgkJaWYgKHJlc291cmNlcyA9PSBudWxsKQoJCQlyZXNvdXJjZXMgPSBuZXcgSU1vZHVsZVJlc291cmNlWzBdOwoJCXJldHVybiBoYXNEZWx0YShtcGkuZ2V0UmVzb3VyY2VzKCksIHJlc291cmNlcyk7Cgl9CgoJcHJvdGVjdGVkIGJvb2xlYW4gaGFzRGVsdGEoSU1vZHVsZVJlc291cmNlW10gb3JpZ2luYWwsIElNb2R1bGVSZXNvdXJjZVtdIGN1cnJlbnQpIHsKCQlpZiAob3JpZ2luYWwgPT0gbnVsbCB8fCBjdXJyZW50ID09IG51bGwpCgkJCXJldHVybiBmYWxzZTsKCQkKCQlpbnQgc2l6ZSA9IG9yaWdpbmFsLmxlbmd0aDsKCQlpbnQgc2l6ZTIgPSBjdXJyZW50Lmxlbmd0aDsKCQkKCQlNYXAgb3JpZ2luYWxNYXAgPSBuZXcgSGFzaE1hcChzaXplKTsKCQlmb3IgKGludCBpID0gMDsgaSA8IHNpemU7IGkrKykKCQkJb3JpZ2luYWxNYXAucHV0KG9yaWdpbmFsW2ldLCBvcmlnaW5hbFtpXSk7CgkJCgkJLy8gYWRkZWQgYW5kIGNoYW5nZWQgcmVzb3VyY2VzCgkJZm9yIChpbnQgaSA9IDA7IGkgPCBzaXplMjsgaSsrKSB7CgkJCUlNb2R1bGVSZXNvdXJjZSBvbGQgPSAoSU1vZHVsZVJlc291cmNlKSBvcmlnaW5hbE1hcC5yZW1vdmUoY3VycmVudFtpXSk7CgkJCWlmIChvbGQgPT0gbnVsbCkKCQkJCXJldHVybiB0cnVlOwoJCQkKCQkJaWYgKGN1cnJlbnRbaV0gaW5zdGFuY2VvZiBJTW9kdWxlRmlsZSkgewoJCQkJLy8gaW5jbHVkZSBmaWxlcyBvbmx5IGlmIHRoZSBtb2RpZmljYXRpb24gc3RhbXAgaGFzIGNoYW5nZWQKCQkJCUlNb2R1bGVGaWxlIG1mMSA9IChJTW9kdWxlRmlsZSkgb2xkOwoJCQkJSU1vZHVsZUZpbGUgbWYyID0gKElNb2R1bGVGaWxlKSBjdXJyZW50W2ldOwoJCQkJaWYgKG1mMS5nZXRNb2RpZmljYXRpb25TdGFtcCgpICE9IG1mMi5nZXRNb2RpZmljYXRpb25TdGFtcCgpKQoJCQkJCXJldHVybiB0cnVlOwoJCQl9IGVsc2UgewoJCQkJLy8gaW5jbHVkZSBmb2xkZXJzIG9ubHkgaWYgdGhlaXIgY29udGVudHMgaGF2ZSBjaGFuZ2VkCgkJCQlJTW9kdWxlRm9sZGVyIG1mMSA9IChJTW9kdWxlRm9sZGVyKSBvbGQ7CgkJCQlJTW9kdWxlRm9sZGVyIG1mMiA9IChJTW9kdWxlRm9sZGVyKSBjdXJyZW50W2ldOwoJCQkJaWYgKGhhc0RlbHRhKG1mMS5tZW1iZXJzKCksIG1mMi5tZW1iZXJzKCkpKQoJCQkJCXJldHVybiB0cnVlOwoJCQl9CgkJfQoJCQoJCS8vIHJlbW92ZWQgcmVzb3VyY2VzCgkJcmV0dXJuICFvcmlnaW5hbE1hcC5pc0VtcHR5KCk7Cgl9CgoJLyoqCgkgKiBDcmVhdGUgYSByZXNvdXJjZSBkZWx0YSBmb3IgYW4gZW50aXJlIHRyZWUuCgkgKi8KCXByb3RlY3RlZCBJTW9kdWxlUmVzb3VyY2VEZWx0YVtdIGdldERlbHRhVHJlZShJTW9kdWxlUmVzb3VyY2VbXSByZXNvdXJjZXMsIGludCBraW5kKSB7CgkJaWYgKHJlc291cmNlcyA9PSBudWxsKQoJCQlyZXR1cm4gbmV3IElNb2R1bGVSZXNvdXJjZURlbHRhWzBdOwoJCgkJTGlzdCBsaXN0ID0gbmV3IEFycmF5TGlzdCgpOwoJCQoJCS8vIGxvb2sgZm9yIGR1cGxpY2F0ZXMKCQlpbnQgc2l6ZSA9IHJlc291cmNlcy5sZW5ndGg7CgkJZm9yIChpbnQgaSA9IDA7IGkgPCBzaXplOyBpKyspIHsKCQkJTW9kdWxlUmVzb3VyY2VEZWx0YSBtcmQgPSBuZXcgTW9kdWxlUmVzb3VyY2VEZWx0YShyZXNvdXJjZXNbaV0sIGtpbmQpOwoJCQlpZiAocmVzb3VyY2VzW2ldIGluc3RhbmNlb2YgSU1vZHVsZUZvbGRlcikgewoJCQkJSU1vZHVsZUZvbGRlciBtZiA9IChJTW9kdWxlRm9sZGVyKSByZXNvdXJjZXNbaV07CgkJCQltcmQuc2V0Q2hpbGRyZW4oZ2V0RGVsdGFUcmVlKG1mLm1lbWJlcnMoKSwga2luZCkpOwoJCQl9CgkJCWxpc3QuYWRkKG1yZCk7CgkJfQoJCQoJCUlNb2R1bGVSZXNvdXJjZURlbHRhW10gZGVsdGEgPSBuZXcgSU1vZHVsZVJlc291cmNlRGVsdGFbbGlzdC5zaXplKCldOwoJCWxpc3QudG9BcnJheShkZWx0YSk7CgkJcmV0dXJuIGRlbHRhOwoJfQoKCS8qKgoJICogUmV0dXJucyB0cnVlIGlmIHRoZSBsaXN0IG9mIG1vZHVsZXMgYmVpbmcgcHVibGlzaGVkIGRvZXMgbm90IG1hdGNoIHRoZSBwcmV2aW91cwoJICogbGlzdCBvZiBwdWJsaXNoZWQgbW9kdWxlcy4KCSAqIAoJICogVE9ETzogVGhpcyBtZXRob2Qgc2hvdWxkIGNvbXBhcmUgdGhlIG1vZHVsZXMuIEZvciBub3csIGNvbXBhcmluZyB0aGUgc2l6ZSBpcyBmaW5lLgoJICogCgkgKiBAcGFyYW0gbW9kdWxlcyBhIGxpc3Qgb2YgbW9kdWxlcwoJICogQHJldHVybiA8Y29kZT50cnVlPC9jb2RlPiBpZiB0aGUgc3RydWN0dXJlIG9mIHB1Ymxpc2hlZCBtb2R1bGVzIGhhcyBjaGFuZ2VkLCBvcgoJICogICAgPGNvZGU+ZmFsc2U8L2NvZGU+IG90aGVyd2lzZQoJICovCglwcm90ZWN0ZWQgYm9vbGVhbiBoYXNTdHJ1Y3R1cmVDaGFuZ2VkKExpc3QgbW9kdWxlcykgewoJCXJldHVybiBtb2R1bGVzLnNpemUoKSAhPSBtb2R1bGVQdWJsaXNoSW5mby5rZXlTZXQoKS5zaXplKCk7Cgl9Cn0=