LyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogQ29weXJpZ2h0IChjKSAyMDAzLCAyMDA1IElCTSBDb3Jwb3JhdGlvbiBhbmQgb3RoZXJzLgogKiBBbGwgcmlnaHRzIHJlc2VydmVkLiBUaGlzIHByb2dyYW0gYW5kIHRoZSBhY2NvbXBhbnlpbmcgbWF0ZXJpYWxzCiAqIGFyZSBtYWRlIGF2YWlsYWJsZSB1bmRlciB0aGUgdGVybXMgb2YgdGhlIEVjbGlwc2UgUHVibGljIExpY2Vuc2UgdjEuMAogKiB3aGljaCBhY2NvbXBhbmllcyB0aGlzIGRpc3RyaWJ1dGlvbiwgYW5kIGlzIGF2YWlsYWJsZSBhdAogKiBodHRwOi8vd3d3LmVjbGlwc2Uub3JnL2xlZ2FsL2VwbC12MTAuaHRtbAqgKgogKiBDb250cmlidXRvcnM6CiAqICAgIElCTSBDb3Jwb3JhdGlvbiAtIEluaXRpYWwgQVBJIGFuZCBpbXBsZW1lbnRhdGlvbgogKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKi8KcGFja2FnZSBvcmcuZWNsaXBzZS53c3QuaW50ZXJuZXQubW9uaXRvci5jb3JlLmludGVybmFsOwoKaW1wb3J0IGphdmEudGV4dC5NZXNzYWdlRm9ybWF0OwppbXBvcnQgamF2YS51dGlsLio7CgppbXBvcnQgb3JnLmVjbGlwc2UuY29yZS5ydW50aW1lLio7Ci8qKgogKiBUaGUgbW9uaXRvciBjb3JlIHBsdWdpbi4KICovCnB1YmxpYyBjbGFzcyBNb25pdG9yUGx1Z2luIGV4dGVuZHMgUGx1Z2luIHsKCXB1YmxpYyBzdGF0aWMgZmluYWwgU3RyaW5nIFBMVUdJTl9JRCA9ICJvcmcuZWNsaXBzZS53c3QuaW50ZXJuZXQubW9uaXRvci5jb3JlIjsKCglwcml2YXRlIHN0YXRpYyBNb25pdG9yUGx1Z2luIHNpbmdsZXRvbjsKCQoJcHJvdGVjdGVkIE1hcCBwcm90b2NvbEFkYXB0ZXJzOwoJcHJvdGVjdGVkIE1hcCBjb250ZW50RmlsdGVyczsKCXByb3RlY3RlZCBib29sZWFuIHN0YXJ0dXBzTG9hZGVkOwoKCS8qKgoJICogTW9uaXRvclBsdWdpbiBjb25zdHJ1Y3RvciBjb21tZW50LgoJICovCglwdWJsaWMgTW9uaXRvclBsdWdpbigpIHsKCQlzdXBlcigpOwoJCXNpbmdsZXRvbiA9IHRoaXM7CgkJbG9hZFByb3RvY29sQWRhcHRlcnMoKTsKCQlsb2FkQ29udGVudEZpbHRlcnMoKTsKCX0KCgkvKioKCSAqIFJldHVybnMgdGhlIHNpbmdsZXRvbiBpbnN0YW5jZSBvZiB0aGlzIHBsdWdpbi4KCSAqCgkgKiBAcmV0dXJuIG9yZy5lY2xpcHNlLndzdC5pbnRlcm5ldC5tb25pdG9yLmNvcmUuTW9uaXRvclBsdWdpbgoJICovCglwdWJsaWMgc3RhdGljIE1vbml0b3JQbHVnaW4gZ2V0SW5zdGFuY2UoKSB7CgkJcmV0dXJuIHNpbmdsZXRvbjsKCX0KCgkvKioKCSAqIFJldHVybnMgdGhlIHRyYW5zbGF0ZWQgU3RyaW5nIGZvdW5kIHdpdGggdGhlIGdpdmVuIGtleS4KCSAqCgkgKiBAcmV0dXJuIGphdmEubGFuZy5TdHJpbmcKCSAqIEBwYXJhbSBrZXkgamF2YS5sYW5nLlN0cmluZwoJICovCglwdWJsaWMgc3RhdGljIFN0cmluZyBnZXRSZXNvdXJjZShTdHJpbmcga2V5KSB7CgkJdHJ5IHsKCQkJcmV0dXJuIFBsYXRmb3JtLmdldFJlc291cmNlU3RyaW5nKGdldEluc3RhbmNlKCkuZ2V0QnVuZGxlKCksIGtleSk7CgkJfSBjYXRjaCAoRXhjZXB0aW9uIGUpIHsKCQkJcmV0dXJuIGtleTsKCQl9Cgl9CgkKCS8qKgoJICogUmV0dXJucyB0aGUgdHJhbnNsYXRlZCBTdHJpbmcgZm91bmQgd2l0aCB0aGUgZ2l2ZW4ga2V5LAoJICogYW5kIGZvcm1hdHRlZCB3aXRoIHRoZSBnaXZlbiBhcmd1bWVudHMgdXNpbmcgamF2YS50ZXh0Lk1lc3NhZ2VGb3JtYXQuCgkgKgoJICogQHBhcmFtIGtleSBqYXZhLmxhbmcuU3RyaW5nCgkgKiBAcGFyYW0gYXJndW1lbnRzIGphdmEubGFuZy5PYmplY3RbXQoJICogQHJldHVybiBqYXZhLmxhbmcuU3RyaW5nCgkgKi8KCXB1YmxpYyBzdGF0aWMgU3RyaW5nIGdldFJlc291cmNlKFN0cmluZyBrZXksIE9iamVjdFtdIGFyZ3VtZW50cykgewoJCXRyeSB7CgkJCVN0cmluZyB0ZXh0ID0gZ2V0UmVzb3VyY2Uoa2V5KTsKCQkJcmV0dXJuIE1lc3NhZ2VGb3JtYXQuZm9ybWF0KHRleHQsIGFyZ3VtZW50cyk7CgkJfSBjYXRjaCAoRXhjZXB0aW9uIGUpIHsKCQkJcmV0dXJuIGtleTsKCQl9Cgl9CgkKCS8qKgoJICogUmV0dXJucyB0aGUgdHJhbnNsYXRlZCBTdHJpbmcgZm91bmQgd2l0aCB0aGUgZ2l2ZW4ga2V5LAoJICogYW5kIGZvcm1hdHRlZCB3aXRoIHRoZSBnaXZlbiBhcmd1bWVudHMgdXNpbmcgamF2YS50ZXh0Lk1lc3NhZ2VGb3JtYXQuCgkgKgoJICogQHBhcmFtIGtleSB0aGUga2V5CgkgKiBAcGFyYW0gYXJnIGFuIGFyZ3VtZW50CgkgKiBAcmV0dXJuIHRoZSB0cmFuc2xhdGVkIHN0cmluZwoJICovCglwdWJsaWMgc3RhdGljIFN0cmluZyBnZXRSZXNvdXJjZShTdHJpbmcga2V5LCBTdHJpbmcgYXJnKSB7CgkJcmV0dXJuIGdldFJlc291cmNlKGtleSwgbmV3IFN0cmluZ1tdIHsgYXJnIH0pOwoJfQoJCglwdWJsaWMgU3RyaW5nIGdldERlZmF1bHRUeXBlKCkgewoJCXJldHVybiAiSFRUUCI7Cgl9CgoJLyoqCgkgKiBSZXR1cm5zIHRoZSBwcm90b2NvbCBhZGFwdGVyIHdpdGggdGhlIGdpdmVuIGlkLCBvciA8Y29kZT5udWxsPC9jb2RlPgoJICogaWYgbm9uZS4gVGhpcyBjb252ZW5pZW5jZSBtZXRob2Qgc2VhcmNoZXMgdGhlIGxpc3Qgb2Yga25vd24KCSAqIHByb3RvY29sIGFkYXB0ZXJzICh7QGxpbmsgI2dldFByb3RvY29sQWRhcHRlcnMoKX0pIGZvciB0aGUgb25lIHdpdGggYQoJICogbWF0Y2hpbmcgaWQuCgkgKgoJICogQHBhcmFtIGlkIHRoZSBwcm90b2NvbCBhZGFwdGVyIGlkOyBtdXN0IG5vdCBiZSA8Y29kZT5udWxsPC9jb2RlPgoJICogQHJldHVybiB0aGUgcHJvdG9jb2wgYWRhcHRlciBpbnN0YW5jZSwgb3IgPGNvZGU+bnVsbDwvY29kZT4gaWYgdGhlcmUKCSAqICAgaXMgbm8gcHJvdG9jb2wgYWRhcHRlciB3aXRoIHRoZSBnaXZlbiBpZAoJICovCglwdWJsaWMgUHJvdG9jb2xBZGFwdGVyIGdldFByb3RvY29sQWRhcHRlcihTdHJpbmcgaWQpIHsKCQlyZXR1cm4gKFByb3RvY29sQWRhcHRlcikgcHJvdG9jb2xBZGFwdGVycy5nZXQoaWQpOwoJfQoKCS8qKgoJICogUmV0dXJucyBhIGxpc3Qgb2YgYWxsIGtub3duIHByb3RvY29sIGFkYXB0ZXIgaW5zdGFuY2VzLgoJICogPHA+CgkgKiBQcm90b2NvbCBhZGFwdGVycyBhcmUgcmVnaXN0ZXJlZCB2aWEgdGhlIDxjb2RlPnByb3RvY29sQWRhcGF0ZXJzPC9jb2RlPgoJICogZXh0ZW5zaW9uIHBvaW50IGluIHRoZSA8Y29kZT5vcmcuZWNsaXBzZS53c3QuaW50ZXJuZXQubW9uaXRvci5jb3JlPC9jb2RlPgoJICogcGx1Zy1pbi4KCSAqIDwvcD4KCSAqIDxwPgoJICogQSBuZXcgYXJyYXkgaXMgcmV0dXJuZWQgb24gZWFjaCBjYWxsOyBjbGllbnRzIG1heSBzYWZlbHkgc3RvcmUgb3IgbW9kaWZ5IHRoZSByZXN1bHQuCgkgKiA8L3A+CgkgKiAKCSAqIEByZXR1cm4gYSBwb3NzaWJseS1lbXB0eSBhcnJheSBvZiBwcm90b2NvbCBhZGFwdGVyIGluc3RhbmNlcwoJICovCglwdWJsaWMgUHJvdG9jb2xBZGFwdGVyW10gZ2V0UHJvdG9jb2xBZGFwdGVycygpIHsKCQlMaXN0IGxpc3QgPSBuZXcgQXJyYXlMaXN0KCk7CgkJSXRlcmF0b3IgaXRlcmF0b3IgPSBwcm90b2NvbEFkYXB0ZXJzLnZhbHVlcygpLml0ZXJhdG9yKCk7CgkJd2hpbGUgKGl0ZXJhdG9yLmhhc05leHQoKSkgewoJCQlsaXN0LmFkZChpdGVyYXRvci5uZXh0KCkpOwoJCX0KCQlQcm90b2NvbEFkYXB0ZXJbXSB0eXBlcyA9IG5ldyBQcm90b2NvbEFkYXB0ZXJbbGlzdC5zaXplKCldOwoJCWxpc3QudG9BcnJheSh0eXBlcyk7CgkJcmV0dXJuIHR5cGVzOwoJfQoKCXB1YmxpYyBJQ29udGVudEZpbHRlcltdIGdldENvbnRlbnRGaWx0ZXJzKCkgewoJCUxpc3QgbGlzdCA9IG5ldyBBcnJheUxpc3QoKTsKCQlJdGVyYXRvciBpdGVyYXRvciA9IGNvbnRlbnRGaWx0ZXJzLnZhbHVlcygpLml0ZXJhdG9yKCk7CgkJd2hpbGUgKGl0ZXJhdG9yLmhhc05leHQoKSkgewoJCQlsaXN0LmFkZChpdGVyYXRvci5uZXh0KCkpOwoJCX0KCQlJQ29udGVudEZpbHRlcltdIGNmID0gbmV3IElDb250ZW50RmlsdGVyW2xpc3Quc2l6ZSgpXTsKCQlsaXN0LnRvQXJyYXkoY2YpOwoJCXJldHVybiBjZjsKCX0KCQoJcHVibGljIElDb250ZW50RmlsdGVyIGZpbmRDb250ZW50RmlsdGVyKFN0cmluZyBpZCkgewoJCWlmIChpZCA9PSBudWxsKQoJCQl0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKCk7CgkJcmV0dXJuIChJQ29udGVudEZpbHRlcikgY29udGVudEZpbHRlcnMuZ2V0KGlkKTsKCX0KCglwcm90ZWN0ZWQgc3luY2hyb25pemVkIHZvaWQgbG9hZFByb3RvY29sQWRhcHRlcnMoKSB7CgkJaWYgKHByb3RvY29sQWRhcHRlcnMgIT0gbnVsbCkKCQkJcmV0dXJuOwoJCVRyYWNlLnRyYWNlKFRyYWNlLkNPTkZJRywgIkxvYWRpbmcgcHJvdG9jb2wgYWRhcHRlcnMiKTsgCgkJSUV4dGVuc2lvblJlZ2lzdHJ5IHJlZ2lzdHJ5ID0gUGxhdGZvcm0uZ2V0RXh0ZW5zaW9uUmVnaXN0cnkoKTsKCQlJQ29uZmlndXJhdGlvbkVsZW1lbnRbXSBjZiA9IHJlZ2lzdHJ5LmdldENvbmZpZ3VyYXRpb25FbGVtZW50c0ZvcihNb25pdG9yUGx1Z2luLlBMVUdJTl9JRCwgImludGVybmFsUHJvdG9jb2xBZGFwdGVycyIpOwoKCQlpbnQgc2l6ZSA9IGNmLmxlbmd0aDsKCQlwcm90b2NvbEFkYXB0ZXJzID0gbmV3IEhhc2hNYXAoc2l6ZSk7CgkJZm9yIChpbnQgaSA9IDA7IGkgPCBzaXplOyBpKyspIHsKCQkJU3RyaW5nIGlkID0gY2ZbaV0uZ2V0QXR0cmlidXRlKCJpZCIpOwoJCQlUcmFjZS50cmFjZShUcmFjZS5DT05GSUcsICJMb2FkaW5nIGFkYXB0ZXI6ICIgKyBpZCk7CgkJCXByb3RvY29sQWRhcHRlcnMucHV0KGlkLCBuZXcgUHJvdG9jb2xBZGFwdGVyKGNmW2ldKSk7CgkJfQoJfQoJCglwcm90ZWN0ZWQgc3luY2hyb25pemVkIHZvaWQgbG9hZENvbnRlbnRGaWx0ZXJzKCkgewoJCWlmIChjb250ZW50RmlsdGVycyAhPSBudWxsKQoJCQlyZXR1cm47CgkJVHJhY2UudHJhY2UoVHJhY2UuQ09ORklHLCAiTG9hZGluZyBjb250ZW50IGZpbHRlcnMiKTsgCgkJSUV4dGVuc2lvblJlZ2lzdHJ5IHJlZ2lzdHJ5ID0gUGxhdGZvcm0uZ2V0RXh0ZW5zaW9uUmVnaXN0cnkoKTsKCQlJQ29uZmlndXJhdGlvbkVsZW1lbnRbXSBjZiA9IHJlZ2lzdHJ5LmdldENvbmZpZ3VyYXRpb25FbGVtZW50c0ZvcihNb25pdG9yUGx1Z2luLlBMVUdJTl9JRCwgImNvbnRlbnRGaWx0ZXJzIik7CgoJCWludCBzaXplID0gY2YubGVuZ3RoOwoJCWNvbnRlbnRGaWx0ZXJzID0gbmV3IEhhc2hNYXAoc2l6ZSk7CgkJZm9yIChpbnQgaSA9IDA7IGkgPCBzaXplOyBpKyspIHsKCQkJU3RyaW5nIGlkID0gY2ZbaV0uZ2V0QXR0cmlidXRlKCJpZCIpOwoJCQlUcmFjZS50cmFjZShUcmFjZS5DT05GSUcsICJMb2FkaW5nIGZpbHRlcjogIiArIGlkKTsKCQkJY29udGVudEZpbHRlcnMucHV0KGlkLCBuZXcgQ29udGVudEZpbHRlcihjZltpXSkpOwoJCX0KCX0KCQoJcHJvdGVjdGVkIHN5bmNocm9uaXplZCB2b2lkIGV4ZWN1dGVTdGFydHVwcygpIHsKCQlpZiAoc3RhcnR1cHNMb2FkZWQpCgkJCXJldHVybjsKCQkKCQlUcmFjZS50cmFjZShUcmFjZS5DT05GSUcsICJMb2FkaW5nIHN0YXJ0dXBzIik7IAoJCUlFeHRlbnNpb25SZWdpc3RyeSByZWdpc3RyeSA9IFBsYXRmb3JtLmdldEV4dGVuc2lvblJlZ2lzdHJ5KCk7CgkJSUNvbmZpZ3VyYXRpb25FbGVtZW50W10gY2YgPSByZWdpc3RyeS5nZXRDb25maWd1cmF0aW9uRWxlbWVudHNGb3IoTW9uaXRvclBsdWdpbi5QTFVHSU5fSUQsICJpbnRlcm5hbFN0YXJ0dXAiKTsKCgkJaW50IHNpemUgPSBjZi5sZW5ndGg7CgkJZm9yIChpbnQgaSA9IDA7IGkgPCBzaXplOyBpKyspIHsKCQkJU3RyaW5nIGlkID0gY2ZbaV0uZ2V0QXR0cmlidXRlKCJpZCIpOwoJCQlUcmFjZS50cmFjZShUcmFjZS5DT05GSUcsICJMb2FkaW5nIHN0YXJ0dXA6ICIgKyBpZCk7CgkJCXRyeSB7CgkJCQlJU3RhcnR1cCBzdGFydHVwID0gKElTdGFydHVwKSBjZltpXS5jcmVhdGVFeGVjdXRhYmxlRXh0ZW5zaW9uKCJjbGFzcyIpOwoJCQkJdHJ5IHsKCQkJCQlzdGFydHVwLnN0YXJ0dXAoKTsKCQkJCX0gY2F0Y2ggKEV4Y2VwdGlvbiBleCkgewoJCQkJCVRyYWNlLnRyYWNlKFRyYWNlLlNFVkVSRSwgIlN0YXJ0dXAgZmFpbGVkIiArIHN0YXJ0dXAudG9TdHJpbmcoKSwgZXgpOwoJCQkJfQoJCQl9IGNhdGNoIChFeGNlcHRpb24gZSkgewoJCQkJVHJhY2UudHJhY2UoVHJhY2UuU0VWRVJFLCAiQ291bGQgbm90IGNyZWF0ZSBzdGFydHVwOiAiICsgaWQsIGUpOwoJCQl9CgkJfQoJfQp9