LyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogQ29weXJpZ2h0IChjKSAyMDAzLCAyMDA1IElCTSBDb3Jwb3JhdGlvbiBhbmQgb3RoZXJzLgogKiBBbGwgcmlnaHRzIHJlc2VydmVkLiBUaGlzIHByb2dyYW0gYW5kIHRoZSBhY2NvbXBhbnlpbmcgbWF0ZXJpYWxzCiAqIGFyZSBtYWRlIGF2YWlsYWJsZSB1bmRlciB0aGUgdGVybXMgb2YgdGhlIEVjbGlwc2UgUHVibGljIExpY2Vuc2UgdjEuMAogKiB3aGljaCBhY2NvbXBhbmllcyB0aGlzIGRpc3RyaWJ1dGlvbiwgYW5kIGlzIGF2YWlsYWJsZSBhdAogKiBodHRwOi8vd3d3LmVjbGlwc2Uub3JnL2xlZ2FsL2VwbC12MTAuaHRtbAqgKiAKICogQ29udHJpYnV0b3JzOgogKiAgICAgSUJNIENvcnBvcmF0aW9uIC0gSW5pdGlhbCBBUEkgYW5kIGltcGxlbWVudGF0aW9uCiAqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqLwpwYWNrYWdlIG9yZy5lY2xpcHNlLndzdC5zZXJ2ZXIuY29yZS5pbnRlcm5hbDsKCmltcG9ydCBqYXZhLmlvLkZpbGVJbnB1dFN0cmVhbTsKaW1wb3J0IGphdmEuaW8uSW5wdXRTdHJlYW07CmltcG9ydCBqYXZhLnV0aWwuQXJyYXlMaXN0OwppbXBvcnQgamF2YS51dGlsLkhhc2hNYXA7CmltcG9ydCBqYXZhLnV0aWwuSXRlcmF0b3I7CmltcG9ydCBqYXZhLnV0aWwuTGlzdDsKaW1wb3J0IGphdmEudXRpbC5NYXA7CgppbXBvcnQgb3JnLmVjbGlwc2UuY29yZS5yZXNvdXJjZXMuSUZpbGU7CmltcG9ydCBvcmcuZWNsaXBzZS5jb3JlLnJ1bnRpbWUuKjsKLyoqCiAqIEhlbHBlciBjbGFzcyBmb3Igc3RvcmluZyBydW50aW1lIGFuZCBzZXJ2ZXIgYXR0cmlidXRlcy4KICovCnB1YmxpYyBhYnN0cmFjdCBjbGFzcyBCYXNlIHsKCXByb3RlY3RlZCBzdGF0aWMgZmluYWwgU3RyaW5nIFBST1BfTE9DS0VEID0gImxvY2tlZCI7Cglwcm90ZWN0ZWQgc3RhdGljIGZpbmFsIFN0cmluZyBQUk9QX1BSSVZBVEUgPSAicHJpdmF0ZSI7Cglwcm90ZWN0ZWQgc3RhdGljIGZpbmFsIFN0cmluZyBQUk9QX05BTUUgPSAibmFtZSI7Cglwcm90ZWN0ZWQgc3RhdGljIGZpbmFsIFN0cmluZyBQUk9QX0lEID0gImlkIjsKCglwcm90ZWN0ZWQgTWFwIG1hcCA9IG5ldyBIYXNoTWFwKCk7CgkKCS8vIGZpbGUgbG9hZGVkIGZyb20sIG9yIG51bGwgaWYgaXQgaXMgc2F2ZWQgaW4gbWV0YWRhdGEKCXByb3RlY3RlZCBJRmlsZSBmaWxlOwoJCglwdWJsaWMgQmFzZShJRmlsZSBmaWxlKSB7CgkJdGhpcy5maWxlID0gZmlsZTsKCX0KCglwdWJsaWMgQmFzZShJRmlsZSBmaWxlLCBTdHJpbmcgaWQpIHsKCQl0aGlzLmZpbGUgPSBmaWxlOwoJCS8vdGhpcy5tYXAgPSBtYXA7CgkJbWFwLnB1dChQUk9QX0lELCBpZCk7Cgl9CgoJLyoqCgkgKiBSZXR1cm5zIHRoZSB0aW1lc3RhbXAgb2YgdGhpcyBvYmplY3QuCgkgKiBUaW1lc3RhbXBzIGFyZSBtb25vdG9uaWNhbGx5IGluY3JlYXNlZCBlYWNoIHRpbWUgdGhlIG9iamVjdCBpcyBzYXZlZAoJICogYW5kIGNhbiBiZSB1c2VkIHRvIGRldGVybWluZSBpZiBhbnkgY2hhbmdlcyBoYXZlIGJlZW4gbWFkZSBvbiBkaXNrCgkgKiBzaW5jZSB0aGUgb2JqZWN0IHdhcyBsb2FkZWQuCgkgKiAKCSAqIEByZXR1cm4gdGhlIG9iamVjdCdzIHRpbWVzdGFtcAoJICovCglwdWJsaWMgaW50IGdldFRpbWVzdGFtcCgpIHsKCQlyZXR1cm4gZ2V0QXR0cmlidXRlKCJ0aW1lc3RhbXAiLCAtMSk7Cgl9CgoJLyoqCgkgKiBSZXR1cm5zIHRoZSBmaWxlIHdoZXJlIHRoaXMgc2VydmVyIGluc3RhbmNlIGlzIHNlcmlhbGl6ZWQuCgkgKiAKCSAqIEByZXR1cm4gdGhlIGZpbGUgaW4gdGhlIHdvcmtzcGFjZSB3aGVyZSB0aGUgc2VydmVyIGluc3RhbmNlCgkgKiBpcyBzZXJpYWxpemVkLCBvciA8Y29kZT5udWxsPC9jb2RlPiBpZiB0aGUgaW5mb3JtYXRpb24gaXMKCSAqIGluc3RlYWQgdG8gYmUgcGVyc2lzdGVkIHdpdGggdGhlIHdvcmtzcGFjZSBidXQgbm90IHdpdGggYW55CgkgKiBwYXJ0aWN1bGFyIHdvcmtzcGFjZSByZXNvdXJjZQoJICovCglwdWJsaWMgSUZpbGUgZ2V0RmlsZSgpIHsKCQlyZXR1cm4gZmlsZTsKCX0KCglwdWJsaWMgU3RyaW5nIGdldEF0dHJpYnV0ZShTdHJpbmcgYXR0cmlidXRlTmFtZSwgU3RyaW5nIGRlZmF1bHRWYWx1ZSkgewoJCXRyeSB7CgkJCU9iamVjdCBvYmogPSBtYXAuZ2V0KGF0dHJpYnV0ZU5hbWUpOwoJCQlpZiAob2JqID09IG51bGwpCgkJCQlyZXR1cm4gZGVmYXVsdFZhbHVlOwoJCQlyZXR1cm4gKFN0cmluZykgb2JqOwoJCX0gY2F0Y2ggKEV4Y2VwdGlvbiBlKSB7CgkJCS8vIGlnbm9yZQoJCX0KCQlyZXR1cm4gZGVmYXVsdFZhbHVlOwoJfQoKCXB1YmxpYyBpbnQgZ2V0QXR0cmlidXRlKFN0cmluZyBhdHRyaWJ1dGVOYW1lLCBpbnQgZGVmYXVsdFZhbHVlKSB7CgkJdHJ5IHsKCQkJT2JqZWN0IG9iaiA9IG1hcC5nZXQoYXR0cmlidXRlTmFtZSk7CgkJCWlmIChvYmogPT0gbnVsbCkKCQkJCXJldHVybiBkZWZhdWx0VmFsdWU7CgkJCXJldHVybiBJbnRlZ2VyLnBhcnNlSW50KChTdHJpbmcpIG9iaik7CgkJfSBjYXRjaCAoRXhjZXB0aW9uIGUpIHsKCQkJLy8gaWdub3JlCgkJfQoJCXJldHVybiBkZWZhdWx0VmFsdWU7Cgl9CgoJcHVibGljIGJvb2xlYW4gZ2V0QXR0cmlidXRlKFN0cmluZyBhdHRyaWJ1dGVOYW1lLCBib29sZWFuIGRlZmF1bHRWYWx1ZSkgewoJCXRyeSB7CgkJCU9iamVjdCBvYmogPSBtYXAuZ2V0KGF0dHJpYnV0ZU5hbWUpOwoJCQlpZiAob2JqID09IG51bGwpCgkJCQlyZXR1cm4gZGVmYXVsdFZhbHVlOwoJCQlyZXR1cm4gQm9vbGVhbi52YWx1ZU9mKChTdHJpbmcpIG9iaikuYm9vbGVhblZhbHVlKCk7CgkJfSBjYXRjaCAoRXhjZXB0aW9uIGUpIHsKCQkJLy8gaWdub3JlCgkJfQoJCXJldHVybiBkZWZhdWx0VmFsdWU7Cgl9CgkKCXB1YmxpYyBMaXN0IGdldEF0dHJpYnV0ZShTdHJpbmcgYXR0cmlidXRlTmFtZSwgTGlzdCBkZWZhdWx0VmFsdWUpIHsKCQl0cnkgewoJCQlPYmplY3Qgb2JqID0gbWFwLmdldChhdHRyaWJ1dGVOYW1lKTsKCQkJaWYgKG9iaiA9PSBudWxsKQoJCQkJcmV0dXJuIGRlZmF1bHRWYWx1ZTsKCQkJTGlzdCBsaXN0ID0gKExpc3QpIG9iajsKCQkJaWYgKGxpc3QgIT0gbnVsbCkKCQkJCXJldHVybiBsaXN0OwoJCX0gY2F0Y2ggKEV4Y2VwdGlvbiBlKSB7CgkJCS8vIGlnbm9yZQoJCX0KCQlyZXR1cm4gZGVmYXVsdFZhbHVlOwoJfQoJCglwdWJsaWMgTWFwIGdldEF0dHJpYnV0ZShTdHJpbmcgYXR0cmlidXRlTmFtZSwgTWFwIGRlZmF1bHRWYWx1ZSkgewoJCXRyeSB7CgkJCU9iamVjdCBvYmogPSBtYXAuZ2V0KGF0dHJpYnV0ZU5hbWUpOwoJCQlpZiAob2JqID09IG51bGwpCgkJCQlyZXR1cm4gZGVmYXVsdFZhbHVlOwoJCQlNYXAgbWFwMiA9IChNYXApIG9iajsKCQkJaWYgKG1hcDIgIT0gbnVsbCkKCQkJCXJldHVybiBtYXAyOwoJCX0gY2F0Y2ggKEV4Y2VwdGlvbiBlKSB7CgkJCS8vIGlnbm9yZQoJCX0KCQlyZXR1cm4gZGVmYXVsdFZhbHVlOwoJfQoKCXB1YmxpYyBTdHJpbmcgZ2V0SWQoKSB7CgkJcmV0dXJuIGdldEF0dHJpYnV0ZShQUk9QX0lELCAiIik7Cgl9CgoJcHVibGljIFN0cmluZyBnZXROYW1lKCkgewoJCXJldHVybiBnZXRBdHRyaWJ1dGUoUFJPUF9OQU1FLCAiIik7Cgl9CgoJcHVibGljIGJvb2xlYW4gaXNSZWFkT25seSgpIHsKCQlyZXR1cm4gZ2V0QXR0cmlidXRlKFBST1BfTE9DS0VELCBmYWxzZSk7Cgl9CgoJLyoqCgkgKiBSZXR1cm5zIDxjb2RlPnRydWU8L2NvZGU+IGlmIHRoaXMgcnVudGltZSBpcyBwcml2YXRlIChub3Qgc2hvd24KCSAqIGluIHRoZSBVSSB0byB0aGUgdXNlcnMpLCBhbmQgPGNvZGU+ZmFsc2U8L2NvZGU+IG90aGVyd2lzZS4KCSAqIAoJICogQHJldHVybiA8Y29kZT50cnVlPC9jb2RlPiBpZiB0aGlzIHJ1bnRpbWUgaXMgcHJpdmF0ZSwKCSAqICAgIGFuZCA8Y29kZT5mYWxzZTwvY29kZT4gb3RoZXJ3aXNlCgkgKi8KCXB1YmxpYyBib29sZWFuIGlzUHJpdmF0ZSgpIHsKCQlyZXR1cm4gZ2V0QXR0cmlidXRlKFBST1BfUFJJVkFURSwgZmFsc2UpOwoJfQoJCglwdWJsaWMgYm9vbGVhbiBpc1dvcmtpbmdDb3B5KCkgewoJCXJldHVybiBmYWxzZTsKCX0KCQoJcHJvdGVjdGVkIGFic3RyYWN0IFN0cmluZyBnZXRYTUxSb290KCk7CgkKCXByb3RlY3RlZCB2b2lkIHNhdmUoSU1lbWVudG8gbWVtZW50bykgewoJCS8vSU1lbWVudG8gY2hpbGQgPSBtZW1lbnRvLmNyZWF0ZUNoaWxkKCJwcm9wZXJ0aWVzIik7CgkJSU1lbWVudG8gY2hpbGQgPSBtZW1lbnRvOwoJCUl0ZXJhdG9yIGl0ZXJhdG9yID0gbWFwLmtleVNldCgpLml0ZXJhdG9yKCk7CgkJd2hpbGUgKGl0ZXJhdG9yLmhhc05leHQoKSkgewoJCQlTdHJpbmcga2V5ID0gKFN0cmluZykgaXRlcmF0b3IubmV4dCgpOwoJCQlPYmplY3Qgb2JqID0gbWFwLmdldChrZXkpOwoJCQlpZiAob2JqIGluc3RhbmNlb2YgU3RyaW5nKQoJCQkJY2hpbGQucHV0U3RyaW5nKGtleSwgKFN0cmluZykgb2JqKTsKCQkJZWxzZSBpZiAob2JqIGluc3RhbmNlb2YgSW50ZWdlcikgewoJCQkJSW50ZWdlciBpbiA9IChJbnRlZ2VyKSBvYmo7CgkJCQljaGlsZC5wdXRJbnRlZ2VyKGtleSwgaW4uaW50VmFsdWUoKSk7CgkJCX0gZWxzZSBpZiAob2JqIGluc3RhbmNlb2YgQm9vbGVhbikgewoJCQkJQm9vbGVhbiBib29sID0gKEJvb2xlYW4pIG9iajsKCQkJCWNoaWxkLnB1dEJvb2xlYW4oa2V5LCBib29sLmJvb2xlYW5WYWx1ZSgpKTsKCQkJfSBlbHNlIGlmIChvYmogaW5zdGFuY2VvZiBMaXN0KSB7CgkJCQlMaXN0IGxpc3QgPSAoTGlzdCkgb2JqOwoJCQkJc2F2ZUxpc3QoY2hpbGQsIGtleSwgbGlzdCk7CgkJCX0gZWxzZSBpZiAob2JqIGluc3RhbmNlb2YgTWFwKSB7CgkJCQlNYXAgbWFwMiA9IChNYXApIG9iajsKCQkJCXNhdmVNYXAoY2hpbGQsIGtleSwgbWFwMik7CgkJCQkKCQkJfQoJCX0KCQlzYXZlU3RhdGUoY2hpbGQpOwoJfQoJcHJvdGVjdGVkIHZvaWQgc2F2ZU1hcChJTWVtZW50byBtZW1lbnRvLCBTdHJpbmcga2V5LCBNYXAgbWFwMikgewoJCUlNZW1lbnRvIGNoaWxkID0gbWVtZW50by5jcmVhdGVDaGlsZCgibWFwIik7CgkJY2hpbGQucHV0U3RyaW5nKCJrZXkiLCBrZXkpOwoJCUl0ZXJhdG9yIGl0ZXJhdG9yID0gbWFwMi5rZXlTZXQoKS5pdGVyYXRvcigpOwoJCXdoaWxlIChpdGVyYXRvci5oYXNOZXh0KCkpIHsKCQkJU3RyaW5nIHMgPSAoU3RyaW5nKSBpdGVyYXRvci5uZXh0KCk7CgkJCWNoaWxkLnB1dFN0cmluZyhzLCAoU3RyaW5nKW1hcDIuZ2V0KHMpKTsKCQl9Cgl9CgkKCXByb3RlY3RlZCB2b2lkIHNhdmVMaXN0KElNZW1lbnRvIG1lbWVudG8sIFN0cmluZyBrZXksIExpc3QgbGlzdCkgewoJCUlNZW1lbnRvIGNoaWxkID0gbWVtZW50by5jcmVhdGVDaGlsZCgibGlzdCIpOwoJCWNoaWxkLnB1dFN0cmluZygia2V5Iiwga2V5KTsKCQlpbnQgaSA9IDA7CgkJSXRlcmF0b3IgaXRlcmF0b3IgPSBsaXN0Lml0ZXJhdG9yKCk7CgkJd2hpbGUgKGl0ZXJhdG9yLmhhc05leHQoKSkgewoJCQlTdHJpbmcgcyA9IChTdHJpbmcpIGl0ZXJhdG9yLm5leHQoKTsKCQkJY2hpbGQucHV0U3RyaW5nKCJ2YWx1ZSIgKyAoaSsrKSwgcyk7CgkJfQoJfQoKCXByb3RlY3RlZCB2b2lkIHNhdmVUb0ZpbGUoSVByb2dyZXNzTW9uaXRvciBtb25pdG9yKSB0aHJvd3MgQ29yZUV4Y2VwdGlvbiB7CgkJdHJ5IHsKCQkJWE1MTWVtZW50byBtZW1lbnRvID0gWE1MTWVtZW50by5jcmVhdGVXcml0ZVJvb3QoZ2V0WE1MUm9vdCgpKTsKCQkJc2F2ZShtZW1lbnRvKTsKCgkJCUlucHV0U3RyZWFtIGluID0gbWVtZW50by5nZXRJbnB1dFN0cmVhbSgpOwoJCQlpZiAoZmlsZS5leGlzdHMoKSkKCQkJCWZpbGUuc2V0Q29udGVudHMoaW4sIHRydWUsIHRydWUsIFByb2dyZXNzVXRpbC5nZXRTdWJNb25pdG9yRm9yKG1vbml0b3IsIDEwMDApKTsKCQkJZWxzZQoJCQkJZmlsZS5jcmVhdGUoaW4sIHRydWUsIFByb2dyZXNzVXRpbC5nZXRTdWJNb25pdG9yRm9yKG1vbml0b3IsIDEwMDApKTsKCQl9IGNhdGNoIChFeGNlcHRpb24gZSkgewoJCQlUcmFjZS50cmFjZShUcmFjZS5TRVZFUkUsICJDb3VsZCBub3Qgc2F2ZSAiICsgZ2V0WE1MUm9vdCgpLCBlKTsKCQkJdGhyb3cgbmV3IENvcmVFeGNlcHRpb24obmV3IFN0YXR1cyhJU3RhdHVzLkVSUk9SLCBTZXJ2ZXJQbHVnaW4uUExVR0lOX0lELCAwLCBTZXJ2ZXJQbHVnaW4uZ2V0UmVzb3VyY2UoIiVlcnJvclNhdmluZyIsIGdldEZpbGUoKS50b1N0cmluZygpKSwgZSkpOwoJCX0KCX0KCQoJcHJvdGVjdGVkIHZvaWQgZG9TYXZlKElQcm9ncmVzc01vbml0b3IgbW9uaXRvcikgdGhyb3dzIENvcmVFeGNlcHRpb24gewoJCWlmIChmaWxlICE9IG51bGwpCgkJCXNhdmVUb0ZpbGUobW9uaXRvcik7CgkJZWxzZQoJCQlzYXZlVG9NZXRhZGF0YShtb25pdG9yKTsKCQlSZXNvdXJjZU1hbmFnZXIuZ2V0SW5zdGFuY2UoKS5yZXNvbHZlU2VydmVycygpOwoJfQoJCglwcm90ZWN0ZWQgdm9pZCBzYXZlVG9NZXRhZGF0YShJUHJvZ3Jlc3NNb25pdG9yIG1vbml0b3IpIHsKCQkvLyBkbyBub3RoaW5nCgl9CgkKCXByb3RlY3RlZCBhYnN0cmFjdCB2b2lkIHNhdmVTdGF0ZShJTWVtZW50byBtZW1lbnRvKTsKCglwcm90ZWN0ZWQgdm9pZCBsb2FkKElNZW1lbnRvIG1lbWVudG8pIHsKCQltYXAgPSBuZXcgSGFzaE1hcCgpOwoJCQoJCUl0ZXJhdG9yIGl0ZXJhdG9yID0gbWVtZW50by5nZXROYW1lcygpLml0ZXJhdG9yKCk7CgkJd2hpbGUgKGl0ZXJhdG9yLmhhc05leHQoKSkgewoJCQlTdHJpbmcga2V5ID0gKFN0cmluZykgaXRlcmF0b3IubmV4dCgpOwoJCQltYXAucHV0KGtleSwgbWVtZW50by5nZXRTdHJpbmcoa2V5KSk7CgkJfQoJCUlNZW1lbnRvW10gY2hpbGRyZW4gPSBtZW1lbnRvLmdldENoaWxkcmVuKCJsaXN0Iik7CgkJaWYgKGNoaWxkcmVuICE9IG51bGwpIHsKCQkJaW50IHNpemUgPSBjaGlsZHJlbi5sZW5ndGg7CgkJCWZvciAoaW50IGkgPSAwOyBpIDwgc2l6ZTsgaSsrKSB7CgkJCQlsb2FkTGlzdChjaGlsZHJlbltpXSk7CgkJCX0KCQl9CgkJSU1lbWVudG9bXSBtYXBzID0gbWVtZW50by5nZXRDaGlsZHJlbigibWFwIik7CgkJaWYgKG1hcHMgIT0gbnVsbCkgewoJCQlmb3IgKGludCBpID0gMDsgaSA8bWFwcy5sZW5ndGggOyBpKyspIHsKCQkJCWxvYWRNYXAobWFwc1tpXSk7CgkJCX0KCQl9CgkJCgkJbG9hZFN0YXRlKG1lbWVudG8pOwoJfQoKCQoJcHJvdGVjdGVkIHZvaWQgbG9hZE1hcChJTWVtZW50byBtZW1lbnRvKSB7CgkJU3RyaW5nIGtleSA9IG1lbWVudG8uZ2V0U3RyaW5nKCJrZXkiKTsKCQlNYXAgdk1hcCA9IG5ldyBIYXNoTWFwKCk7CgkJTGlzdCBrZXlzID0gbWVtZW50by5nZXROYW1lcygpOwoJCUl0ZXJhdG9yIGl0ZXJhdG9yID0ga2V5cy5pdGVyYXRvcigpOwoJCXdoaWxlKGl0ZXJhdG9yLmhhc05leHQoKSkgewoJCQlTdHJpbmcgcyA9IChTdHJpbmcpaXRlcmF0b3IubmV4dCgpOwoJCQlTdHJpbmcgdiA9IG1lbWVudG8uZ2V0U3RyaW5nKHMpOwoJCQl2TWFwLnB1dChzLHYpOwoJCX0KCQltYXAucHV0KGtleSwgdk1hcCk7Cgl9CgkKCQoJcHJvdGVjdGVkIHZvaWQgbG9hZExpc3QoSU1lbWVudG8gbWVtZW50bykgewoJCVN0cmluZyBrZXkgPSBtZW1lbnRvLmdldFN0cmluZygia2V5Iik7CgkJTGlzdCBsaXN0ID0gbmV3IEFycmF5TGlzdCgpOwoJCWludCBpID0gMDsKCQlTdHJpbmcga2V5MiA9IG1lbWVudG8uZ2V0U3RyaW5nKCJ2YWx1ZSIgKyAoaSsrKSk7CgkJd2hpbGUgKGtleTIgIT0gbnVsbCkgewoJCQlsaXN0LmFkZChrZXkyKTsKCQkJa2V5MiA9IG1lbWVudG8uZ2V0U3RyaW5nKCJ2YWx1ZSIgKyAoaSsrKSk7CgkJfQoJCW1hcC5wdXQoa2V5LCBsaXN0KTsKCX0KCQoJcHJvdGVjdGVkIGFic3RyYWN0IHZvaWQgbG9hZFN0YXRlKElNZW1lbnRvIG1lbWVudG8pOwoJCglwcm90ZWN0ZWQgdm9pZCByZXNvbHZlKCkgewoJCS8vIGRvIG5vdGhpbmcKCX0KCQoJcHVibGljIHZvaWQgZGVsZXRlKCkgdGhyb3dzIENvcmVFeGNlcHRpb24gewoJCWlmIChpc1dvcmtpbmdDb3B5KCkpCgkJCXRocm93IG5ldyBDb3JlRXhjZXB0aW9uKG5ldyBTdGF0dXMoSVN0YXR1cy5FUlJPUiwgU2VydmVyUGx1Z2luLlBMVUdJTl9JRCwgMCwgIkNhbm5vdCBkZWxldGUgYSB3b3JraW5nIGNvcHkiLCBudWxsKSk7CgkJCgkJaWYgKGZpbGUgIT0gbnVsbCkKCQkJZmlsZS5kZWxldGUodHJ1ZSwgdHJ1ZSwgbmV3IE51bGxQcm9ncmVzc01vbml0b3IoKSk7CgkJZWxzZQoJCQlkZWxldGVGcm9tTWV0YWRhdGEoKTsKCX0KCglwcm90ZWN0ZWQgdm9pZCBkZWxldGVGcm9tTWV0YWRhdGEoKSB7CgkJLy8gZG8gbm90aGluZwoJfQoKCXB1YmxpYyBib29sZWFuIGVxdWFscyhPYmplY3Qgb2JqKSB7CgkJaWYgKCEob2JqIGluc3RhbmNlb2YgQmFzZSkpCgkJCXJldHVybiBmYWxzZTsKCQkKCQlCYXNlIGJhc2UgPSAoQmFzZSkgb2JqOwoJCWlmIChnZXRJZCgpID09IG51bGwpCgkJCXJldHVybiBmYWxzZTsKCQlyZXR1cm4gZ2V0SWQoKS5lcXVhbHMoYmFzZS5nZXRJZCgpKTsKCX0KCgkvKioKCSAqIAoJICovCglwcm90ZWN0ZWQgdm9pZCBsb2FkRnJvbUZpbGUoSVByb2dyZXNzTW9uaXRvciBtb25pdG9yKSB0aHJvd3MgQ29yZUV4Y2VwdGlvbiB7CgkJSW5wdXRTdHJlYW0gaW4gPSBudWxsOwoJCXRyeSB7CgkJCWluID0gZmlsZS5nZXRDb250ZW50cygpOwoJCQlJTWVtZW50byBtZW1lbnRvID0gWE1MTWVtZW50by5sb2FkTWVtZW50byhpbik7CgkJCWxvYWQobWVtZW50byk7CgkJfSBjYXRjaCAoRXhjZXB0aW9uIGUpIHsKCQkJVHJhY2UudHJhY2UoVHJhY2UuU0VWRVJFLCAiQ291bGQgbm90IGxvYWQgZnJvbSBmaWxlIiArIGUuZ2V0TWVzc2FnZSgpLCBlKTsKCQkJdGhyb3cgbmV3IENvcmVFeGNlcHRpb24obmV3IFN0YXR1cyhJU3RhdHVzLkVSUk9SLCBTZXJ2ZXJQbHVnaW4uUExVR0lOX0lELCAwLCBTZXJ2ZXJQbHVnaW4uZ2V0UmVzb3VyY2UoIiVlcnJvckxvYWRpbmciLCBnZXRGaWxlKCkudG9TdHJpbmcoKSksIGUpKTsKCQl9IGZpbmFsbHkgewoJCQl0cnkgewoJCQkJaW4uY2xvc2UoKTsKCQkJfSBjYXRjaCAoRXhjZXB0aW9uIGUpIHsKCQkJCS8vIGlnbm9yZQoJCQl9CgkJfQoJfQoJCglwcm90ZWN0ZWQgdm9pZCBsb2FkRnJvbU1lbWVudG8oSU1lbWVudG8gbWVtZW50bywgSVByb2dyZXNzTW9uaXRvciBtb25pdG9yKSB7CgkJbG9hZChtZW1lbnRvKTsKCX0KCQoJLyoqCgkgKiAKCSAqLwoJcHJvdGVjdGVkIHZvaWQgbG9hZEZyb21QYXRoKElQYXRoIHBhdGgsIElQcm9ncmVzc01vbml0b3IgbW9uaXRvcikgdGhyb3dzIENvcmVFeGNlcHRpb24gewoJCUZpbGVJbnB1dFN0cmVhbSBmaW4gPSBudWxsOwoJCXRyeSB7CgkJCWZpbiA9IG5ldyBGaWxlSW5wdXRTdHJlYW0ocGF0aC50b0ZpbGUoKSk7CgkJCUlNZW1lbnRvIG1lbWVudG8gPSBYTUxNZW1lbnRvLmxvYWRNZW1lbnRvKGZpbik7CgkJCWxvYWQobWVtZW50byk7CgkJfSBjYXRjaCAoRXhjZXB0aW9uIGUpIHsKCQkJVHJhY2UudHJhY2UoVHJhY2UuU0VWRVJFLCAiQ291bGQgbm90IGxvYWQgZnJvbSBwYXRoOiAiICsgZS5nZXRNZXNzYWdlKCksIGUpOwoJCQl0aHJvdyBuZXcgQ29yZUV4Y2VwdGlvbihuZXcgU3RhdHVzKElTdGF0dXMuRVJST1IsIFNlcnZlclBsdWdpbi5QTFVHSU5fSUQsIDAsIFNlcnZlclBsdWdpbi5nZXRSZXNvdXJjZSgiJWVycm9yTG9hZGluZyIsIHBhdGgudG9TdHJpbmcoKSksIGUpKTsKCQl9IGZpbmFsbHkgewoJCQl0cnkgewoJCQkJZmluLmNsb3NlKCk7CgkJCX0gY2F0Y2ggKEV4Y2VwdGlvbiBlKSB7CgkJCQkvLyBpZ25vcmUKCQkJfQoJCX0KCX0KCQoJcHVibGljIElTdGF0dXMgdmFsaWRhdGVFZGl0KE9iamVjdCBjb250ZXh0KSB7CgkJaWYgKGZpbGUgPT0gbnVsbCkKCQkJcmV0dXJuIG51bGw7CgkKCQlyZXR1cm4gZmlsZS5nZXRXb3Jrc3BhY2UoKS52YWxpZGF0ZUVkaXQobmV3IElGaWxlW10geyBmaWxlIH0sIGNvbnRleHQpOwoJfQp9