LyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogQ29weXJpZ2h0IChjKSAyMDAzLCAyMDA0IElCTSBDb3Jwb3JhdGlvbiBhbmQgb3RoZXJzLgogKiBBbGwgcmlnaHRzIHJlc2VydmVkLiBUaGlzIHByb2dyYW0gYW5kIHRoZSBhY2NvbXBhbnlpbmcgbWF0ZXJpYWxzCiAqIGFyZSBtYWRlIGF2YWlsYWJsZSB1bmRlciB0aGUgdGVybXMgb2YgdGhlIENvbW1vbiBQdWJsaWMgTGljZW5zZSB2MS4wCiAqIHdoaWNoIGFjY29tcGFuaWVzIHRoaXMgZGlzdHJpYnV0aW9uLCBhbmQgaXMgYXZhaWxhYmxlIGF0CiAqIGh0dHA6Ly93d3cuZWNsaXBzZS5vcmcvbGVnYWwvY3BsLXYxMC5odG1sCqAqCiAqIENvbnRyaWJ1dG9yczoKICogICAgIElCTSBDb3Jwb3JhdGlvbiAtIEluaXRpYWwgQVBJIGFuZCBpbXBsZW1lbnRhdGlvbgogKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKi8KcGFja2FnZSBvcmcuZWNsaXBzZS53c3Quc2VydmVyLmNvcmUuaW50ZXJuYWw7CgppbXBvcnQgamF2YS5pby4qOwppbXBvcnQgamF2YS51dGlsLio7CmltcG9ydCBqYXZhLnRleHQuQ29sbGF0b3I7CmltcG9ydCBqYXZhLnRleHQuRGF0ZUZvcm1hdDsKaW1wb3J0IGphdmEudGV4dC5NZXNzYWdlRm9ybWF0OwoKaW1wb3J0IG9yZy5lY2xpcHNlLmNvcmUucnVudGltZS4qOwppbXBvcnQgb3JnLmVjbGlwc2Uud3N0LnNlcnZlci5jb3JlLklSdW50aW1lOwppbXBvcnQgb3JnLmVjbGlwc2Uud3N0LnNlcnZlci5jb3JlLklTZXJ2ZXI7CmltcG9ydCBvcmcuZWNsaXBzZS53c3Quc2VydmVyLmNvcmUudXRpbC4qOwppbXBvcnQgb3JnLm9zZ2kuZnJhbWV3b3JrLkJ1bmRsZUNvbnRleHQ7Ci8qKgogKiBUaGUgbWFpbiBzZXJ2ZXIgcGx1Z2luIGNsYXNzLgogKi8KcHVibGljIGNsYXNzIFNlcnZlclBsdWdpbiBleHRlbmRzIFBsdWdpbiB7CglwdWJsaWMgc3RhdGljIGZpbmFsIFN0cmluZyBQUk9KRUNUX1BSRUZfRklMRSA9ICIuc2VydmVyUHJlZmVyZW5jZSI7CgkKCXByb3RlY3RlZCBzdGF0aWMgZmluYWwgRGF0ZUZvcm1hdCBkZiA9IERhdGVGb3JtYXQuZ2V0RGF0ZVRpbWVJbnN0YW5jZShEYXRlRm9ybWF0LlNIT1JULCBEYXRlRm9ybWF0LlNIT1JUKTsKCXByb3RlY3RlZCBzdGF0aWMgaW50IG51bSA9IDA7CgoJLy8gc2luZ2xldG9uIGluc3RhbmNlIG9mIHRoaXMgY2xhc3MKCXByaXZhdGUgc3RhdGljIFNlcnZlclBsdWdpbiBzaW5nbGV0b247CgoJcHJpdmF0ZSBzdGF0aWMgZmluYWwgU3RyaW5nIFRFTVBfREFUQV9GSUxFID0gInRtcC1kYXRhLnhtbCI7CgoJY2xhc3MgVGVtcERpciB7CgkJU3RyaW5nIHBhdGg7CgkJaW50IGFnZTsKCX0KCgkvLyB0ZW1wIGRpcmVjdG9yaWVzIC0gU3RyaW5nIGtleSB0byBUZW1wRGlyCglwcm90ZWN0ZWQgTWFwIHRlbXBEaXJIYXNoOwoKCS8qKgoJICogc2VydmVyIGNvcmUgcGx1Z2luIGlkCgkgKi8KCXB1YmxpYyBzdGF0aWMgZmluYWwgU3RyaW5nIFBMVUdJTl9JRCA9ICJvcmcuZWNsaXBzZS53c3Quc2VydmVyLmNvcmUiOwoKCS8qKgoJICogQ3JlYXRlIHRoZSBTZXJ2ZXJQbHVnaW4uCgkgKi8KCXB1YmxpYyBTZXJ2ZXJQbHVnaW4oKSB7CgkJc3VwZXIoKTsKCQlzaW5nbGV0b24gPSB0aGlzOwoJfQoKCS8qKgoJICogUmV0dXJucyB0aGUgc2luZ2xldG9uIGluc3RhbmNlIG9mIHRoaXMgcGx1Z2luLgoJICoKCSAqIEByZXR1cm4gb3JnLmVjbGlwc2Uud3N0LnNlcnZlci5jb3JlLmludGVybmFsLnBsdWdpbi5TZXJ2ZXJQbHVnaW4KCSAqLwoJcHVibGljIHN0YXRpYyBTZXJ2ZXJQbHVnaW4gZ2V0SW5zdGFuY2UoKSB7CgkJcmV0dXJuIHNpbmdsZXRvbjsKCX0KCgkvKioKCSAqIFJldHVybnMgdGhlIHRyYW5zbGF0ZWQgU3RyaW5nIGZvdW5kIHdpdGggdGhlIGdpdmVuIGtleS4KCSAqCgkgKiBAcGFyYW0ga2V5IGphdmEubGFuZy5TdHJpbmcKCSAqIEByZXR1cm4gamF2YS5sYW5nLlN0cmluZwoJICovCglwdWJsaWMgc3RhdGljIFN0cmluZyBnZXRSZXNvdXJjZShTdHJpbmcga2V5KSB7CgkJdHJ5IHsKCQkJcmV0dXJuIFBsYXRmb3JtLmdldFJlc291cmNlU3RyaW5nKGdldEluc3RhbmNlKCkuZ2V0QnVuZGxlKCksIGtleSk7CgkJfSBjYXRjaCAoRXhjZXB0aW9uIGUpIHsKCQkJcmV0dXJuIGtleTsKCQl9Cgl9CgoJLyoqCgkgKiBSZXR1cm5zIHRoZSB0cmFuc2xhdGVkIFN0cmluZyBmb3VuZCB3aXRoIHRoZSBnaXZlbiBrZXksCgkgKiBhbmQgZm9ybWF0dGVkIHdpdGggdGhlIGdpdmVuIGFyZ3VtZW50cyB1c2luZyBqYXZhLnRleHQuTWVzc2FnZUZvcm1hdC4KCSAqCgkgKiBAcGFyYW0ga2V5IGphdmEubGFuZy5TdHJpbmcKCSAqIEBwYXJhbSBhcmd1bWVudHMgamF2YS5sYW5nLk9iamVjdFtdCgkgKiBAcmV0dXJuIGphdmEubGFuZy5TdHJpbmcKCSAqLwoJcHVibGljIHN0YXRpYyBTdHJpbmcgZ2V0UmVzb3VyY2UoU3RyaW5nIGtleSwgT2JqZWN0W10gYXJndW1lbnRzKSB7CgkJdHJ5IHsKCQkJU3RyaW5nIHRleHQgPSBnZXRSZXNvdXJjZShrZXkpOwoJCQlyZXR1cm4gTWVzc2FnZUZvcm1hdC5mb3JtYXQodGV4dCwgYXJndW1lbnRzKTsKCQl9IGNhdGNoIChFeGNlcHRpb24gZSkgewoJCQlyZXR1cm4ga2V5OwoJCX0KCX0KCQoJLyoqCgkgKiBSZXR1cm5zIHRoZSB0cmFuc2xhdGVkIFN0cmluZyBmb3VuZCB3aXRoIHRoZSBnaXZlbiBrZXksCgkgKiBhbmQgZm9ybWF0dGVkIHdpdGggdGhlIGdpdmVuIGFyZ3VtZW50cyB1c2luZyBqYXZhLnRleHQuTWVzc2FnZUZvcm1hdC4KCSAqCgkgKiBAcGFyYW0ga2V5IGphdmEubGFuZy5TdHJpbmcKCSAqIEBwYXJhbSBhcmd1bWVudHMgamF2YS5sYW5nLk9iamVjdFtdCgkgKiBAcmV0dXJuIGphdmEubGFuZy5TdHJpbmcKCSAqLwoJcHVibGljIHN0YXRpYyBTdHJpbmcgZ2V0UmVzb3VyY2UoU3RyaW5nIGtleSwgU3RyaW5nIGFyZykgewoJCXJldHVybiBnZXRSZXNvdXJjZShrZXksIG5ldyBTdHJpbmdbXSB7IGFyZyB9KTsKCX0KCgkvKioKCSAqIFJldHVybnMgYSB0ZW1wb3JhcnkgZGlyZWN0b3J5IHRoYXQgdGhlIHJlcXVlc3RvciBjYW4gdXNlCgkgKiB0aHJvdWdob3V0IGl0J3MgbGlmZWN5Y2xlLiBUaGlzIGlzIHByaW1hcnkgdG8gYmUgdXNlZCBieQoJICogc2VydmVyIGluc3RhbmNlcyBmb3Igd29ya2luZyBkaXJlY3RvcmllcywgaW5zdGFuY2Ugc3BlY2lmaWMKCSAqIGZpbGVzLCBldGMuCgkgKgoJICogPHA+QXMgbG9uZyBhcyB0aGUgc2FtZSBrZXkgaXMgdXNlZCB0byBjYWxsIHRoaXMgbWV0aG9kIG9uCgkgKiBlYWNoIHVzZSBvZiB0aGUgd29ya2JlbmNoLCB0aGlzIG1ldGhvZCBkaXJlY3Rvcnkgd2lsbCByZXR1cm4KCSAqIHRoZSBzYW1lIGRpcmVjdG9yeS4gSWYgdGhlIGRpcmVjdG9yeSBpcyBub3QgcmVxdWVzdGVkIG92ZXIgYQoJICogcGVyaW9kIG9mIHRpbWUsIHRoZSBkaXJlY3RvcnkgbWF5IGJlIGRlbGV0ZWQgYW5kIGEgbmV3IG9uZQoJICogd2lsbCBiZSBhc3NpZ25lZCBvbiB0aGUgbmV4dCByZXF1ZXN0LiBGb3IgdGhpcyByZWFzb24sIGEKCSAqIHNlcnZlciBpbnN0YW5jZSBzaG91bGQgcmVxdWVzdCB0aGUgdGVtcCBkaXJlY3Rvcnkgb24gc3RhcnR1cAoJICogaWYgaXQgd2FudHMgdG8gc3RvcmUgZmlsZXMgdGhlcmUuIEluIGFsbCBjYXNlcywgdGhlIGluc3RhbmNlCgkgKiBzaG91bGQgaGF2ZSBhIGJhY2t1cCBwbGFuIGFueXdheSwgYXMgdGhpcyBkaXJlY3RvcnkgbWF5IGJlCgkgKiBkZWxldGVkIGFjY2lkZW50YWxseS48L3A+CgkgKgoJICogQHBhcmFtIGtleQoJICogQHJldHVybiBqYXZhLmlvLkZpbGUKCSAqLwoJcHVibGljIElQYXRoIGdldFRlbXBEaXJlY3RvcnkoU3RyaW5nIGtleSkgewoJCWlmIChrZXkgPT0gbnVsbCkKCQkJcmV0dXJuIG51bGw7CgkKCQkvLyBmaXJzdCwgbG9vayB0aHJvdWdoIGhhc2ggb2YgY3VycmVudCBkaXJlY3RvcmllcwoJCUlQYXRoIHN0YXRlUGF0aCA9IFNlcnZlclBsdWdpbi5nZXRJbnN0YW5jZSgpLmdldFN0YXRlTG9jYXRpb24oKTsKCQl0cnkgewoJCQlUZW1wRGlyIGRpciA9IChUZW1wRGlyKSB0ZW1wRGlySGFzaC5nZXQoa2V5KTsKCQkJaWYgKGRpciAhPSBudWxsKSB7CgkJCQlkaXIuYWdlID0gMDsKCQkJCXJldHVybiBzdGF0ZVBhdGguYXBwZW5kKGRpci5wYXRoKTsKCQkJfQoJCX0gY2F0Y2ggKEV4Y2VwdGlvbiBlKSB7CgkJCS8vIGlnbm9yZQoJCX0KCQoJCS8vIG90aGVyd2lzZSwgY3JlYXRlIGEgbmV3IGRpcmVjdG9yeQoJCgkJLy8gZmluZCBmaXJzdCBmcmVlIGRpcmVjdG9yeQoJCVN0cmluZyBwYXRoID0gbnVsbDsKCQlGaWxlIGRpciA9IG51bGw7CgkJaW50IGNvdW50ID0gMDsKCQl3aGlsZSAoZGlyID09IG51bGwgfHwgZGlyLmV4aXN0cygpKSB7CgkJCXBhdGggPSAidG1wIiArIGNvdW50OwoJCQlkaXIgPSBzdGF0ZVBhdGguYXBwZW5kKHBhdGgpLnRvRmlsZSgpOwoJCQljb3VudCArKzsKCQl9CgkKCQlkaXIubWtkaXJzKCk7CgkKCQlUZW1wRGlyIGQgPSBuZXcgVGVtcERpcigpOwoJCWQucGF0aCA9IHBhdGg7CgkJdGVtcERpckhhc2gucHV0KGtleSwgZCk7CgkJc2F2ZVRlbXBEaXJJbmZvKCk7CgkJcmV0dXJuIHN0YXRlUGF0aC5hcHBlbmQocGF0aCk7Cgl9CgkKCS8qKgoJICogUmVtb3ZlIGEgdGVtcCBkaXJlY3RvcnkuCgkgKiBAcGFyYW0ga2V5CgkgKi8KCXB1YmxpYyB2b2lkIHJlbW92ZVRlbXBEaXJlY3RvcnkoU3RyaW5nIGtleSwgSVByb2dyZXNzTW9uaXRvciBtb25pdG9yKSB7CgkJaWYgKGtleSA9PSBudWxsKQoJCQlyZXR1cm47CgkJCgkJSVBhdGggc3RhdGVQYXRoID0gU2VydmVyUGx1Z2luLmdldEluc3RhbmNlKCkuZ2V0U3RhdGVMb2NhdGlvbigpOwoJCXRyeSB7CgkJCVRlbXBEaXIgZGlyID0gKFRlbXBEaXIpIHRlbXBEaXJIYXNoLmdldChrZXkpOwoJCQlpZiAoZGlyICE9IG51bGwpIHsKCQkJCXRlbXBEaXJIYXNoLnJlbW92ZShrZXkpOwoJCQkJc2F2ZVRlbXBEaXJJbmZvKCk7CgkJCQlGaWxlVXRpbC5kZWxldGVEaXJlY3Rvcnkoc3RhdGVQYXRoLmFwcGVuZChkaXIucGF0aCkudG9GaWxlKCksIG1vbml0b3IpOwoJCQl9CgkJfSBjYXRjaCAoRXhjZXB0aW9uIGUpIHsKCQkJVHJhY2UudHJhY2UoVHJhY2UuV0FSTklORywgIkNvdWxkIG5vdCByZW1vdmUgdGVtcCBkaXJlY3RvcnkiLCBlKTsKCQl9Cgl9CgkKCS8qKgoJICogTG9hZCB0aGUgdGVtcG9yYXJ5IGRpcmVjdG9yeSBpbmZvcm1hdGlvbi4KCSAqLwoJcHJpdmF0ZSB2b2lkIGxvYWRUZW1wRGlySW5mbygpIHsKCQlUcmFjZS50cmFjZShUcmFjZS5GSU5FU1QsICJMb2FkaW5nIHRlbXBvcmFyeSBkaXJlY3RvcnkgaW5mb3JtYXRpb24iKTsKCQlJUGF0aCBzdGF0ZVBhdGggPSBTZXJ2ZXJQbHVnaW4uZ2V0SW5zdGFuY2UoKS5nZXRTdGF0ZUxvY2F0aW9uKCk7CgkJU3RyaW5nIGZpbGVuYW1lID0gc3RhdGVQYXRoLmFwcGVuZChURU1QX0RBVEFfRklMRSkudG9PU1N0cmluZygpOwoJCgkJdGVtcERpckhhc2ggPSBuZXcgSGFzaE1hcCgpOwoJCXRyeSB7CgkJCUlNZW1lbnRvIG1lbWVudG8gPSBYTUxNZW1lbnRvLmxvYWRNZW1lbnRvKGZpbGVuYW1lKTsKCQoJCQlJTWVtZW50b1tdIGNoaWxkcmVuID0gbWVtZW50by5nZXRDaGlsZHJlbigidGVtcC1kaXJlY3RvcnkiKTsKCQkJaW50IHNpemUgPSBjaGlsZHJlbi5sZW5ndGg7CgkJCWZvciAoaW50IGkgPSAwOyBpIDwgc2l6ZTsgaSsrKSB7CgkJCQlTdHJpbmcga2V5ID0gY2hpbGRyZW5baV0uZ2V0U3RyaW5nKCJrZXkiKTsKCQoJCQkJVGVtcERpciBkID0gbmV3IFRlbXBEaXIoKTsKCQkJCWQucGF0aCA9IGNoaWxkcmVuW2ldLmdldFN0cmluZygicGF0aCIpOwoJCQkJZC5hZ2UgPSBjaGlsZHJlbltpXS5nZXRJbnRlZ2VyKCJhZ2UiKS5pbnRWYWx1ZSgpOwoJCQkJZC5hZ2UrKzsKCQoJCQkJdGVtcERpckhhc2gucHV0KGtleSwgZCk7CgkJCX0KCQl9IGNhdGNoIChFeGNlcHRpb24gZSkgewoJCQlUcmFjZS50cmFjZShUcmFjZS5TRVZFUkUsICJDb3VsZCBub3QgbG9hZCB0ZW1wb3JhcnkgZGlyZWN0b3J5IGluZm9ybWF0aW9uOiAiICsgZS5nZXRNZXNzYWdlKCkpOwoJCX0KCX0KCgkvKioKCSAqIENvbnZlbmllbmNlIG1ldGhvZCBmb3IgbG9nZ2luZy4KCSAqCgkgKiBAcGFyYW0gc3RhdHVzIG9yZy5lY2xpcHNlLmNvcmUucnVudGltZS5JU3RhdHVzCgkgKi8KCXB1YmxpYyBzdGF0aWMgdm9pZCBsb2coSVN0YXR1cyBzdGF0dXMpIHsKCQlnZXRJbnN0YW5jZSgpLmdldExvZygpLmxvZyhzdGF0dXMpOwoJfQoKCS8qKgoJICogU2F2ZSB0aGUgdGVtcG9yYXJ5IGRpcmVjdG9yeSBpbmZvcm1hdGlvbi4KCSAqLwoJcHJpdmF0ZSB2b2lkIHNhdmVUZW1wRGlySW5mbygpIHsKCQkvLyBzYXZlIHJlbWFpbmluZyBkaXJlY3RvcmllcwoJCUlQYXRoIHN0YXRlUGF0aCA9IFNlcnZlclBsdWdpbi5nZXRJbnN0YW5jZSgpLmdldFN0YXRlTG9jYXRpb24oKTsKCQlTdHJpbmcgZmlsZW5hbWUgPSBzdGF0ZVBhdGguYXBwZW5kKFRFTVBfREFUQV9GSUxFKS50b09TU3RyaW5nKCk7CgkKCQl0cnkgewoJCQlYTUxNZW1lbnRvIG1lbWVudG8gPSBYTUxNZW1lbnRvLmNyZWF0ZVdyaXRlUm9vdCgidGVtcC1kaXJlY3RvcmllcyIpOwoJCgkJCUl0ZXJhdG9yIGl0ZXJhdG9yID0gdGVtcERpckhhc2gua2V5U2V0KCkuaXRlcmF0b3IoKTsKCQkJd2hpbGUgKGl0ZXJhdG9yLmhhc05leHQoKSkgewoJCQkJU3RyaW5nIGtleSA9IChTdHJpbmcpIGl0ZXJhdG9yLm5leHQoKTsKCQkJCVRlbXBEaXIgZCA9IChUZW1wRGlyKSB0ZW1wRGlySGFzaC5nZXQoa2V5KTsKCQoJCQkJaWYgKGQuYWdlIDwgNSkgewoJCQkJCUlNZW1lbnRvIGNoaWxkID0gbWVtZW50by5jcmVhdGVDaGlsZCgidGVtcC1kaXJlY3RvcnkiKTsKCQkJCQljaGlsZC5wdXRTdHJpbmcoImtleSIsIGtleSk7CgkJCQkJY2hpbGQucHV0U3RyaW5nKCJwYXRoIiwgZC5wYXRoKTsKCQkJCQljaGlsZC5wdXRJbnRlZ2VyKCJhZ2UiLCBkLmFnZSk7CgkJCQl9IGVsc2UgewoJCQkJCUZpbGVVdGlsLmRlbGV0ZURpcmVjdG9yeShzdGF0ZVBhdGguYXBwZW5kKGQucGF0aCkudG9GaWxlKCksIG5ldyBOdWxsUHJvZ3Jlc3NNb25pdG9yKCkpOwoJCQkJfQoJCQl9CgkKCQkJbWVtZW50by5zYXZlVG9GaWxlKGZpbGVuYW1lKTsKCQl9IGNhdGNoIChFeGNlcHRpb24gZSkgewoJCQlUcmFjZS50cmFjZShUcmFjZS5TRVZFUkUsICJDb3VsZCBub3Qgc2F2ZSB0ZW1wb3JhcnkgZGlyZWN0b3J5IGluZm9ybWF0aW9uIiwgZSk7CgkJfQoJfQoJCglwcm90ZWN0ZWQgdm9pZCBpbml0aWFsaXplRGVmYXVsdFBsdWdpblByZWZlcmVuY2VzKCkgewoJCVNlcnZlclByZWZlcmVuY2VzLmdldFNlcnZlclByZWZlcmVuY2VzKCkuc2V0RGVmYXVsdHMoKTsKCX0KCgkvKioKCSAqIFN0YXJ0IHVwIHRoaXMgcGx1Zy1pbi4KCSAqLwoJcHVibGljIHZvaWQgc3RhcnQoQnVuZGxlQ29udGV4dCBjb250ZXh0KSB0aHJvd3MgRXhjZXB0aW9uIHsKCQlUcmFjZS50cmFjZShUcmFjZS5DT05GSUcsICItLS0tLT4tLS0tLSBTZXJ2ZXIgQ29yZSBwbHVnaW4gc3RhcnR1cCAtLS0tLT4tLS0tLSIpOwoJCXN1cGVyLnN0YXJ0KGNvbnRleHQpOwoJCQoJCWluaXRpYWxpemVEZWZhdWx0UGx1Z2luUHJlZmVyZW5jZXMoKTsKCgkJLy8gbG9hZCB0ZW1wIGRpcmVjdG9yeSBpbmZvcm1hdGlvbgoJCWxvYWRUZW1wRGlySW5mbygpOwoJfQoKCS8qKgoJICogU2h1dHMgZG93biB0aGlzIHBsdWctaW4gYW5kIHNhdmVzIGFsbCBwbHVnLWluIHN0YXRlLgoJICovCglwdWJsaWMgdm9pZCBzdG9wKEJ1bmRsZUNvbnRleHQgY29udGV4dCkgdGhyb3dzIEV4Y2VwdGlvbiB7CgkJVHJhY2UudHJhY2UoVHJhY2UuQ09ORklHLCAiLS0tLS08LS0tLS0gU2VydmVyIENvcmUgcGx1Z2luIHNodXRkb3duIC0tLS0tPC0tLS0tIik7CgkJc3VwZXIuc3RvcChjb250ZXh0KTsKCQkKCQlSZXNvdXJjZU1hbmFnZXIuc2h1dGRvd24oKTsKCQlTZXJ2ZXJNb25pdG9yTWFuYWdlci5zaHV0ZG93bigpOwoJfQoJCgkvKioKCSAqIFNvcnQgdGhlIGdpdmVuIHJ1bnRpbWVzIGludG8gYWxwaGFiZXRpY2FsIG9yZGVyLiBUaGlzCgkgKiBtZXRob2QgbW9kaWZpZXMgdGhlIG9yaWdpbmFsIGxpc3QsIGJ1dCByZXR1cm5zIHRoZSB2YWx1ZQoJICogZm9yIGNvbnZlbmllbmNlLgoJICoKCSAqIEBwYXJhbSBsaXN0IGphdmEudXRpbC5MaXN0CgkgKiBAcmV0dXJuIGphdmEudXRpbC5MaXN0CgkgKi8KCXB1YmxpYyBzdGF0aWMgTGlzdCBzb3J0UnVudGltZUxpc3QoTGlzdCBsaXN0KSB7CgkJaWYgKGxpc3QgPT0gbnVsbCkKCQkJcmV0dXJuIG51bGw7CgkKCQlDb2xsYXRvciBjb2xsYXRvciA9IENvbGxhdG9yLmdldEluc3RhbmNlKCk7CgkKCQlpbnQgc2l6ZSA9IGxpc3Quc2l6ZSgpOwoJCWZvciAoaW50IGkgPSAwOyBpIDwgc2l6ZSAtIDE7IGkrKykgewoJCQlmb3IgKGludCBqID0gaSArIDE7IGogPCBzaXplOyBqKyspIHsKCQkJCUlSdW50aW1lIGEgPSAoSVJ1bnRpbWUpIGxpc3QuZ2V0KGkpOwoJCQkJSVJ1bnRpbWUgYiA9IChJUnVudGltZSkgbGlzdC5nZXQoaik7CgkJCQlpZiAoY29sbGF0b3IuY29tcGFyZShhLmdldE5hbWUoKSwgYi5nZXROYW1lKCkpID4gMCkgewoJCQkJCU9iamVjdCB0ZW1wID0gYTsKCQkJCQlsaXN0LnNldChpLCBiKTsKCQkJCQlsaXN0LnNldChqLCB0ZW1wKTsKCQkJCX0KCQkJfQoJCX0KCQlyZXR1cm4gbGlzdDsKCX0KCgkvKioKCSAqIFNvcnQgdGhlIGdpdmVuIHNlcnZlcnMgaW50byBhbHBoYWJldGljYWwgb3JkZXIuIFRoaXMKCSAqIG1ldGhvZCBtb2RpZmllcyB0aGUgb3JpZ2luYWwgbGlzdCwgYnV0IHJldHVybnMgdGhlIHZhbHVlCgkgKiBmb3IgY29udmVuaWVuY2UuCgkgKgoJICogQHBhcmFtIGxpc3QgamF2YS51dGlsLkxpc3QKCSAqIEByZXR1cm4gamF2YS51dGlsLkxpc3QKCSAqLwoJcHVibGljIHN0YXRpYyBMaXN0IHNvcnRTZXJ2ZXJMaXN0KExpc3QgbGlzdCkgewoJCWlmIChsaXN0ID09IG51bGwpCgkJCXJldHVybiBudWxsOwoJCgkJQ29sbGF0b3IgY29sbGF0b3IgPSBDb2xsYXRvci5nZXRJbnN0YW5jZSgpOwoJCgkJaW50IHNpemUgPSBsaXN0LnNpemUoKTsKCQlmb3IgKGludCBpID0gMDsgaSA8IHNpemUgLSAxOyBpKyspIHsKCQkJZm9yIChpbnQgaiA9IGkgKyAxOyBqIDwgc2l6ZTsgaisrKSB7CgkJCQlJU2VydmVyIGEgPSAoSVNlcnZlcikgbGlzdC5nZXQoaSk7CgkJCQlJU2VydmVyIGIgPSAoSVNlcnZlcikgbGlzdC5nZXQoaik7CgkJCQlpZiAoY29sbGF0b3IuY29tcGFyZShhLmdldE5hbWUoKSwgYi5nZXROYW1lKCkpID4gMCkgewoJCQkJCU9iamVjdCB0ZW1wID0gYTsKCQkJCQlsaXN0LnNldChpLCBiKTsKCQkJCQlsaXN0LnNldChqLCB0ZW1wKTsKCQkJCX0KCQkJfQoJCX0KCQlyZXR1cm4gbGlzdDsKCX0KCQoJcHVibGljIHN0YXRpYyBTdHJpbmdbXSB0b2tlbml6ZShTdHJpbmcgcGFyYW0sIFN0cmluZyBkZWxpbSkgewoJCWlmIChwYXJhbSA9PSBudWxsKQoJCQlyZXR1cm4gbmV3IFN0cmluZ1swXTsKCQkKCQlMaXN0IGxpc3QgPSBuZXcgQXJyYXlMaXN0KCk7CgkJCgkJU3RyaW5nVG9rZW5pemVyIHN0ID0gbmV3IFN0cmluZ1Rva2VuaXplcihwYXJhbSwgZGVsaW0pOwoJCXdoaWxlIChzdC5oYXNNb3JlVG9rZW5zKCkpIHsKCQkJU3RyaW5nIHN0ciA9IHN0Lm5leHRUb2tlbigpOwoJCQlpZiAoc3RyICE9IG51bGwgJiYgc3RyLmxlbmd0aCgpID4gMCkKCQkJCWxpc3QuYWRkKHN0ci50cmltKCkpOwoJCX0KCgkJU3RyaW5nW10gcyA9IG5ldyBTdHJpbmdbbGlzdC5zaXplKCldOwoJCWxpc3QudG9BcnJheShzKTsKCQlyZXR1cm4gczsKCX0KCglwcm90ZWN0ZWQgc3RhdGljIExpc3QgZ2V0TW9kdWxlVHlwZXMoSUNvbmZpZ3VyYXRpb25FbGVtZW50W10gZWxlbWVudHMpIHsKCQlMaXN0IGxpc3QgPSBuZXcgQXJyYXlMaXN0KCk7CgkJaWYgKGVsZW1lbnRzID09IG51bGwpCgkJCXJldHVybiBsaXN0OwoJCgkJaW50IHNpemUgPSBlbGVtZW50cy5sZW5ndGg7CgkJZm9yIChpbnQgaSA9IDA7IGkgPCBzaXplOyBpKyspIHsKCQkJU3RyaW5nW10gdHlwZXMgPSB0b2tlbml6ZShlbGVtZW50c1tpXS5nZXRBdHRyaWJ1dGUoInR5cGVzIiksICIsIik7CgkJCVN0cmluZ1tdIHZlcnNpb25zID0gdG9rZW5pemUoZWxlbWVudHNbaV0uZ2V0QXR0cmlidXRlKCJ2ZXJzaW9ucyIpLCAiLCIpOwoJCQlpbnQgc2l6ZVQgPSB0eXBlcy5sZW5ndGg7CgkJCWludCBzaXplViA9IHZlcnNpb25zLmxlbmd0aDsKCQkJZm9yIChpbnQgaiA9IDA7IGogPCBzaXplVDsgaisrKSB7CgkJCQlmb3IgKGludCBrID0gMDsgayA8IHNpemVWOyBrKyspIHsKCQkJCQlNb2R1bGVUeXBlIG1vZHVsZSA9IG5ldyBNb2R1bGVUeXBlKHR5cGVzW2pdLCB2ZXJzaW9uc1trXSk7CgkJCQkJbGlzdC5hZGQobW9kdWxlKTsKCQkJCX0KCQkJfQoJCX0KCQlyZXR1cm4gbGlzdDsKCX0KCQoJcHVibGljIHN0YXRpYyBTdHJpbmcgZ2VuZXJhdGVJZCgpIHsKCQlTdHJpbmcgcyA9IGRmLmZvcm1hdChuZXcgRGF0ZSgpKS50b1N0cmluZygpICsgbnVtKys7CgkJcyA9IHMucmVwbGFjZSgnICcsICdfJyk7CgkJcyA9IHMucmVwbGFjZSgnOicsICdfJyk7CgkJcyA9IHMucmVwbGFjZSgnLycsICdfJyk7CgkJcyA9IHMucmVwbGFjZSgnXFwnLCAnXycpOwoJCXJldHVybiBzOwoJfQoKCS8qKgoJICogUmV0dXJucyB0cnVlIGlmIGlkcyBjb250YWlucyBpZC4KCSAqIEBwYXJhbSBpZHMKCSAqIEBwYXJhbSBpZAoJICogQHJldHVybgoJICovCglwdWJsaWMgc3RhdGljIGJvb2xlYW4gc3VwcG9ydHNUeXBlKFN0cmluZ1tdIGlkcywgU3RyaW5nIGlkKSB7CgkJaWYgKGlkID09IG51bGwgfHwgaWQubGVuZ3RoKCkgPT0gMCkKCQkJcmV0dXJuIGZhbHNlOwoKCQlpZiAoaWRzID09IG51bGwpCgkJCXJldHVybiB0cnVlOwoJCQoJCWludCBzaXplID0gaWRzLmxlbmd0aDsKCQlmb3IgKGludCBpID0gMDsgaSA8IHNpemU7IGkrKykgewoJCQlpZiAoaWRzW2ldLmVuZHNXaXRoKCIqIikpIHsKCQkJCWlmIChpZC5sZW5ndGgoKSA+PSBpZHNbaV0ubGVuZ3RoKCkgJiYgaWQuc3RhcnRzV2l0aChpZHNbaV0uc3Vic3RyaW5nKDAsIGlkc1tpXS5sZW5ndGgoKSAtIDEpKSkKCQkJCQlyZXR1cm4gdHJ1ZTsKCQkJfSBlbHNlIGlmIChpZC5lcXVhbHMoaWRzW2ldKSkKCQkJCXJldHVybiB0cnVlOwoJCX0KCQlyZXR1cm4gZmFsc2U7Cgl9Cn0=