LyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogQ29weXJpZ2h0IChjKSAyMDAxLCAyMDA4IElCTSBDb3Jwb3JhdGlvbiBhbmQgb3RoZXJzLgogKiBBbGwgcmlnaHRzIHJlc2VydmVkLiBUaGlzIHByb2dyYW0gYW5kIHRoZSBhY2NvbXBhbnlpbmcgbWF0ZXJpYWxzCiAqIGFyZSBtYWRlIGF2YWlsYWJsZSB1bmRlciB0aGUgdGVybXMgb2YgdGhlIEVjbGlwc2UgUHVibGljIExpY2Vuc2UgdjEuMAogKiB3aGljaCBhY2NvbXBhbmllcyB0aGlzIGRpc3RyaWJ1dGlvbiwgYW5kIGlzIGF2YWlsYWJsZSBhdAogKiBodHRwOi8vd3d3LmVjbGlwc2Uub3JnL2xlZ2FsL2VwbC12MTAuaHRtbAogKiAKICogQ29udHJpYnV0b3JzOgogKiAgICAgSUJNIENvcnBvcmF0aW9uIC0gaW5pdGlhbCBBUEkgYW5kIGltcGxlbWVudGF0aW9uCiAqICAgICBKZW5zIEx1a293c2tpL0lubm9vcHJhY3QgLSBpbml0aWFsIHJlbmFtaW5nL3Jlc3RydWN0dXJpbmcKICogICAgIEplc3BlciBTdGVlbiBN+GxsZXIgLSB4bWw6c3BhY2U9J3ByZXNlcnZlJyBzdXBwb3J0CiAqICAgICAKICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovCnBhY2thZ2Ugb3JnLmVjbGlwc2Uud3N0LnhtbC5jb3JlLmludGVybmFsLnByb3Zpc2lvbmFsLmZvcm1hdDsKCmltcG9ydCBqYXZhLnV0aWwuTGlzdDsKaW1wb3J0IGphdmEudXRpbC5WZWN0b3I7CgppbXBvcnQgb3JnLmVjbGlwc2UuamZhY2UudGV4dC5CYWRMb2NhdGlvbkV4Y2VwdGlvbjsKaW1wb3J0IG9yZy5lY2xpcHNlLndzdC5zc2UuY29yZS5pbnRlcm5hbC5mb3JtYXQuSVN0cnVjdHVyZWRGb3JtYXRDb250cmFpbnRzOwppbXBvcnQgb3JnLmVjbGlwc2Uud3N0LnNzZS5jb3JlLmludGVybmFsLnByb3Zpc2lvbmFsLnRleHQuSVN0cnVjdHVyZWREb2N1bWVudDsKaW1wb3J0IG9yZy5lY2xpcHNlLndzdC5zc2UuY29yZS51dGlscy5TdHJpbmdVdGlsczsKaW1wb3J0IG9yZy5lY2xpcHNlLndzdC54bWwuY29yZS5pbnRlcm5hbC5Mb2dnZXI7CmltcG9ydCBvcmcuZWNsaXBzZS53c3QueG1sLmNvcmUuaW50ZXJuYWwucHJvdmlzaW9uYWwuZG9jdW1lbnQuSURPTU5vZGU7CmltcG9ydCBvcmcudzNjLmRvbS5Ob2RlOwoKCnB1YmxpYyBjbGFzcyBUZXh0Tm9kZUZvcm1hdHRlciBleHRlbmRzIE5vZGVGb3JtYXR0ZXIgewoJc3RhdGljIHByaXZhdGUgZmluYWwgU3RyaW5nIENSID0gIlxyIjsgLy8kTk9OLU5MUy0xJAoJc3RhdGljIHByaXZhdGUgZmluYWwgU3RyaW5nIERFTElNSVRFUlMgPSAiIFx0XG5cclxmIjsgLy8kTk9OLU5MUy0xJAoJc3RhdGljIHByaXZhdGUgZmluYWwgU3RyaW5nIEVNUFRZX1NUUklORyA9ICIiOyAvLyROT04tTkxTLTEkCglzdGF0aWMgcHJpdmF0ZSBmaW5hbCBTdHJpbmcgRkYgPSAiXGYiOyAvLyROT04tTkxTLTEkCglzdGF0aWMgcHJpdmF0ZSBmaW5hbCBTdHJpbmcgTEYgPSAiXG4iOyAvLyROT04tTkxTLTEkCglzdGF0aWMgcHJpdmF0ZSBmaW5hbCBTdHJpbmcgU1BBQ0UgPSAiICI7IC8vJE5PTi1OTFMtMSQKCXN0YXRpYyBwcml2YXRlIGZpbmFsIFN0cmluZyBUQUIgPSAiXHQiOyAvLyROT04tTkxTLTEkCgoJcHJpdmF0ZSBTdHJpbmcgY29tcHJlc3NTcGFjZXMoU3RyaW5nIHN0cmluZywgSVN0cnVjdHVyZWRGb3JtYXRDb250cmFpbnRzIGZvcm1hdENvbnRyYWludHMpIHsKCQkvKgoJCSAqIE5vdGUgdGhhdCB0aGUgU3RydWN0dXJlZFRleHRFZGl0b3Igc3VwcG9ydHMgbWl4ZWQgbmV3IGxpbmUKCQkgKiBjaGFyYWN0ZXJzIChDUiwgTEYsIENSTEYpIGluIG9uZSBmaWxlLiBXZSBoYXZlIHRvIGhhbmRsZSB0aGF0IHdoZW4KCQkgKiB3ZSB0cnkgdG8gcHJlc2VydmUgYmxhbmsgbGluZXMuCgkJICovCgkJU3RyaW5nW10gc3RyaW5nQXJyYXkgPSBudWxsOwoJCWJvb2xlYW4gY2xlYXJBbGxCbGFua0xpbmVzID0gZm9ybWF0Q29udHJhaW50cy5nZXRDbGVhckFsbEJsYW5rTGluZXMoKTsKCgkJaWYgKGNsZWFyQWxsQmxhbmtMaW5lcykKCQkJc3RyaW5nQXJyYXkgPSBTdHJpbmdVdGlscy5hc0FycmF5KHN0cmluZyk7CgkJZWxzZQoJCQlzdHJpbmdBcnJheSA9IFN0cmluZ1V0aWxzLmFzQXJyYXkoc3RyaW5nLCBERUxJTUlURVJTLCB0cnVlKTsKCgkJU3RyaW5nQnVmZmVyIGNvbXByZXNzZWRTdHJpbmcgPSBuZXcgU3RyaW5nQnVmZmVyKCk7CgkJaWYgKHN0cmluZ0FycmF5Lmxlbmd0aCA+IDApIHsKCQkJYm9vbGVhbiBjciA9IGZhbHNlLCBsZiA9IGZhbHNlLCBjcjIgPSBmYWxzZSwgbm9uU3BhY2UgPSB0cnVlOwoKCQkJaWYgKHN0cmluZ0FycmF5WzBdLmNvbXBhcmVUbyhDUikgPT0gMCkKCQkJCWNyID0gdHJ1ZTsKCQkJZWxzZSBpZiAoc3RyaW5nQXJyYXlbMF0uY29tcGFyZVRvKExGKSA9PSAwKQoJCQkJbGYgPSB0cnVlOwoJCQllbHNlIGlmICgoc3RyaW5nQXJyYXlbMF0uY29tcGFyZVRvKFNQQUNFKSAhPSAwKSAmJiAoc3RyaW5nQXJyYXlbMF0uY29tcGFyZVRvKFRBQikgIT0gMCkgJiYgKHN0cmluZ0FycmF5WzBdLmNvbXBhcmVUbyhGRikgIT0gMCkpIHsKCQkJCWNvbXByZXNzZWRTdHJpbmcuYXBwZW5kKHN0cmluZ0FycmF5WzBdKTsKCQkJCW5vblNwYWNlID0gdHJ1ZTsKCQkJfQoKCQkJZm9yIChpbnQgaSA9IDE7IGkgPCBzdHJpbmdBcnJheS5sZW5ndGg7IGkrKykgewoJCQkJaWYgKHN0cmluZ0FycmF5W2ldLmNvbXBhcmVUbyhDUikgPT0gMCkgewoJCQkJCWlmIChjciAmJiBsZikgewoJCQkJCQlpZiAobm9uU3BhY2UpIHsKCQkJCQkJCWNvbXByZXNzZWRTdHJpbmcuYXBwZW5kKENSICsgTEYpOwoJCQkJCQkJbm9uU3BhY2UgPSBmYWxzZTsKCQkJCQkJfQoJCQkJCQljb21wcmVzc2VkU3RyaW5nLmFwcGVuZChzdHJpbmdBcnJheVtpXSk7CgkJCQkJCWNyMiA9IHRydWU7CgkJCQkJfQoJCQkJCWVsc2UgaWYgKGNyKSB7CgkJCQkJCWlmIChub25TcGFjZSkgewoJCQkJCQkJY29tcHJlc3NlZFN0cmluZy5hcHBlbmQoQ1IpOwoJCQkJCQkJbm9uU3BhY2UgPSBmYWxzZTsKCQkJCQkJfQoJCQkJCQljb21wcmVzc2VkU3RyaW5nLmFwcGVuZChzdHJpbmdBcnJheVtpXSk7CgkJCQkJCWNyMiA9IHRydWU7CgkJCQkJfQoJCQkJCWVsc2UKCQkJCQkJY3IgPSB0cnVlOwoJCQkJfQoJCQkJZWxzZSBpZiAoc3RyaW5nQXJyYXlbaV0uY29tcGFyZVRvKExGKSA9PSAwKSB7CgkJCQkJaWYgKGNyICYmIGxmICYmIGNyMikgewoJCQkJCQljb21wcmVzc2VkU3RyaW5nLmFwcGVuZChzdHJpbmdBcnJheVtpXSk7CgkJCQkJfQoJCQkJCWVsc2UgaWYgKGxmKSB7CgkJCQkJCWlmIChub25TcGFjZSkgewoJCQkJCQkJY29tcHJlc3NlZFN0cmluZy5hcHBlbmQoTEYpOwoJCQkJCQkJbm9uU3BhY2UgPSBmYWxzZTsKCQkJCQkJfQoJCQkJCQljb21wcmVzc2VkU3RyaW5nLmFwcGVuZChzdHJpbmdBcnJheVtpXSk7CgkJCQkJfQoJCQkJCWVsc2UKCQkJCQkJbGYgPSB0cnVlOwoJCQkJfQoJCQkJZWxzZSBpZiAoKHN0cmluZ0FycmF5W2ldLmNvbXBhcmVUbyhTUEFDRSkgIT0gMCkgJiYgKHN0cmluZ0FycmF5W2ldLmNvbXBhcmVUbyhUQUIpICE9IDApICYmIChzdHJpbmdBcnJheVtpXS5jb21wYXJlVG8oRkYpICE9IDApKSB7CgkJCQkJaWYgKGNvbXByZXNzZWRTdHJpbmcubGVuZ3RoKCkgPiAwKQoJCQkJCQljb21wcmVzc2VkU3RyaW5nLmFwcGVuZChTUEFDRSk7CgkJCQkJY29tcHJlc3NlZFN0cmluZy5hcHBlbmQoc3RyaW5nQXJyYXlbaV0pOwoKCQkJCQljciA9IGZhbHNlOwoJCQkJCWxmID0gZmFsc2U7CgkJCQkJY3IyID0gZmFsc2U7CgkJCQkJbm9uU3BhY2UgPSB0cnVlOwoJCQkJfQoJCQl9CgkJfQoKCQlyZXR1cm4gY29tcHJlc3NlZFN0cmluZy50b1N0cmluZygpOwoJfQoKCXByb3RlY3RlZCB2b2lkIGZvcm1hdE5vZGUoSURPTU5vZGUgbm9kZSwgSVN0cnVjdHVyZWRGb3JtYXRDb250cmFpbnRzIGZvcm1hdENvbnRyYWludHMpIHsKCQkvLyBbMTExNjc0XSBJZiBpbnNpZGUgeG1sOnNwYWNlPSJwcmVzZXJ2ZSIgZWxlbWVudCwgd2UgYmFpbAoJCWlmIChmb3JtYXRDb250cmFpbnRzLmdldEluUHJlc2VydmVTcGFjZUVsZW1lbnQoKSkKCQkJcmV0dXJuOwoJCWlmIChub2RlICE9IG51bGwpIHsKCQkJSVN0cnVjdHVyZWREb2N1bWVudCBkb2MgPSBub2RlLmdldFN0cnVjdHVyZWREb2N1bWVudCgpOwoKCQkJaW50IGxpbmVXaWR0aCA9IGdldEZvcm1hdFByZWZlcmVuY2VzKCkuZ2V0TGluZVdpZHRoKCk7CgkJCWludCBjdXJyZW50QXZhaWxhYmxlTGluZVdpZHRoID0gY29tcHV0ZUF2YWlsYWJsZUxpbmVXaWR0aChkb2MsIG5vZGUuZ2V0U3RhcnRPZmZzZXQoKSwgbGluZVdpZHRoKTsKCgkJCVN0cmluZyBub2RlVGV4dCA9IGdldE5vZGVUZXh0KG5vZGUpOwoJCQlTdHJpbmcgY29tcHJlc3NlZFRleHQgPSBjb21wcmVzc1NwYWNlcyhub2RlVGV4dCwgZm9ybWF0Q29udHJhaW50cyk7CgoJCQlJRE9NTm9kZSBwYXJlbnROb2RlID0gKElET01Ob2RlKSBub2RlLmdldFBhcmVudE5vZGUoKTsKCgkJCWlmICgoKGVub3VnaFNwYWNlKHBhcmVudE5vZGUsIGN1cnJlbnRBdmFpbGFibGVMaW5lV2lkdGgsIGNvbXByZXNzZWRUZXh0KSkgJiYgKG5vU2libGluZ3NBbmROb0ZvbGxvd2luZ0NvbW1lbnQobm9kZSkpICYmICFmaXJzdFN0cnVjdHVyZWREb2N1bWVudFJlZ2lvbkNvbnRhaW5zTGluZURlbGltaXRlcnMocGFyZW50Tm9kZSkpIHx8IG5vZGUuZ2V0U3RhcnRPZmZzZXQoKSA9PSAwKSB7CgkJCQloYW5kbGVOb1JlZmxvdyhub2RlLCBkb2MsIGNvbXByZXNzZWRUZXh0LCBwYXJlbnROb2RlKTsKCQkJfQoJCQllbHNlIHsKCQkJCS8vIG5vdCBlbm91Z2ggc3BhY2UsIG5lZWQgdG8gcmVmbG93IHRleHQKCQkJCVN0cmluZyBub2RlSW5kZW50YXRpb24gPSBmb3JtYXRDb250cmFpbnRzLmdldEN1cnJlbnRJbmRlbnQoKTsKCQkJCWN1cnJlbnRBdmFpbGFibGVMaW5lV2lkdGggPSBsaW5lV2lkdGggLSBnZXRJbmRlbnRhdGlvbkxlbmd0aChub2RlSW5kZW50YXRpb24pOwoJCQkJTGlzdCB2ZWN0b3IgPSByZWZsb3dUZXh0KGNvbXByZXNzZWRUZXh0LCBjdXJyZW50QXZhaWxhYmxlTGluZVdpZHRoKTsKCQkJCWludCB2ZWN0b3JTaXplID0gdmVjdG9yLnNpemUoKTsKCQkJCVN0cmluZ0J1ZmZlciByZWZsb3dlZFRleHRCdWZmZXIgPSBuZXcgU3RyaW5nQnVmZmVyKCk7CgkJCQlTdHJpbmcgbGluZURlbGltaXRlciA9IGdldExpbmVEZWxpbWl0ZXIoZG9jLCBub2RlLmdldFN0YXJ0T2Zmc2V0KCkpOwoJCQkJLy8gaGFuZGxlIGZpcnN0IGxpbmUgc3BlY2lhbGx5IHRvIGNoZWNrIGZvciBhbGxvd1doaXRlc3BhY2UKCQkJCWlmICh2ZWN0b3JTaXplID4gMCkgewoJCQkJCS8vIGRldGVybWluZXMgd2hldGhlciBvciBub3QgdG8gYWxsb3cgd2hpdGVzcGFjZSBpZiB0aGVyZQoJCQkJCS8vIGlzIGFuIGVudGl0eSBvciBjZGF0YSBiZWZvcmUgaXQKCQkJCQlib29sZWFuIGFsbG93V2hpdGVzcGFjZSA9IHRydWU7CgkJCQkJLy8gWzIwNjA3Ml0gU3RyaW5nSW5kZXhPdXRPZkJvdW5kc0V4Y2VwdGlvbgoJCQkJCWlmIChub2RlVGV4dC5sZW5ndGgoKSA9PSAwIHx8ICFDaGFyYWN0ZXIuaXNXaGl0ZXNwYWNlKG5vZGVUZXh0LmNoYXJBdCgwKSkpIHsKCQkJCQkJTm9kZSBwcmV2aW91c1NpYmxpbmcgPSBub2RlLmdldFByZXZpb3VzU2libGluZygpOwoJCQkJCQlpZiAocHJldmlvdXNTaWJsaW5nICE9IG51bGwgJiYgKHByZXZpb3VzU2libGluZy5nZXROb2RlVHlwZSgpID09IE5vZGUuRU5USVRZX1JFRkVSRU5DRV9OT0RFIHx8IHByZXZpb3VzU2libGluZy5nZXROb2RlVHlwZSgpID09IE5vZGUuQ0RBVEFfU0VDVElPTl9OT0RFKSkKCQkJCQkJCWFsbG93V2hpdGVzcGFjZSA9IGZhbHNlOwoJCQkJCX0KCQkJCQlTdHJpbmcgdGhlU3RyaW5nID0gKFN0cmluZykgdmVjdG9yLmdldCgwKTsKCQkJCQlpZiAoYWxsb3dXaGl0ZXNwYWNlKSB7CgkJCQkJCXJlZmxvd2VkVGV4dEJ1ZmZlci5hcHBlbmQobGluZURlbGltaXRlcik7CgkJCQkJCWlmICh0aGVTdHJpbmcudHJpbSgpLmxlbmd0aCgpID4gMCkKCQkJCQkJCXJlZmxvd2VkVGV4dEJ1ZmZlci5hcHBlbmQobm9kZUluZGVudGF0aW9uKS5hcHBlbmQodGhlU3RyaW5nKTsKCQkJCQl9CgkJCQkJZWxzZSB7CgkJCQkJCXJlZmxvd2VkVGV4dEJ1ZmZlci5hcHBlbmQodGhlU3RyaW5nKTsKCQkJCQl9CgkJCQl9CgkJCQkvLyBkbyB0aGUgcmVzdCBvZiB0aGUgbGluZXMKCQkJCWZvciAoaW50IGkgPSAxOyBpIDwgdmVjdG9yU2l6ZTsgaSsrKSB7CgkJCQkJU3RyaW5nIHRoZVN0cmluZyA9IChTdHJpbmcpIHZlY3Rvci5nZXQoaSk7CgkJCQkJaWYgKHRoZVN0cmluZy50cmltKCkubGVuZ3RoKCkgPiAwKQoJCQkJCQlyZWZsb3dlZFRleHRCdWZmZXIuYXBwZW5kKGxpbmVEZWxpbWl0ZXIpLmFwcGVuZChub2RlSW5kZW50YXRpb24pLmFwcGVuZCh0aGVTdHJpbmcpOwoJCQkJCWVsc2UKCQkJCQkJcmVmbG93ZWRUZXh0QnVmZmVyLmFwcGVuZChsaW5lRGVsaW1pdGVyKTsKCQkJCX0KCQkJCVN0cmluZyByZWZsb3dlZFRleHQgPSByZWZsb3dlZFRleHRCdWZmZXIudG9TdHJpbmcoKTsKCQkJCWlmIChub2RlLmdldE5leHRTaWJsaW5nKCkgPT0gbnVsbCkgewoJCQkJCWlmIChpc0VuZFRhZ01pc3NpbmcocGFyZW50Tm9kZSkpIHsKCQkJCQkJLy8gZG9uJ3QgYWRkIGluZGVudGF0aW9uIHRvIGVuZCBpZiBwYXJlbnQgZW5kIHRhZyBpcwoJCQkJCQkvLyBtaXNzaW5nCgkJCQkJfQoKCQkJCQllbHNlIHsKCQkJCQkJLy8gYWRkIHBhcmVudCdzIGluZGVudGF0aW9uIHRvIGVuZAoJCQkJCQlub2RlSW5kZW50YXRpb24gPSBnZXROb2RlSW5kZW50KHBhcmVudE5vZGUpOwoJCQkJCQlpZiAoIXJlZmxvd2VkVGV4dC5lbmRzV2l0aChsaW5lRGVsaW1pdGVyICsgbm9kZUluZGVudGF0aW9uKSkgewoJCQkJCQkJcmVmbG93ZWRUZXh0ID0gU3RyaW5nVXRpbHMuYXBwZW5kSWZOb3RFbmRXaXRoKHJlZmxvd2VkVGV4dCwgbGluZURlbGltaXRlcik7CgkJCQkJCQlyZWZsb3dlZFRleHQgPSBTdHJpbmdVdGlscy5hcHBlbmRJZk5vdEVuZFdpdGgocmVmbG93ZWRUZXh0LCBub2RlSW5kZW50YXRpb24pOwoJCQkJCQl9CgkJCQkJfQoJCQkJfQoJCQkJZWxzZSB7CgkJCQkJaWYgKCFyZWZsb3dlZFRleHQuZW5kc1dpdGgobGluZURlbGltaXRlciArIG5vZGVJbmRlbnRhdGlvbikpIHsKCQkJCQkJLy8gbm90IGFscmVhZHkgZW5kZWQgd2l0aCB0aGUgZXhwZWN0ZWQgaW5kZW50YXRpb24KCQkJCQkJTm9kZSBuZXh0U2libGluZyA9IG5vZGUuZ2V0TmV4dFNpYmxpbmcoKTsKCQkJCQkJaWYgKG5leHRTaWJsaW5nLmdldE5vZGVUeXBlKCkgPT0gTm9kZS5DT01NRU5UX05PREUpIHsKCQkJCQkJCS8vIGFkZCBpbmRlbnRhdGlvbiB0byBlbmQgaWYKCQkJCQkJCS8vIGN1cnJlbnRUZXh0RW5kc1dpdGhMaW5lRGVsaW1pdGVyCgkJCQkJCQkvLyBvciBmb2xsb3dlZCBieSBtdWx0aUxpbmVDb21tZW50CgoJCQkJCQkJaW50IGluZGV4T2ZMYXN0TGluZURlbGltaXRlciA9IFN0cmluZ1V0aWxzLmluZGV4T2ZMYXN0TGluZURlbGltaXRlcihub2RlVGV4dCk7CgkJCQkJCQlib29sZWFuIGN1cnJlbnRUZXh0RW5kc1dpdGhMaW5lRGVsaW1pdGVyID0gaW5kZXhPZkxhc3RMaW5lRGVsaW1pdGVyICE9IC0xOwoJCQkJCQkJaWYgKGN1cnJlbnRUZXh0RW5kc1dpdGhMaW5lRGVsaW1pdGVyKSB7CgkJCQkJCQkJLy8gbm8gbW9yZSBub24gYmxhbmsgY2hhcmFjdGVyIGFmdGVyIHRoZSBsYXN0CgkJCQkJCQkJLy8gbGluZSBkZWxpbWl0ZXIKCQkJCQkJCQljdXJyZW50VGV4dEVuZHNXaXRoTGluZURlbGltaXRlciA9IFN0cmluZ1V0aWxzLmluZGV4T2ZOb25ibGFuayhub2RlVGV4dCwgaW5kZXhPZkxhc3RMaW5lRGVsaW1pdGVyKSA9PSAtMTsKCQkJCQkJCX0KCgkJCQkJCQlTdHJpbmcgbm9kZVZhbHVlID0gbmV4dFNpYmxpbmcuZ2V0Tm9kZVZhbHVlKCk7CgkJCQkJCQlib29sZWFuIG11bHRpTGluZUNvbW1lbnQgPSBTdHJpbmdVdGlscy5jb250YWluc0xpbmVEZWxpbWl0ZXIobm9kZVZhbHVlKTsKCgkJCQkJCQlpZiAoY3VycmVudFRleHRFbmRzV2l0aExpbmVEZWxpbWl0ZXIgfHwgbXVsdGlMaW5lQ29tbWVudCkgewoJCQkJCQkJCXJlZmxvd2VkVGV4dCA9IFN0cmluZ1V0aWxzLmFwcGVuZElmTm90RW5kV2l0aChyZWZsb3dlZFRleHQsIGxpbmVEZWxpbWl0ZXIpOwoJCQkJCQkJCXJlZmxvd2VkVGV4dCA9IFN0cmluZ1V0aWxzLmFwcGVuZElmTm90RW5kV2l0aChyZWZsb3dlZFRleHQsIG5vZGVJbmRlbnRhdGlvbik7CgkJCQkJCQl9CgkJCQkJCX0KCQkJCQkJZWxzZSBpZiAobmV4dFNpYmxpbmcuZ2V0Tm9kZVR5cGUoKSA9PSBOb2RlLkVOVElUWV9SRUZFUkVOQ0VfTk9ERSB8fCBuZXh0U2libGluZy5nZXROb2RlVHlwZSgpID09IE5vZGUuQ0RBVEFfU0VDVElPTl9OT0RFKSB7CgkJCQkJCQlpbnQgdGV4dExlbmd0aCA9IG5vZGVUZXh0Lmxlbmd0aCgpOwoJCQkJCQkJaWYgKHRleHRMZW5ndGggPiAwICYmIENoYXJhY3Rlci5pc1doaXRlc3BhY2Uobm9kZVRleHQuY2hhckF0KHRleHRMZW5ndGggLSAxKSkpIHsKCQkJCQkJCQlyZWZsb3dlZFRleHQgPSBTdHJpbmdVdGlscy5hcHBlbmRJZk5vdEVuZFdpdGgocmVmbG93ZWRUZXh0LCBsaW5lRGVsaW1pdGVyKTsKCQkJCQkJCQlyZWZsb3dlZFRleHQgPSBTdHJpbmdVdGlscy5hcHBlbmRJZk5vdEVuZFdpdGgocmVmbG93ZWRUZXh0LCBub2RlSW5kZW50YXRpb24pOwoJCQkJCQkJfQoJCQkJCQl9CgkJCQkJCWVsc2UgewoJCQkJCQkJLy8gbm90IGEgY29tbWVudCwganVzdCBhZGQgYWRkIGluZGVudGF0aW9uIHRvIGVuZAoJCQkJCQkJcmVmbG93ZWRUZXh0ID0gU3RyaW5nVXRpbHMuYXBwZW5kSWZOb3RFbmRXaXRoKHJlZmxvd2VkVGV4dCwgbGluZURlbGltaXRlcik7CgkJCQkJCQlyZWZsb3dlZFRleHQgPSBTdHJpbmdVdGlscy5hcHBlbmRJZk5vdEVuZFdpdGgocmVmbG93ZWRUZXh0LCBub2RlSW5kZW50YXRpb24pOwoJCQkJCQl9CgkJCQkJfQoJCQkJfQoKCQkJCXJlcGxhY2VOb2RlVmFsdWUobm9kZSwgcmVmbG93ZWRUZXh0KTsKCQkJfQoJCX0KCX0KCgkvKioKCSAqIEtlZXBzIHRleHQgaW5saW5lIHdpdGggaXRzIHBhcmVudCAobm8gcmVmbG93IG5lY2Vzc2FyeSkKCSAqIAoJICogQHBhcmFtIG5vZGUKCSAqIEBwYXJhbSBkb2MKCSAqIEBwYXJhbSBjb21wcmVzc2VkVGV4dAoJICogQHBhcmFtIHBhcmVudE5vZGUKCSAqLwoJcHJpdmF0ZSB2b2lkIGhhbmRsZU5vUmVmbG93KElET01Ob2RlIG5vZGUsIElTdHJ1Y3R1cmVkRG9jdW1lbnQgZG9jLCBTdHJpbmcgY29tcHJlc3NlZFRleHQsIElET01Ob2RlIHBhcmVudE5vZGUpIHsKCQlTdHJpbmcgbm9kZUluZGVudGF0aW9uOwoJCS8vIGVub3VnaCBzcGFjZSBhbmQgdGV4dCBoYXMgbm8gbGluZSBkZWxpbWl0ZXJzIGFuZCAobm9kZSBoYXMgbm8KCQkvLyBzaWJsaW5ncyBvciBmb2xsb3dlZCBieSBpbmxpbmUgY29tbWVudCkgYW5kCgkJLy8gcGFyZW50Rmlyc3RTdHJ1Y3R1cmVkRG9jdW1lbnRSZWdpb25Db250YWluc0xpbmVEZWxpbWl0ZXJzCgoJCWlmIChpc0VuZFRhZ01pc3NpbmcocGFyZW50Tm9kZSkpIHsKCQkJcGFyZW50Tm9kZSA9IChJRE9NTm9kZSkgcGFyZW50Tm9kZS5nZXRQYXJlbnROb2RlKCk7CgkJCXdoaWxlIChpc0VuZFRhZ01pc3NpbmcocGFyZW50Tm9kZSkpCgkJCQlwYXJlbnROb2RlID0gKElET01Ob2RlKSBwYXJlbnROb2RlLmdldFBhcmVudE5vZGUoKTsKCgkJCS8vIGFkZCBwYXJlbnQncyBpbmRlbnRhdGlvbiB0byBlbmQKCQkJbm9kZUluZGVudGF0aW9uID0gZ2V0Tm9kZUluZGVudChwYXJlbnROb2RlKTsKCQkJU3RyaW5nIGxpbmVEZWxpbWl0ZXIgPSBnZXRMaW5lRGVsaW1pdGVyKGRvYywgbm9kZS5nZXRTdGFydE9mZnNldCgpKTsKCQkJaWYgKCFjb21wcmVzc2VkVGV4dC5lbmRzV2l0aChsaW5lRGVsaW1pdGVyICsgbm9kZUluZGVudGF0aW9uKSkgewoJCQkJY29tcHJlc3NlZFRleHQgPSBTdHJpbmdVdGlscy5hcHBlbmRJZk5vdEVuZFdpdGgoY29tcHJlc3NlZFRleHQsIGxpbmVEZWxpbWl0ZXIpOwoJCQkJY29tcHJlc3NlZFRleHQgPSBTdHJpbmdVdGlscy5hcHBlbmRJZk5vdEVuZFdpdGgoY29tcHJlc3NlZFRleHQsIG5vZGVJbmRlbnRhdGlvbik7CgkJCX0KCQl9CgoJCWlmICgocGFyZW50Tm9kZSAhPSBudWxsKSAmJiAocGFyZW50Tm9kZS5nZXROb2RlVHlwZSgpID09IE5vZGUuRE9DVU1FTlRfTk9ERSkgJiYgKG5vZGUuZ2V0Tm9kZVZhbHVlKCkubGVuZ3RoKCkgPiAwKSAmJiAobm9kZS5nZXROb2RlVmFsdWUoKS50cmltKCkubGVuZ3RoKCkgPT0gMCkgJiYgKChub2RlLmdldFByZXZpb3VzU2libGluZygpID09IG51bGwpIHx8IChub2RlLmdldE5leHRTaWJsaW5nKCkgPT0gbnVsbCkpKQoJCQkvLyBkZWxldGUgc3BhY2VzIGF0IHRoZSBiZWdpbm5pbmcgb3IgZW5kIG9mIHRoZSBkb2N1bWVudAoJCQljb21wcmVzc2VkVGV4dCA9IEVNUFRZX1NUUklORzsKCgkJcmVwbGFjZU5vZGVWYWx1ZShub2RlLCBjb21wcmVzc2VkVGV4dCk7Cgl9CgoJcHJpdmF0ZSBib29sZWFuIG5vU2libGluZ3NBbmROb0ZvbGxvd2luZ0NvbW1lbnQoSURPTU5vZGUgbm9kZSkgewoJCUlET01Ob2RlIG5leHRTaWJsaW5nID0gKElET01Ob2RlKSBub2RlLmdldE5leHRTaWJsaW5nKCk7CgkJcmV0dXJuICFub2RlSGFzU2libGluZ3Mobm9kZSkgfHwgKG5vTGluZURlbGltaXRlcihub2RlKSAmJiBpc0NvbW1lbnQobmV4dFNpYmxpbmcpICYmIG5vTGluZURlbGltaXRlcihuZXh0U2libGluZykpOwoJfQoKCXByaXZhdGUgYm9vbGVhbiBpc0NvbW1lbnQoSURPTU5vZGUgbm9kZSkgewoJCWJvb2xlYW4gcmVzdWx0ID0gZmFsc2U7CgkJaWYgKG5vZGUgIT0gbnVsbCkgewoJCQlyZXN1bHQgPSBub2RlLmdldE5vZGVUeXBlKCkgPT0gTm9kZS5DT01NRU5UX05PREU7CgkJfQoJCXJldHVybiByZXN1bHQ7Cgl9CgoJcHJpdmF0ZSBib29sZWFuIG5vTGluZURlbGltaXRlcihJRE9NTm9kZSBub2RlKSB7CgkJYm9vbGVhbiByZXN1bHQgPSBmYWxzZTsKCQlpZiAobm9kZSAhPSBudWxsKSB7CgkJCXJlc3VsdCA9ICFTdHJpbmdVdGlscy5jb250YWluc0xpbmVEZWxpbWl0ZXIobm9kZS5nZXROb2RlVmFsdWUoKSk7CgkJfQoJCXJldHVybiByZXN1bHQ7Cgl9CgoJLyoqCgkgKiBDYWxjdWxhdGVzIGlmIHRoZXJlIGlzIGVub3VnaCBzcGFjZSBvbiB0aGUgY3VycmVudCBsaW5lIGZvcgoJICogY29tcHJlc3NlZFRleHQgKGFuZCBmb3IgaXRzIHBhcmVudCBlbmQgdGFnKQoJICogCgkgKiBAcGFyYW0gcGFyZW50Tm9kZQoJICogQHBhcmFtIGN1cnJlbnRBdmFpbGFibGVMaW5lV2lkdGgKCSAqIEBwYXJhbSBjb21wcmVzc2VkVGV4dAoJICogQHJldHVybgoJICovCglwcml2YXRlIGJvb2xlYW4gZW5vdWdoU3BhY2UoSURPTU5vZGUgcGFyZW50Tm9kZSwgaW50IGN1cnJlbnRBdmFpbGFibGVMaW5lV2lkdGgsIFN0cmluZyBjb21wcmVzc2VkVGV4dCkgewoJCWludCBwYXJlbnRFbmRUYWdMZW5ndGggPSBwYXJlbnROb2RlLmdldE5vZGVOYW1lKCkubGVuZ3RoKCkgKyAzOwoJCXJldHVybiBjb21wcmVzc2VkVGV4dC5sZW5ndGgoKSA8PSAoY3VycmVudEF2YWlsYWJsZUxpbmVXaWR0aCAtIHBhcmVudEVuZFRhZ0xlbmd0aCkgJiYgIVN0cmluZ1V0aWxzLmNvbnRhaW5zTGluZURlbGltaXRlcihjb21wcmVzc2VkVGV4dCk7Cgl9CgoJcHJvdGVjdGVkIFZlY3RvciByZWZsb3dUZXh0KFN0cmluZyB0ZXh0LCBpbnQgYXZhaWxhYmxlV2lkdGgpIHsKCQlTdHJpbmdbXSBzdHJpbmdBcnJheSA9IG51bGw7CgkJYm9vbGVhbiBjbGVhckFsbEJsYW5rTGluZXMgPSBnZXRGb3JtYXRQcmVmZXJlbmNlcygpLmdldENsZWFyQWxsQmxhbmtMaW5lcygpOwoKCQlpZiAoY2xlYXJBbGxCbGFua0xpbmVzKQoJCQlzdHJpbmdBcnJheSA9IFN0cmluZ1V0aWxzLmFzQXJyYXkodGV4dCk7CgkJZWxzZQoJCQlzdHJpbmdBcnJheSA9IFN0cmluZ1V0aWxzLmFzQXJyYXkodGV4dCwgREVMSU1JVEVSUywgdHJ1ZSk7CgoJCVZlY3RvciBvdXRwdXQgPSBuZXcgVmVjdG9yKCk7CgkJaWYgKChzdHJpbmdBcnJheSAhPSBudWxsKSAmJiAoc3RyaW5nQXJyYXkubGVuZ3RoID4gMCkpIHsKCQkJU3RyaW5nQnVmZmVyIGJ1ZmZlciA9IG5ldyBTdHJpbmdCdWZmZXIoKTsKCQkJaWYgKHN0cmluZ0FycmF5WzBdLmNvbXBhcmVUbyhDUikgIT0gMCkKCQkJCWJ1ZmZlci5hcHBlbmQoc3RyaW5nQXJyYXlbMF0pOwoJCQlpbnQgYnVmZmVyTGVuZ3RoID0gc3RyaW5nQXJyYXlbMF0udG9TdHJpbmcoKS5sZW5ndGgoKTsKCQkJYm9vbGVhbiBjciA9IHN0cmluZ0FycmF5WzBdLmNvbXBhcmVUbyhDUikgPT0gMDsKCgkJCWZvciAoaW50IGkgPSAxOyBpIDwgc3RyaW5nQXJyYXkubGVuZ3RoOyBpKyspIHsKCQkJCVN0cmluZyBlYWNoU3RyaW5nID0gc3RyaW5nQXJyYXlbaV07CgkJCQlpZiAoKGVhY2hTdHJpbmcuY29tcGFyZVRvKFNQQUNFKSAhPSAwKSAmJiAoZWFjaFN0cmluZy5jb21wYXJlVG8oVEFCKSAhPSAwKSAmJiAoZWFjaFN0cmluZy5jb21wYXJlVG8oRkYpICE9IDApKSB7CgkJCQkJaWYgKChidWZmZXJMZW5ndGggKyAxICsgZWFjaFN0cmluZy5sZW5ndGgoKSA+IGF2YWlsYWJsZVdpZHRoKSB8fCAoZWFjaFN0cmluZy5jb21wYXJlVG8oQ1IpID09IDApIHx8IChlYWNoU3RyaW5nLmNvbXBhcmVUbyhMRikgPT0gMCkpIHsKCQkJCQkJaWYgKChlYWNoU3RyaW5nLmNvbXBhcmVUbyhMRikgPT0gMCkgJiYgY3IpIHsKCQkJCQkJCS8vIGRvIG5vdGhpbmcKCQkJCQkJfQoJCQkJCQllbHNlIHsKCQkJCQkJCW91dHB1dC5hZGQoYnVmZmVyLnRvU3RyaW5nKCkpOwoJCQkJCQkJYnVmZmVyID0gbmV3IFN0cmluZ0J1ZmZlcigpOwoJCQkJCQkJYnVmZmVyTGVuZ3RoID0gMDsKCQkJCQkJfQoJCQkJCQljciA9IGVhY2hTdHJpbmcuY29tcGFyZVRvKENSKSA9PSAwOwoJCQkJCX0KCQkJCQllbHNlIGlmIChidWZmZXIudG9TdHJpbmcoKS50cmltKCkubGVuZ3RoKCkgPiAwKSB7CgkJCQkJCWJ1ZmZlci5hcHBlbmQoU1BBQ0UpOwoJCQkJCQlidWZmZXJMZW5ndGgrKzsKCQkJCQl9CgkJCQkJaWYgKChlYWNoU3RyaW5nLmNvbXBhcmVUbyhDUikgIT0gMCkgJiYgKGVhY2hTdHJpbmcuY29tcGFyZVRvKExGKSAhPSAwKSkgewoJCQkJCQlidWZmZXIuYXBwZW5kKGVhY2hTdHJpbmcpOwoJCQkJCQlidWZmZXJMZW5ndGggPSBidWZmZXJMZW5ndGggKyBlYWNoU3RyaW5nLmxlbmd0aCgpOwoJCQkJCX0KCQkJCX0KCQkJfQoJCQlvdXRwdXQuYWRkKGJ1ZmZlci50b1N0cmluZygpKTsKCQl9CgkJZWxzZQoJCQlvdXRwdXQuYWRkKHRleHQpOwoKCQlyZXR1cm4gb3V0cHV0OwoJfQoKCXByaXZhdGUgU3RyaW5nIGdldExpbmVEZWxpbWl0ZXIoSVN0cnVjdHVyZWREb2N1bWVudCBkb2MsIGludCBub2RlT2Zmc2V0KSB7CgkJaW50IGxpbmUgPSBkb2MuZ2V0TGluZU9mT2Zmc2V0KG5vZGVPZmZzZXQpOwoJCVN0cmluZyBsaW5lRGVsaW1pdGVyID0gZG9jLmdldExpbmVEZWxpbWl0ZXIoKTsKCQl0cnkgewoJCQlpZiAobGluZSA+IDApIHsKCQkJCWxpbmVEZWxpbWl0ZXIgPSBkb2MuZ2V0TGluZURlbGltaXRlcihsaW5lIC0gMSk7CgkJCX0KCQl9CgkJY2F0Y2ggKEJhZExvY2F0aW9uRXhjZXB0aW9uIGUpIHsKCQkJLy8gbG9nIGZvciBub3csIHVubGVzcyB3ZSBmaW5kIHJlYXNvbiBub3QgdG8KCQkJTG9nZ2VyLmxvZyhMb2dnZXIuSU5GTywgZS5nZXRNZXNzYWdlKCkpOwoJCX0KCQkvLyBCVUcxMTU3MTY6IGlmIGNhbm5vdCBnZXQgbGluZSBkZWxpbWl0ZXIgZnJvbSBjdXJyZW50IGxpbmUsIGp1c3QKCQkvLyB1c2UgZGVmYXVsdCBsaW5lIGRlbGltaXRlcgoJCWlmIChsaW5lRGVsaW1pdGVyID09IG51bGwpCgkJCWxpbmVEZWxpbWl0ZXIgPSBkb2MuZ2V0TGluZURlbGltaXRlcigpOwoJCXJldHVybiBsaW5lRGVsaW1pdGVyOwoJfQoKCXByaXZhdGUgaW50IGNvbXB1dGVBdmFpbGFibGVMaW5lV2lkdGgoSVN0cnVjdHVyZWREb2N1bWVudCBkb2MsIGludCBub2RlT2Zmc2V0LCBpbnQgbGluZVdpZHRoKSB7CgkJLy8gY29tcHV0ZSBjdXJyZW50IGF2YWlsYWJsZSBsaW5lIHdpZHRoCgkJaW50IGN1cnJlbnRBdmFpbGFibGVMaW5lV2lkdGggPSAwOwoJCXRyeSB7CgkJCWludCBsaW5lT2Zmc2V0ID0gZG9jLmdldExpbmVJbmZvcm1hdGlvbk9mT2Zmc2V0KG5vZGVPZmZzZXQpLmdldE9mZnNldCgpOwoJCQlTdHJpbmcgdGV4dCA9IGRvYy5nZXQobGluZU9mZnNldCwgbm9kZU9mZnNldCAtIGxpbmVPZmZzZXQpOwoJCQlpbnQgdXNlZFdpZHRoID0gZ2V0SW5kZW50YXRpb25MZW5ndGgodGV4dCk7CgkJCWN1cnJlbnRBdmFpbGFibGVMaW5lV2lkdGggPSBsaW5lV2lkdGggLSB1c2VkV2lkdGg7CgkJfQoJCWNhdGNoIChCYWRMb2NhdGlvbkV4Y2VwdGlvbiBlKSB7CgkJCS8vIGxvZyBmb3Igbm93LCB1bmxlc3Mgd2UgZmluZCByZWFzb24gbm90IHRvCgkJCUxvZ2dlci5sb2coTG9nZ2VyLklORk8sIGUuZ2V0TWVzc2FnZSgpKTsKCQl9CgkJcmV0dXJuIGN1cnJlbnRBdmFpbGFibGVMaW5lV2lkdGg7Cgl9Cgp9Cg==