LyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogQ29weXJpZ2h0IChjKSAyMDAzLCAyMDA0IElCTSBDb3Jwb3JhdGlvbiBhbmQgb3RoZXJzLgogKiBBbGwgcmlnaHRzIHJlc2VydmVkLiBUaGlzIHByb2dyYW0gYW5kIHRoZSBhY2NvbXBhbnlpbmcgbWF0ZXJpYWxzCiAqIGFyZSBtYWRlIGF2YWlsYWJsZSB1bmRlciB0aGUgdGVybXMgb2YgdGhlIENvbW1vbiBQdWJsaWMgTGljZW5zZSB2MS4wCiAqIHdoaWNoIGFjY29tcGFuaWVzIHRoaXMgZGlzdHJpYnV0aW9uLCBhbmQgaXMgYXZhaWxhYmxlIGF0CiAqIGh0dHA6Ly93d3cuZWNsaXBzZS5vcmcvbGVnYWwvY3BsLXYxMC5odG1sCqAqCiAqIENvbnRyaWJ1dG9yczoKICogICAgIElCTSBDb3Jwb3JhdGlvbiAtIEluaXRpYWwgQVBJIGFuZCBpbXBsZW1lbnRhdGlvbgogKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKi8KcGFja2FnZSBvcmcuZWNsaXBzZS53c3Quc2VydmVyLmNvcmUuaW50ZXJuYWw7CgppbXBvcnQgamF2YS5pby4qOwppbXBvcnQgamF2YS51dGlsLio7CmltcG9ydCBqYXZhLm5ldC5VUkw7CmltcG9ydCBvcmcudzNjLmRvbS4qOwppbXBvcnQgb3JnLnhtbC5zYXguKjsKCmltcG9ydCBqYXZheC54bWwucGFyc2Vycy4qOwppbXBvcnQgamF2YXgueG1sLnRyYW5zZm9ybS4qOwppbXBvcnQgamF2YXgueG1sLnRyYW5zZm9ybS5kb20uRE9NU291cmNlOwppbXBvcnQgamF2YXgueG1sLnRyYW5zZm9ybS5zdHJlYW0uU3RyZWFtUmVzdWx0OwovKioKICogQSBNZW1lbnRvIGlzIGEgY2xhc3MgaW5kZXBlbmRlbnQgY29udGFpbmVyIGZvciBwZXJzaXN0ZW5jZQogKiBpbmZvLiAgSXQgaXMgYSByZWZsZWN0aW9uIG9mIDMgc3RvcmFnZSByZXF1aXJlbWVudHMuCiAqCiAqIDEpICAgV2UgbmVlZCB0aGUgYWJpbGl0eSB0byBwZXJzaXN0IGFuIG9iamVjdCBhbmQgcmVzdG9yZSBpdC4gIAogKiAyKSAgIFRoZSBjbGFzcyBmb3IgYW4gb2JqZWN0IG1heSBiZSBhYnNlbnQuICBJZiBzbyB3ZSB3b3VsZCAKICogICAgICBsaWtlIHRvIHNraXAgdGhlIG9iamVjdCBhbmQga2VlcCByZWFkaW5nLiAKICogMykgICBUaGUgY2xhc3MgZm9yIGFuIG9iamVjdCBtYXkgY2hhbmdlLiAgSWYgc28gdGhlIG5ldyBjbGFzcyAKICogICAgICBzaG91bGQgYmUgYWJsZSB0byByZWFkIHRoZSBvbGQgcGVyc2lzdGVuY2UgaW5mby4KICoKICogV2UgY291bGQgYXNrIHRoZSBvYmplY3RzIHRvIHNlcmlhbGl6ZSB0aGVtc2VsdmVzIGludG8gYW4gCiAqIE9iamVjdE91dHB1dFN0cmVhbSwgRGF0YU91dHB1dFN0cmVhbSwgb3IgSGFzaHRhYmxlLiAgSG93ZXZlciAKICogYWxsIG9mIHRoZXNlIGFwcHJvYWNoZXMgZmFpbCB0byBtZWV0IHRoZSBzZWNvbmQgcmVxdWlyZW1lbnQuCiAqCiAqIE1lbWVudG8gc3VwcG9ydHMgYmluYXJ5IHBlcnNpc3RhbmNlIHdpdGggYSB2ZXJzaW9uIElELgogKi8KcHVibGljIGZpbmFsIGNsYXNzIFhNTE1lbWVudG8gaW1wbGVtZW50cyBJTWVtZW50byB7Cglwcml2YXRlIERvY3VtZW50IGZhY3Rvcnk7Cglwcml2YXRlIEVsZW1lbnQgZWxlbWVudDsKCgkvKioKCSAqIEFuc3dlciBhIG1lbWVudG8gZm9yIHRoZSBkb2N1bWVudCBhbmQgZWxlbWVudC4gIEZvciBzaW1wbGljaXR5CgkgKiB5b3Ugc2hvdWxkIHVzZSBjcmVhdGVSZWFkUm9vdCBhbmQgY3JlYXRlV3JpdGVSb290IHRvIGNyZWF0ZSB0aGUgaW5pdGlhbAoJICogbWVtZW50b3Mgb24gYSBkb2N1bWVudC4KCSAqLwoJcHVibGljIFhNTE1lbWVudG8oRG9jdW1lbnQgZG9jLCBFbGVtZW50IGVsKSB7CgkJZmFjdG9yeSA9IGRvYzsKCQllbGVtZW50ID0gZWw7Cgl9CgoJLyoqCgkgKiBAc2VlIElNZW1lbnRvLgoJICovCglwdWJsaWMgSU1lbWVudG8gY3JlYXRlQ2hpbGQoU3RyaW5nIHR5cGUpIHsKCQlFbGVtZW50IGNoaWxkID0gZmFjdG9yeS5jcmVhdGVFbGVtZW50KHR5cGUpOwoJCWVsZW1lbnQuYXBwZW5kQ2hpbGQoY2hpbGQpOwoJCXJldHVybiBuZXcgWE1MTWVtZW50byhmYWN0b3J5LCBjaGlsZCk7Cgl9CgoJLyoqCgkgKiBAc2VlIElNZW1lbnRvLgoJICovCglwdWJsaWMgSU1lbWVudG8gY3JlYXRlQ2hpbGQoU3RyaW5nIHR5cGUsIFN0cmluZyBpZCkgewoJCUVsZW1lbnQgY2hpbGQgPSBmYWN0b3J5LmNyZWF0ZUVsZW1lbnQodHlwZSk7CgkJY2hpbGQuc2V0QXR0cmlidXRlKFRBR19JRCwgaWQpOwoJCWVsZW1lbnQuYXBwZW5kQ2hpbGQoY2hpbGQpOwoJCXJldHVybiBuZXcgWE1MTWVtZW50byhmYWN0b3J5LCBjaGlsZCk7Cgl9CgoJLyoqCgkgKiBDcmVhdGUgYSBEb2N1bWVudCBmcm9tIGEgUmVhZGVyIGFuZCBhbnN3ZXIgYSByb290IG1lbWVudG8gZm9yIHJlYWRpbmcgCgkgKiBhIGRvY3VtZW50LgoJICovCglwcm90ZWN0ZWQgc3RhdGljIFhNTE1lbWVudG8gY3JlYXRlUmVhZFJvb3QoUmVhZGVyIHJlYWRlcikgewoJCURvY3VtZW50IGRvY3VtZW50ID0gbnVsbDsKCQl0cnkgewoJCQlEb2N1bWVudEJ1aWxkZXJGYWN0b3J5IGZhY3RvcnkgPSBEb2N1bWVudEJ1aWxkZXJGYWN0b3J5Lm5ld0luc3RhbmNlKCk7CgkJCURvY3VtZW50QnVpbGRlciBwYXJzZXIgPSBmYWN0b3J5Lm5ld0RvY3VtZW50QnVpbGRlcigpOwoJCQlkb2N1bWVudCA9IHBhcnNlci5wYXJzZShuZXcgSW5wdXRTb3VyY2UocmVhZGVyKSk7CgkJCU5vZGUgbm9kZSA9IGRvY3VtZW50LmdldEZpcnN0Q2hpbGQoKTsKCQkJaWYgKG5vZGUgaW5zdGFuY2VvZiBFbGVtZW50KQoJCQkJcmV0dXJuIG5ldyBYTUxNZW1lbnRvKGRvY3VtZW50LCAoRWxlbWVudCkgbm9kZSk7CgkJfSBjYXRjaCAoRXhjZXB0aW9uIGUpIHsKCQkJLy8gaWdub3JlCgkJfSBmaW5hbGx5IHsKCQkJdHJ5IHsKCQkJCXJlYWRlci5jbG9zZSgpOwoJCQl9IGNhdGNoIChFeGNlcHRpb24gZSkgewoJCQkJLy8gaWdub3JlCgkJCX0KCQl9CgkJcmV0dXJuIG51bGw7Cgl9CgkKCS8qKgoJICogQW5zd2VyIGEgcm9vdCBtZW1lbnRvIGZvciB3cml0aW5nIGEgZG9jdW1lbnQuCgkgKi8KCXB1YmxpYyBzdGF0aWMgWE1MTWVtZW50byBjcmVhdGVXcml0ZVJvb3QoU3RyaW5nIHR5cGUpIHsKCQlEb2N1bWVudCBkb2N1bWVudDsKCQl0cnkgewoJCQlkb2N1bWVudCA9IERvY3VtZW50QnVpbGRlckZhY3RvcnkubmV3SW5zdGFuY2UoKS5uZXdEb2N1bWVudEJ1aWxkZXIoKS5uZXdEb2N1bWVudCgpOwoJCQlFbGVtZW50IGVsZW1lbnQgPSBkb2N1bWVudC5jcmVhdGVFbGVtZW50KHR5cGUpOwoJCQlkb2N1bWVudC5hcHBlbmRDaGlsZChlbGVtZW50KTsKCQkJcmV0dXJuIG5ldyBYTUxNZW1lbnRvKGRvY3VtZW50LCBlbGVtZW50KTsgICAgICAgICAgICAKCQl9IGNhdGNoIChQYXJzZXJDb25maWd1cmF0aW9uRXhjZXB0aW9uIGUpIHsKCQkJdGhyb3cgbmV3IEVycm9yKGUpOwoJCX0KCX0KCQoJLyoqCgkgKiBAc2VlIElNZW1lbnRvLgoJICovCglwdWJsaWMgSU1lbWVudG8gZ2V0Q2hpbGQoU3RyaW5nIHR5cGUpIHsKCQkvLyBHZXQgdGhlIG5vZGVzLgoJCU5vZGVMaXN0IG5vZGVzID0gZWxlbWVudC5nZXRDaGlsZE5vZGVzKCk7CgkJaW50IHNpemUgPSBub2Rlcy5nZXRMZW5ndGgoKTsKCQlpZiAoc2l6ZSA9PSAwKQoJCQlyZXR1cm4gbnVsbDsKCQoJCS8vIEZpbmQgdGhlIGZpcnN0IG5vZGUgd2hpY2ggaXMgYSBjaGlsZCBvZiB0aGlzIG5vZGUuCgkJZm9yIChpbnQgblggPSAwOyBuWCA8IHNpemU7IG5YICsrKSB7CgkJCU5vZGUgbm9kZSA9IG5vZGVzLml0ZW0oblgpOwoJCQlpZiAobm9kZSBpbnN0YW5jZW9mIEVsZW1lbnQpIHsKCQkJCUVsZW1lbnQgZWxlbWVudDIgPSAoRWxlbWVudClub2RlOwoJCQkJaWYgKGVsZW1lbnQyLmdldE5vZGVOYW1lKCkuZXF1YWxzKHR5cGUpKQoJCQkJCXJldHVybiBuZXcgWE1MTWVtZW50byhmYWN0b3J5LCBlbGVtZW50Mik7CgkJCX0KCQl9CgkKCQkvLyBBIGNoaWxkIHdhcyBub3QgZm91bmQuCgkJcmV0dXJuIG51bGw7Cgl9CgoJLyoqCgkgKiBAc2VlIElNZW1lbnRvLgoJICovCglwdWJsaWMgSU1lbWVudG8gW10gZ2V0Q2hpbGRyZW4oU3RyaW5nIHR5cGUpIHsKCQkvLyBHZXQgdGhlIG5vZGVzLgoJCU5vZGVMaXN0IG5vZGVzID0gZWxlbWVudC5nZXRDaGlsZE5vZGVzKCk7CgkJaW50IHNpemUgPSBub2Rlcy5nZXRMZW5ndGgoKTsKCQlpZiAoc2l6ZSA9PSAwKQoJCQlyZXR1cm4gbmV3IElNZW1lbnRvWzBdOwoJCgkJLy8gRXh0cmFjdCBlYWNoIG5vZGUgd2l0aCBnaXZlbiB0eXBlLgoJCUFycmF5TGlzdCBsaXN0ID0gbmV3IEFycmF5TGlzdChzaXplKTsKCQlmb3IgKGludCBuWCA9IDA7IG5YIDwgc2l6ZTsgblggKyspIHsKCQkJTm9kZSBub2RlID0gbm9kZXMuaXRlbShuWCk7CgkJCWlmIChub2RlIGluc3RhbmNlb2YgRWxlbWVudCkgewoJCQkJRWxlbWVudCBlbGVtZW50MiA9IChFbGVtZW50KW5vZGU7CgkJCQlpZiAoZWxlbWVudDIuZ2V0Tm9kZU5hbWUoKS5lcXVhbHModHlwZSkpCgkJCQkJbGlzdC5hZGQoZWxlbWVudDIpOwoJCQl9CgkJfQoJCgkJLy8gQ3JlYXRlIGEgbWVtZW50byBmb3IgZWFjaCBub2RlLgoJCXNpemUgPSBsaXN0LnNpemUoKTsKCQlJTWVtZW50byBbXSByZXN1bHRzID0gbmV3IElNZW1lbnRvW3NpemVdOwoJCWZvciAoaW50IHggPSAwOyB4IDwgc2l6ZTsgeCArKykgewoJCQlyZXN1bHRzW3hdID0gbmV3IFhNTE1lbWVudG8oZmFjdG9yeSwgKEVsZW1lbnQpbGlzdC5nZXQoeCkpOwoJCX0KCQlyZXR1cm4gcmVzdWx0czsKCX0KCgkvKioKCSAqIFJldHVybiB0aGUgY29udGVudHMgb2YgdGhpcyBtZW1lbnRvIGFzIGEgYnl0ZSBhcnJheS4KCSAqCgkgKiBAcmV0dXJuIGJ5dGVbXQoJICovCglwdWJsaWMgYnl0ZVtdIGdldENvbnRlbnRzKCkgdGhyb3dzIElPRXhjZXB0aW9uIHsKCQlCeXRlQXJyYXlPdXRwdXRTdHJlYW0gb3V0ID0gbmV3IEJ5dGVBcnJheU91dHB1dFN0cmVhbSgpOwoJCXNhdmUob3V0KTsKCQlyZXR1cm4gb3V0LnRvQnl0ZUFycmF5KCk7Cgl9CgoJLyoqCgkgKiBSZXR1cm5zIGFuIGlucHV0IHN0cmVhbSBmb3Igd3JpdGluZyB0byB0aGUgZGlzayB3aXRoIGEgbG9jYWwgbG9jYWxlLgoJICoKCSAqIEByZXR1cm4gamF2YS5pby5JbnB1dFN0cmVhbQoJICovCglwdWJsaWMgSW5wdXRTdHJlYW0gZ2V0SW5wdXRTdHJlYW0oKSB0aHJvd3MgSU9FeGNlcHRpb24gewoJCUJ5dGVBcnJheU91dHB1dFN0cmVhbSBvdXQgPSBuZXcgQnl0ZUFycmF5T3V0cHV0U3RyZWFtKCk7CgkJc2F2ZShvdXQpOwoJCXJldHVybiBuZXcgQnl0ZUFycmF5SW5wdXRTdHJlYW0ob3V0LnRvQnl0ZUFycmF5KCkpOwoJfQoKCS8qKgoJICogQHNlZSBJTWVtZW50by4KCSAqLwoJcHVibGljIEZsb2F0IGdldEZsb2F0KFN0cmluZyBrZXkpIHsKCQlBdHRyIGF0dHIgPSBlbGVtZW50LmdldEF0dHJpYnV0ZU5vZGUoa2V5KTsKCQlpZiAoYXR0ciA9PSBudWxsKQoJCQlyZXR1cm4gbnVsbDsgCgkJU3RyaW5nIHN0clZhbHVlID0gYXR0ci5nZXRWYWx1ZSgpOwoJCXRyeSB7CgkJCXJldHVybiBuZXcgRmxvYXQoc3RyVmFsdWUpOwoJCX0gY2F0Y2ggKE51bWJlckZvcm1hdEV4Y2VwdGlvbiBlKSB7CgkJCXJldHVybiBudWxsOwoJCX0KCX0KCgkvKioKCSAqIEBzZWUgSU1lbWVudG8uCgkgKi8KCXB1YmxpYyBTdHJpbmcgZ2V0SWQoKSB7CgkJcmV0dXJuIGVsZW1lbnQuZ2V0QXR0cmlidXRlKFRBR19JRCk7Cgl9CgkKCS8qKgoJICogQHNlZSBJTWVtZW50by4KCSAqLwoJcHVibGljIFN0cmluZyBnZXROYW1lKCkgewoJCXJldHVybiBlbGVtZW50LmdldE5vZGVOYW1lKCk7Cgl9CgoJLyoqCgkgKiBAc2VlIElNZW1lbnRvLgoJICovCglwdWJsaWMgSW50ZWdlciBnZXRJbnRlZ2VyKFN0cmluZyBrZXkpIHsKCQlBdHRyIGF0dHIgPSBlbGVtZW50LmdldEF0dHJpYnV0ZU5vZGUoa2V5KTsKCQlpZiAoYXR0ciA9PSBudWxsKQoJCQlyZXR1cm4gbnVsbDsgCgkJU3RyaW5nIHN0clZhbHVlID0gYXR0ci5nZXRWYWx1ZSgpOwoJCXRyeSB7CgkJCXJldHVybiBuZXcgSW50ZWdlcihzdHJWYWx1ZSk7CgkJfSBjYXRjaCAoTnVtYmVyRm9ybWF0RXhjZXB0aW9uIGUpIHsKCQkJcmV0dXJuIG51bGw7CgkJfQoJfQoKCS8qKgoJICogQHNlZSBJTWVtZW50by4KCSAqLwoJcHVibGljIFN0cmluZyBnZXRTdHJpbmcoU3RyaW5nIGtleSkgewoJCUF0dHIgYXR0ciA9IGVsZW1lbnQuZ2V0QXR0cmlidXRlTm9kZShrZXkpOwoJCWlmIChhdHRyID09IG51bGwpCgkJCXJldHVybiBudWxsOyAKCQlyZXR1cm4gYXR0ci5nZXRWYWx1ZSgpOwoJfQoJCglwdWJsaWMgTGlzdCBnZXROYW1lcygpIHsKCQlOYW1lZE5vZGVNYXAgbWFwID0gZWxlbWVudC5nZXRBdHRyaWJ1dGVzKCk7CgkJaW50IHNpemUgPSBtYXAuZ2V0TGVuZ3RoKCk7CgkJTGlzdCBsaXN0ID0gbmV3IEFycmF5TGlzdCgpOwoJCWZvciAoaW50IGkgPSAwOyBpIDwgc2l6ZTsgaSsrKSB7CgkJCU5vZGUgbm9kZSA9IG1hcC5pdGVtKGkpOwoJCQlTdHJpbmcgbmFtZSA9IG5vZGUuZ2V0Tm9kZU5hbWUoKTsKCQkJbGlzdC5hZGQobmFtZSk7CgkJfQoJCXJldHVybiBsaXN0OwoJfQoKCS8qKgoJICogTG9hZHMgYSBtZW1lbnRvIGZyb20gdGhlIGdpdmVuIGZpbGVuYW1lLgoJICoKCSAqIEBwYXJhbSBpbiBqYXZhLmlvLklucHV0U3RyZWFtCgkgKiBAcmV0dXJuIG9yZy5lY2xpcHNlLnVpLklNZW1lbnRvCgkgKiBAZXhjZXB0aW9uIGphdmEuaW8uSU9FeGNlcHRpb24KCSAqLwoJcHVibGljIHN0YXRpYyBJTWVtZW50byBsb2FkTWVtZW50byhJbnB1dFN0cmVhbSBpbikgewoJCXJldHVybiBjcmVhdGVSZWFkUm9vdChuZXcgSW5wdXRTdHJlYW1SZWFkZXIoaW4pKTsKCX0KCQoJLyoqCgkgKiBMb2FkcyBhIG1lbWVudG8gZnJvbSB0aGUgZ2l2ZW4gZmlsZW5hbWUuCgkgKgoJICogQHBhcmFtIGluIGphdmEuaW8uSW5wdXRTdHJlYW0KCSAqIEByZXR1cm4gb3JnLmVjbGlwc2UudWkuSU1lbWVudG8KCSAqIEBleGNlcHRpb24gamF2YS5pby5JT0V4Y2VwdGlvbgoJICovCglwdWJsaWMgc3RhdGljIElNZW1lbnRvIGxvYWRDb3JydXB0TWVtZW50byhJbnB1dFN0cmVhbSBpbikgewoJCURvY3VtZW50IGRvY3VtZW50ID0gbnVsbDsKCQl0cnkgewoJCQlEb2N1bWVudEJ1aWxkZXJGYWN0b3J5IGZhY3RvcnkgPSBEb2N1bWVudEJ1aWxkZXJGYWN0b3J5Lm5ld0luc3RhbmNlKCk7CgkJCURvY3VtZW50QnVpbGRlciBwYXJzZXIgPSBmYWN0b3J5Lm5ld0RvY3VtZW50QnVpbGRlcigpOwoJCQlkb2N1bWVudCA9IHBhcnNlci5wYXJzZShpbik7CgkJCU5vZGUgbm9kZSA9IGRvY3VtZW50LmdldEZpcnN0Q2hpbGQoKTsKCQkJaWYgKG5vZGUgaW5zdGFuY2VvZiBFbGVtZW50KQoJCQkJcmV0dXJuIG5ldyBYTUxNZW1lbnRvKGRvY3VtZW50LCAoRWxlbWVudCkgbm9kZSk7CgkJfSBjYXRjaCAoRXhjZXB0aW9uIGUpIHsKCQkJLy8gaWdub3JlCgkJfSBmaW5hbGx5IHsKCQkJdHJ5IHsKCQkJCWluLmNsb3NlKCk7CgkJCX0gY2F0Y2ggKEV4Y2VwdGlvbiBlKSB7CgkJCQkvLyBpZ25vcmUKCQkJfQoJCX0KCQlyZXR1cm4gbnVsbDsKCX0KCgkvKioKCSAqIExvYWRzIGEgbWVtZW50byBmcm9tIHRoZSBnaXZlbiBmaWxlbmFtZS4KCSAqCgkgKiBAcGFyYW0gZmlsZW5hbWUgamF2YS5sYW5nLlN0cmluZwoJICogQHJldHVybiBvcmcuZWNsaXBzZS51aS5JTWVtZW50bwoJICogQGV4Y2VwdGlvbiBqYXZhLmlvLklPRXhjZXB0aW9uCgkgKi8KCXB1YmxpYyBzdGF0aWMgSU1lbWVudG8gbG9hZE1lbWVudG8oU3RyaW5nIGZpbGVuYW1lKSB0aHJvd3MgSU9FeGNlcHRpb24gewoJCXJldHVybiBYTUxNZW1lbnRvLmNyZWF0ZVJlYWRSb290KG5ldyBGaWxlUmVhZGVyKGZpbGVuYW1lKSk7Cgl9CgoJLyoqCgkgKiBMb2FkcyBhIG1lbWVudG8gZnJvbSB0aGUgZ2l2ZW4gZmlsZW5hbWUuCgkgKgoJICogQHBhcmFtIHVybCBqYXZhLm5ldC5VUkwKCSAqIEByZXR1cm4gb3JnLmVjbGlwc2UudWkuSU1lbWVudG8KCSAqIEBleGNlcHRpb24gamF2YS5pby5JT0V4Y2VwdGlvbgoJICovCglwdWJsaWMgc3RhdGljIElNZW1lbnRvIGxvYWRNZW1lbnRvKFVSTCB1cmwpIHRocm93cyBJT0V4Y2VwdGlvbiB7CgkJcmV0dXJuIFhNTE1lbWVudG8uY3JlYXRlUmVhZFJvb3QobmV3IElucHV0U3RyZWFtUmVhZGVyKHVybC5vcGVuU3RyZWFtKCkpKTsKCX0KCgkvKioKCSAqIEBzZWUgSU1lbWVudG8uCgkgKi8KCXByaXZhdGUgdm9pZCBwdXRFbGVtZW50KEVsZW1lbnQgZWxlbWVudDIpIHsKCQlOYW1lZE5vZGVNYXAgbm9kZU1hcCA9IGVsZW1lbnQyLmdldEF0dHJpYnV0ZXMoKTsKCQlpbnQgc2l6ZSA9IG5vZGVNYXAuZ2V0TGVuZ3RoKCk7CgkJZm9yIChpbnQgaSA9IDA7IGkgPCBzaXplOyBpKyspewoJCQlBdHRyIGF0dHIgPSAoQXR0cilub2RlTWFwLml0ZW0oaSk7CgkJCXB1dFN0cmluZyhhdHRyLmdldE5hbWUoKSxhdHRyLmdldFZhbHVlKCkpOwoJCX0KCQkKCQlOb2RlTGlzdCBub2RlcyA9IGVsZW1lbnQyLmdldENoaWxkTm9kZXMoKTsKCQlzaXplID0gbm9kZXMuZ2V0TGVuZ3RoKCk7CgkJZm9yIChpbnQgaSA9IDA7IGkgPCBzaXplOyBpICsrKSB7CgkJCU5vZGUgbm9kZSA9IG5vZGVzLml0ZW0oaSk7CgkJCWlmIChub2RlIGluc3RhbmNlb2YgRWxlbWVudCkgewoJCQkJWE1MTWVtZW50byBjaGlsZCA9IChYTUxNZW1lbnRvKWNyZWF0ZUNoaWxkKG5vZGUuZ2V0Tm9kZU5hbWUoKSk7CgkJCQljaGlsZC5wdXRFbGVtZW50KChFbGVtZW50KW5vZGUpOwoJCQl9CgkJfQoJfQoKCS8qKgoJICogQHNlZSBJTWVtZW50by4KCSAqLwoJcHVibGljIHZvaWQgcHV0RmxvYXQoU3RyaW5nIGtleSwgZmxvYXQgZikgewoJCWVsZW1lbnQuc2V0QXR0cmlidXRlKGtleSwgU3RyaW5nLnZhbHVlT2YoZikpOwoJfQoKCS8qKgoJICogQHNlZSBJTWVtZW50by4KCSAqLwoJcHVibGljIHZvaWQgcHV0SW50ZWdlcihTdHJpbmcga2V5LCBpbnQgbikgewoJCWVsZW1lbnQuc2V0QXR0cmlidXRlKGtleSwgU3RyaW5nLnZhbHVlT2YobikpOwoJfQoKCS8qKgoJICogQHNlZSBJTWVtZW50by4KCSAqLwoJcHVibGljIHZvaWQgcHV0TWVtZW50byhJTWVtZW50byBtZW1lbnRvKSB7CgkJWE1MTWVtZW50byB4bWxNZW1lbnRvID0gKFhNTE1lbWVudG8pIG1lbWVudG87CgkJcHV0RWxlbWVudCh4bWxNZW1lbnRvLmVsZW1lbnQpOwoJfQoKCS8qKgoJICogQHNlZSBJTWVtZW50by4KCSAqLwoJcHVibGljIHZvaWQgcHV0U3RyaW5nKFN0cmluZyBrZXksIFN0cmluZyB2YWx1ZSkgewoJCWlmICh2YWx1ZSA9PSBudWxsKQoJCQlyZXR1cm47CgkJZWxlbWVudC5zZXRBdHRyaWJ1dGUoa2V5LCB2YWx1ZSk7Cgl9CgoJLyoqCgkgKiBTYXZlIHRoaXMgTWVtZW50byB0byBhIFdyaXRlci4KCSAqLwoJcHVibGljIHZvaWQgc2F2ZShXcml0ZXIgd3JpdGVyKSB0aHJvd3MgSU9FeGNlcHRpb24gewoJCVJlc3VsdCByZXN1bHQgPSBuZXcgU3RyZWFtUmVzdWx0KHdyaXRlcik7CgkJU291cmNlIHNvdXJjZSA9IG5ldyBET01Tb3VyY2UoZmFjdG9yeSk7CgkJdHJ5IHsKCQkJVHJhbnNmb3JtZXIgdHJhbnNmb3JtZXIgPSBUcmFuc2Zvcm1lckZhY3RvcnkubmV3SW5zdGFuY2UoKS5uZXdUcmFuc2Zvcm1lcigpOwoJCQl0cmFuc2Zvcm1lci5zZXRPdXRwdXRQcm9wZXJ0eShPdXRwdXRLZXlzLklOREVOVCwgInllcyIpOwoJCQl0cmFuc2Zvcm1lci5zZXRPdXRwdXRQcm9wZXJ0eShPdXRwdXRLZXlzLk1FVEhPRCwgInhtbCIpOwoJCQl0cmFuc2Zvcm1lci5zZXRPdXRwdXRQcm9wZXJ0eSgie2h0dHA6Ly94bWwuYXBhY2hlLm9yZy94YWxhbn1pbmRlbnQtYW1vdW50IiwgIjIiKTsKCQkJdHJhbnNmb3JtZXIudHJhbnNmb3JtKHNvdXJjZSwgcmVzdWx0KTsKCQl9IGNhdGNoIChFeGNlcHRpb24gZSkgewoJCQl0aHJvdyAoSU9FeGNlcHRpb24pIChuZXcgSU9FeGNlcHRpb24oKS5pbml0Q2F1c2UoZSkpOwoJCX0KCX0KCQoJLyoqCgkgKiBTYXZlIHRoaXMgTWVtZW50byB0byBhIFdyaXRlci4KCSAqLwoJcHVibGljIHZvaWQgc2F2ZShPdXRwdXRTdHJlYW0gb3MpIHRocm93cyBJT0V4Y2VwdGlvbiB7CgkJUmVzdWx0IHJlc3VsdCA9IG5ldyBTdHJlYW1SZXN1bHQob3MpOwoJCVNvdXJjZSBzb3VyY2UgPSBuZXcgRE9NU291cmNlKGZhY3RvcnkpOwoJCXRyeSB7CgkJCVRyYW5zZm9ybWVyIHRyYW5zZm9ybWVyID0gVHJhbnNmb3JtZXJGYWN0b3J5Lm5ld0luc3RhbmNlKCkubmV3VHJhbnNmb3JtZXIoKTsKCQkJdHJhbnNmb3JtZXIuc2V0T3V0cHV0UHJvcGVydHkoT3V0cHV0S2V5cy5JTkRFTlQsICJ5ZXMiKTsKCQkJdHJhbnNmb3JtZXIuc2V0T3V0cHV0UHJvcGVydHkoT3V0cHV0S2V5cy5NRVRIT0QsICJ4bWwiKTsKCQkJdHJhbnNmb3JtZXIuc2V0T3V0cHV0UHJvcGVydHkoIntodHRwOi8veG1sLmFwYWNoZS5vcmcveGFsYW59aW5kZW50LWFtb3VudCIsICIyIik7CgkJCXRyYW5zZm9ybWVyLnRyYW5zZm9ybShzb3VyY2UsIHJlc3VsdCk7CgkJfSBjYXRjaCAoRXhjZXB0aW9uIGUpIHsKCQkJdGhyb3cgKElPRXhjZXB0aW9uKSAobmV3IElPRXhjZXB0aW9uKCkuaW5pdENhdXNlKGUpKTsKCQl9Cgl9CgoJLyoqCgkgKiBTYXZlcyB0aGUgbWVtZW50byB0byB0aGUgZ2l2ZW4gZmlsZS4KCSAqCgkgKiBAcGFyYW0gZmlsZW5hbWUgamF2YS5sYW5nLlN0cmluZwoJICogQGV4Y2VwdGlvbiBqYXZhLmlvLklPRXhjZXB0aW9uCgkgKi8KCXB1YmxpYyB2b2lkIHNhdmVUb0ZpbGUoU3RyaW5nIGZpbGVuYW1lKSB0aHJvd3MgSU9FeGNlcHRpb24gewoJCVdyaXRlciB3ID0gbnVsbDsKCQl0cnkgewoJCQl3ID0gbmV3IEZpbGVXcml0ZXIoZmlsZW5hbWUpOwoJCQlzYXZlKHcpOwoJCX0gY2F0Y2ggKElPRXhjZXB0aW9uIGUpIHsKCQkJdGhyb3cgZTsKCQl9IGNhdGNoIChFeGNlcHRpb24gZSkgewoJCQl0aHJvdyBuZXcgSU9FeGNlcHRpb24oZS5nZXRMb2NhbGl6ZWRNZXNzYWdlKCkpOwoJCX0gZmluYWxseSB7CgkJCWlmICh3ICE9IG51bGwpIHsKCQkJCXRyeSB7CgkJCQkJdy5jbG9zZSgpOwoJCQkJfSBjYXRjaCAoRXhjZXB0aW9uIGUpIHsKCQkJCQkvLyBpZ25vcmUKCQkJCX0KCQkJfQoJCX0KCX0KCQoJcHVibGljIFN0cmluZyBzYXZlVG9TdHJpbmcoKSB0aHJvd3MgSU9FeGNlcHRpb24gewoJCUJ5dGVBcnJheU91dHB1dFN0cmVhbSBvdXQgPSBuZXcgQnl0ZUFycmF5T3V0cHV0U3RyZWFtKCk7CgkJc2F2ZShvdXQpOwoJCXJldHVybiBvdXQudG9TdHJpbmcoIlVURi04Iik7Cgl9CgkKCS8qCgkgKiBAc2VlIElNZW1lbnRvI2dldEJvb2xlYW4oU3RyaW5nKQoJICovCglwdWJsaWMgQm9vbGVhbiBnZXRCb29sZWFuKFN0cmluZyBrZXkpIHsKCQlBdHRyIGF0dHIgPSBlbGVtZW50LmdldEF0dHJpYnV0ZU5vZGUoa2V5KTsKCQlpZiAoYXR0ciA9PSBudWxsKQoJCQlyZXR1cm4gbnVsbDsKCQlTdHJpbmcgc3RyVmFsdWUgPSBhdHRyLmdldFZhbHVlKCk7CgkJaWYgKCJ0cnVlIi5lcXVhbHNJZ25vcmVDYXNlKHN0clZhbHVlKSkKCQkJcmV0dXJuIG5ldyBCb29sZWFuKHRydWUpOwoJCXJldHVybiBuZXcgQm9vbGVhbihmYWxzZSk7Cgl9CgoJLyoKCSAqIEBzZWUgSU1lbWVudG8jcHV0Qm9vbGVhbihTdHJpbmcsIGJvb2xlYW4pCgkgKi8KCXB1YmxpYyB2b2lkIHB1dEJvb2xlYW4oU3RyaW5nIGtleSwgYm9vbGVhbiB2YWx1ZSkgewoJCWVsZW1lbnQuc2V0QXR0cmlidXRlKGtleSwgdmFsdWUgPyAidHJ1ZSIgOiAiZmFsc2UiKTsKCX0KfQ==