LyoqCiAqICBMaWNlbnNlZCB0byB0aGUgQXBhY2hlIFNvZnR3YXJlIEZvdW5kYXRpb24gKEFTRikgdW5kZXIgb25lIG9yIG1vcmUKICogIGNvbnRyaWJ1dG9yIGxpY2Vuc2UgYWdyZWVtZW50cy4gIFNlZSB0aGUgTk9USUNFIGZpbGUgZGlzdHJpYnV0ZWQgd2l0aAogKiAgdGhpcyB3b3JrIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIHJlZ2FyZGluZyBjb3B5cmlnaHQgb3duZXJzaGlwLgogKiAgVGhlIEFTRiBsaWNlbnNlcyB0aGlzIGZpbGUgdG8gWW91IHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAKICogICh0aGUgIkxpY2Vuc2UiKTsgeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoCiAqICB0aGUgTGljZW5zZS4gIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAogKgogKiAgICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCiAqCiAqICBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCiAqICBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLAogKiAgV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuCiAqICBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCiAqICBsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4KICovCgpwYWNrYWdlIG9yZy5hcGFjaGUuZ2Vyb25pbW8udHJhbnNhY3Rpb24ubWFuYWdlcjsKCmltcG9ydCBqYXZhLmlvLlByaW50V3JpdGVyOwppbXBvcnQgamF2YS5pby5TdHJpbmdXcml0ZXI7CmltcG9ydCBqYXZhLmlvLldyaXRlcjsKaW1wb3J0IGphdmEudXRpbC5BcnJheUxpc3Q7CmltcG9ydCBqYXZhLnV0aWwuSGFzaE1hcDsKaW1wb3J0IGphdmEudXRpbC5JZGVudGl0eUhhc2hNYXA7CmltcG9ydCBqYXZhLnV0aWwuSXRlcmF0b3I7CmltcG9ydCBqYXZhLnV0aWwuTGlua2VkTGlzdDsKaW1wb3J0IGphdmEudXRpbC5MaXN0OwppbXBvcnQgamF2YS51dGlsLk1hcDsKaW1wb3J0IGphdmEudXRpbC5TZXQ7CgppbXBvcnQgamF2YXgudHJhbnNhY3Rpb24uSGV1cmlzdGljTWl4ZWRFeGNlcHRpb247CmltcG9ydCBqYXZheC50cmFuc2FjdGlvbi5IZXVyaXN0aWNSb2xsYmFja0V4Y2VwdGlvbjsKaW1wb3J0IGphdmF4LnRyYW5zYWN0aW9uLlJvbGxiYWNrRXhjZXB0aW9uOwppbXBvcnQgamF2YXgudHJhbnNhY3Rpb24uU3RhdHVzOwppbXBvcnQgamF2YXgudHJhbnNhY3Rpb24uU3luY2hyb25pemF0aW9uOwppbXBvcnQgamF2YXgudHJhbnNhY3Rpb24uU3lzdGVtRXhjZXB0aW9uOwppbXBvcnQgamF2YXgudHJhbnNhY3Rpb24uVHJhbnNhY3Rpb247CmltcG9ydCBqYXZheC50cmFuc2FjdGlvbi54YS5YQUV4Y2VwdGlvbjsKaW1wb3J0IGphdmF4LnRyYW5zYWN0aW9uLnhhLlhBUmVzb3VyY2U7CmltcG9ydCBqYXZheC50cmFuc2FjdGlvbi54YS5YaWQ7CgppbXBvcnQgb3JnLnNsZjRqLkxvZ2dlcjsKaW1wb3J0IG9yZy5zbGY0ai5Mb2dnZXJGYWN0b3J5OwoKCi8qKgogKiBCYXNpYyBsb2NhbCB0cmFuc2FjdGlvbiB3aXRoIHN1cHBvcnQgZm9yIG11bHRpcGxlIHJlc291cmNlcy4KICoKICogQHZlcnNpb24gJFJldiQgJERhdGUkCiAqLwpwdWJsaWMgY2xhc3MgVHJhbnNhY3Rpb25JbXBsIGltcGxlbWVudHMgVHJhbnNhY3Rpb24gewogICAgcHJpdmF0ZSBzdGF0aWMgZmluYWwgTG9nZ2VyIGxvZyA9IExvZ2dlckZhY3RvcnkuZ2V0TG9nZ2VyKCJUcmFuc2FjdGlvbiIpOwoKICAgIHByaXZhdGUgZmluYWwgWGlkRmFjdG9yeSB4aWRGYWN0b3J5OwogICAgcHJpdmF0ZSBmaW5hbCBYaWQgeGlkOwogICAgcHJpdmF0ZSBmaW5hbCBUcmFuc2FjdGlvbkxvZyB0eG5Mb2c7CiAgICBwcml2YXRlIGZpbmFsIGxvbmcgdGltZW91dDsKICAgIHByaXZhdGUgZmluYWwgTGlzdCBzeW5jTGlzdCA9IG5ldyBBcnJheUxpc3QoNSk7CiAgICBwcml2YXRlIGZpbmFsIExpc3QgaW50ZXJwb3NlZFN5bmNMaXN0ID0gbmV3IEFycmF5TGlzdCgzKTsKICAgIHByaXZhdGUgZmluYWwgTGlua2VkTGlzdCByZXNvdXJjZU1hbmFnZXJzID0gbmV3IExpbmtlZExpc3QoKTsKICAgIHByaXZhdGUgZmluYWwgSWRlbnRpdHlIYXNoTWFwIGFjdGl2ZVhhUmVzb3VyY2VzID0gbmV3IElkZW50aXR5SGFzaE1hcCgzKTsKICAgIHByaXZhdGUgZmluYWwgSWRlbnRpdHlIYXNoTWFwIHN1c3BlbmRlZFhhUmVzb3VyY2VzID0gbmV3IElkZW50aXR5SGFzaE1hcCgzKTsKICAgIHByaXZhdGUgaW50IHN0YXR1cyA9IFN0YXR1cy5TVEFUVVNfTk9fVFJBTlNBQ1RJT047CiAgICBwcml2YXRlIE9iamVjdCBsb2dNYXJrOwoKICAgIHByaXZhdGUgZmluYWwgTWFwIHJlc291cmNlcyA9IG5ldyBIYXNoTWFwKCk7CgogICAgVHJhbnNhY3Rpb25JbXBsKFhpZEZhY3RvcnkgeGlkRmFjdG9yeSwgVHJhbnNhY3Rpb25Mb2cgdHhuTG9nLCBsb25nIHRyYW5zYWN0aW9uVGltZW91dE1pbGxpc2Vjb25kcykgdGhyb3dzIFN5c3RlbUV4Y2VwdGlvbiB7CiAgICAgICAgdGhpcyh4aWRGYWN0b3J5LmNyZWF0ZVhpZCgpLCB4aWRGYWN0b3J5LCB0eG5Mb2csIHRyYW5zYWN0aW9uVGltZW91dE1pbGxpc2Vjb25kcyk7CiAgICB9CgogICAgVHJhbnNhY3Rpb25JbXBsKFhpZCB4aWQsIFhpZEZhY3RvcnkgeGlkRmFjdG9yeSwgVHJhbnNhY3Rpb25Mb2cgdHhuTG9nLCBsb25nIHRyYW5zYWN0aW9uVGltZW91dE1pbGxpc2Vjb25kcykgdGhyb3dzIFN5c3RlbUV4Y2VwdGlvbiB7CiAgICAgICAgdGhpcy54aWRGYWN0b3J5ID0geGlkRmFjdG9yeTsKICAgICAgICB0aGlzLnR4bkxvZyA9IHR4bkxvZzsKICAgICAgICB0aGlzLnhpZCA9IHhpZDsKICAgICAgICB0aGlzLnRpbWVvdXQgPSB0cmFuc2FjdGlvblRpbWVvdXRNaWxsaXNlY29uZHMgKyBUcmFuc2FjdGlvblRpbWVyLmdldEN1cnJlbnRUaW1lKCk7CiAgICAgICAgdHJ5IHsKICAgICAgICAgICAgdHhuTG9nLmJlZ2luKHhpZCk7CiAgICAgICAgfSBjYXRjaCAoTG9nRXhjZXB0aW9uIGUpIHsKICAgICAgICAgICAgc3RhdHVzID0gU3RhdHVzLlNUQVRVU19NQVJLRURfUk9MTEJBQ0s7CiAgICAgICAgICAgIFN5c3RlbUV4Y2VwdGlvbiBleCA9IG5ldyBTeXN0ZW1FeGNlcHRpb24oIkVycm9yIGxvZ2dpbmcgYmVnaW47IHRyYW5zYWN0aW9uIG1hcmtlZCBmb3Igcm9sbCBiYWNrKSIpOwogICAgICAgICAgICBleC5pbml0Q2F1c2UoZSk7CiAgICAgICAgICAgIHRocm93IGV4OwogICAgICAgIH0KICAgICAgICBzdGF0dXMgPSBTdGF0dXMuU1RBVFVTX0FDVElWRTsKICAgIH0KCiAgICAvL3JlY29uc3RydWN0IGEgdHggZm9yIGFuIGV4dGVybmFsIHR4IGZvdW5kIGluIHJlY292ZXJ5CiAgICBwdWJsaWMgVHJhbnNhY3Rpb25JbXBsKFhpZCB4aWQsIFRyYW5zYWN0aW9uTG9nIHR4TG9nKSB7CiAgICAgICAgdGhpcy54aWRGYWN0b3J5ID0gbnVsbDsKICAgICAgICB0aGlzLnR4bkxvZyA9IHR4TG9nOwogICAgICAgIHRoaXMueGlkID0geGlkOwogICAgICAgIHN0YXR1cyA9IFN0YXR1cy5TVEFUVVNfUFJFUEFSRUQ7CiAgICAgICAgLy9UT0RPIGlzIHRoaXMgYSBnb29kIGlkZWE/CiAgICAgICAgdGhpcy50aW1lb3V0ID0gTG9uZy5NQVhfVkFMVUU7CiAgICB9CgogICAgcHVibGljIHN5bmNocm9uaXplZCBpbnQgZ2V0U3RhdHVzKCkgewogICAgICAgIHJldHVybiBzdGF0dXM7CiAgICB9CgogICAgcHVibGljIE9iamVjdCBnZXRSZXNvdXJjZShPYmplY3Qga2V5KSB7CiAgICAgICAgcmV0dXJuIHJlc291cmNlcy5nZXQoa2V5KTsKICAgIH0KCiAgICBwdWJsaWMgYm9vbGVhbiBnZXRSb2xsYmFja09ubHkoKSB7CiAgICAgICAgcmV0dXJuIHN0YXR1cyA9PSBTdGF0dXMuU1RBVFVTX01BUktFRF9ST0xMQkFDSzsKICAgIH0KCiAgICBwdWJsaWMgT2JqZWN0IGdldFRyYW5zYWN0aW9uS2V5KCkgewogICAgICAgIHJldHVybiB4aWQ7CiAgICB9CgogICAgcHVibGljIGludCBnZXRUcmFuc2FjdGlvblN0YXR1cygpIHsKICAgICAgICByZXR1cm4gc3RhdHVzOwogICAgfQoKICAgIHB1YmxpYyB2b2lkIHB1dFJlc291cmNlKE9iamVjdCBrZXksIE9iamVjdCB2YWx1ZSkgewogICAgICAgIGlmIChrZXkgPT0gbnVsbCkgewogICAgICAgICAgICB0aHJvdyBuZXcgTnVsbFBvaW50ZXJFeGNlcHRpb24oIllvdSBtdXN0IHN1cHBseSBhIG5vbi1udWxsIGtleSBmb3IgcHV0UmVzb3VyY2UiKTsKICAgICAgICB9CiAgICAgICAgcmVzb3VyY2VzLnB1dChrZXksIHZhbHVlKTsKICAgIH0KCiAgICBwdWJsaWMgdm9pZCByZWdpc3RlckludGVycG9zZWRTeW5jaHJvbml6YXRpb24oU3luY2hyb25pemF0aW9uIHN5bmNocm9uaXphdGlvbikgewogICAgICAgIGludGVycG9zZWRTeW5jTGlzdC5hZGQoc3luY2hyb25pemF0aW9uKTsKICAgIH0KCiAgICBwdWJsaWMgc3luY2hyb25pemVkIHZvaWQgc2V0Um9sbGJhY2tPbmx5KCkgdGhyb3dzIElsbGVnYWxTdGF0ZUV4Y2VwdGlvbiB7CiAgICAgICAgc3dpdGNoIChzdGF0dXMpIHsKICAgICAgICAgICAgY2FzZSBTdGF0dXMuU1RBVFVTX0FDVElWRToKICAgICAgICAgICAgY2FzZSBTdGF0dXMuU1RBVFVTX1BSRVBBUklORzoKICAgICAgICAgICAgICAgIHN0YXR1cyA9IFN0YXR1cy5TVEFUVVNfTUFSS0VEX1JPTExCQUNLOwogICAgICAgICAgICAgICAgYnJlYWs7CiAgICAgICAgICAgIGNhc2UgU3RhdHVzLlNUQVRVU19NQVJLRURfUk9MTEJBQ0s6CiAgICAgICAgICAgIGNhc2UgU3RhdHVzLlNUQVRVU19ST0xMSU5HX0JBQ0s6CiAgICAgICAgICAgICAgICAvLyBub3RoaW5nIHRvIGRvCiAgICAgICAgICAgICAgICBicmVhazsKICAgICAgICAgICAgZGVmYXVsdDoKICAgICAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsU3RhdGVFeGNlcHRpb24oIkNhbm5vdCBzZXQgcm9sbGJhY2sgb25seSwgc3RhdHVzIGlzICIgKyBnZXRTdGF0ZVN0cmluZyhzdGF0dXMpKTsKICAgICAgICB9CiAgICB9CgogICAgcHVibGljIHN5bmNocm9uaXplZCB2b2lkIHJlZ2lzdGVyU3luY2hyb25pemF0aW9uKFN5bmNocm9uaXphdGlvbiBzeW5jaCkgdGhyb3dzIElsbGVnYWxTdGF0ZUV4Y2VwdGlvbiwgUm9sbGJhY2tFeGNlcHRpb24sIFN5c3RlbUV4Y2VwdGlvbiB7CiAgICAgICAgaWYgKHN5bmNoID09IG51bGwpIHsKICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbigiU3luY2hyb25pemF0aW9uIGlzIG51bGwiKTsKICAgICAgICB9CiAgICAgICAgc3dpdGNoIChzdGF0dXMpIHsKICAgICAgICAgICAgY2FzZSBTdGF0dXMuU1RBVFVTX0FDVElWRToKICAgICAgICAgICAgY2FzZSBTdGF0dXMuU1RBVFVTX1BSRVBBUklORzoKICAgICAgICAgICAgICAgIGJyZWFrOwogICAgICAgICAgICBjYXNlIFN0YXR1cy5TVEFUVVNfTUFSS0VEX1JPTExCQUNLOgogICAgICAgICAgICAgICAgdGhyb3cgbmV3IFJvbGxiYWNrRXhjZXB0aW9uKCJUcmFuc2FjdGlvbiBpcyBtYXJrZWQgZm9yIHJvbGxiYWNrIik7CiAgICAgICAgICAgIGRlZmF1bHQ6CiAgICAgICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbFN0YXRlRXhjZXB0aW9uKCJTdGF0dXMgaXMgIiArIGdldFN0YXRlU3RyaW5nKHN0YXR1cykpOwogICAgICAgIH0KICAgICAgICBzeW5jTGlzdC5hZGQoc3luY2gpOwogICAgfQoKICAgIHB1YmxpYyBzeW5jaHJvbml6ZWQgYm9vbGVhbiBlbmxpc3RSZXNvdXJjZShYQVJlc291cmNlIHhhUmVzKSB0aHJvd3MgSWxsZWdhbFN0YXRlRXhjZXB0aW9uLCBSb2xsYmFja0V4Y2VwdGlvbiwgU3lzdGVtRXhjZXB0aW9uIHsKICAgICAgICBpZiAoeGFSZXMgPT0gbnVsbCkgewogICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbEFyZ3VtZW50RXhjZXB0aW9uKCJYQVJlc291cmNlIGlzIG51bGwiKTsKICAgICAgICB9CiAgICAgICAgc3dpdGNoIChzdGF0dXMpIHsKICAgICAgICAgICAgY2FzZSBTdGF0dXMuU1RBVFVTX0FDVElWRToKICAgICAgICAgICAgICAgIGJyZWFrOwogICAgICAgICAgICBjYXNlIFN0YXR1cy5TVEFUVVNfTUFSS0VEX1JPTExCQUNLOgogICAgICAgICAgICAgICAgYnJlYWs7CiAgICAgICAgICAgIGRlZmF1bHQ6CiAgICAgICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbFN0YXRlRXhjZXB0aW9uKCJTdGF0dXMgaXMgIiArIGdldFN0YXRlU3RyaW5nKHN0YXR1cykpOwogICAgICAgIH0KCiAgICAgICAgaWYgKGFjdGl2ZVhhUmVzb3VyY2VzLmNvbnRhaW5zS2V5KHhhUmVzKSkgewogICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbFN0YXRlRXhjZXB0aW9uKCJ4YXJlc291cmNlOiAiICsgeGFSZXMgKyAiIGlzIGFscmVhZHkgZW5saXN0ZWQhIik7CiAgICAgICAgfQoKICAgICAgICB0cnkgewogICAgICAgICAgICBUcmFuc2FjdGlvbkJyYW5jaCBtYW5hZ2VyID0gKFRyYW5zYWN0aW9uQnJhbmNoKSBzdXNwZW5kZWRYYVJlc291cmNlcy5yZW1vdmUoeGFSZXMpOwogICAgICAgICAgICBpZiAobWFuYWdlciAhPSBudWxsKSB7CiAgICAgICAgICAgICAgICAvL3dlIGtub3cgYWJvdXQgdGhpcyBvbmUsIGl0IHdhcyBzdXNwZW5kZWQKICAgICAgICAgICAgICAgIHhhUmVzLnN0YXJ0KG1hbmFnZXIuZ2V0QnJhbmNoSWQoKSwgWEFSZXNvdXJjZS5UTVJFU1VNRSk7CiAgICAgICAgICAgICAgICBhY3RpdmVYYVJlc291cmNlcy5wdXQoeGFSZXMsIG1hbmFnZXIpOwogICAgICAgICAgICAgICAgcmV0dXJuIHRydWU7CiAgICAgICAgICAgIH0KICAgICAgICAgICAgLy9pdCBpcyBub3Qgc3VzcGVuZGVkLgogICAgICAgICAgICBmb3IgKEl0ZXJhdG9yIGkgPSByZXNvdXJjZU1hbmFnZXJzLml0ZXJhdG9yKCk7IGkuaGFzTmV4dCgpOykgewogICAgICAgICAgICAgICAgbWFuYWdlciA9IChUcmFuc2FjdGlvbkJyYW5jaCkgaS5uZXh0KCk7CiAgICAgICAgICAgICAgICBib29sZWFuIHNhbWVSTTsKICAgICAgICAgICAgICAgIC8vaWYgdGhlIHhhcmVzIGlzIGFscmVhZHkga25vd24sIHdlIG11c3QgYmUgcmVzdW1pbmcgYWZ0ZXIgYSBzdXNwZW5kLgogICAgICAgICAgICAgICAgaWYgKHhhUmVzID09IG1hbmFnZXIuZ2V0Q29tbWl0dGVyKCkpIHsKICAgICAgICAgICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbFN0YXRlRXhjZXB0aW9uKCJ4YVJlcyAiICsgeGFSZXMgKyAiIGlzIGEgY29tbWl0dGVyIGJ1dCBpcyBub3QgYWN0aXZlIG9yIHN1c3BlbmRlZCIpOwogICAgICAgICAgICAgICAgfQogICAgICAgICAgICAgICAgLy9PdGhlcndpc2UsIHNlZSBpZiB0aGlzIGlzIGEgbmV3IHhhcmVzIGZvciB0aGUgc2FtZSByZXNvdXJjZSBtYW5hZ2VyCiAgICAgICAgICAgICAgICB0cnkgewogICAgICAgICAgICAgICAgICAgIHNhbWVSTSA9IHhhUmVzLmlzU2FtZVJNKG1hbmFnZXIuZ2V0Q29tbWl0dGVyKCkpOwogICAgICAgICAgICAgICAgfSBjYXRjaCAoWEFFeGNlcHRpb24gZSkgewogICAgICAgICAgICAgICAgICAgIGxvZy53YXJuKCJVbmV4cGVjdGVkIGVycm9yIGNoZWNraW5nIGZvciBzYW1lIFJNIiwgZSk7CiAgICAgICAgICAgICAgICAgICAgY29udGludWU7CiAgICAgICAgICAgICAgICB9CiAgICAgICAgICAgICAgICBpZiAoc2FtZVJNKSB7CiAgICAgICAgICAgICAgICAgICAgeGFSZXMuc3RhcnQobWFuYWdlci5nZXRCcmFuY2hJZCgpLCBYQVJlc291cmNlLlRNSk9JTik7CiAgICAgICAgICAgICAgICAgICAgYWN0aXZlWGFSZXNvdXJjZXMucHV0KHhhUmVzLCBtYW5hZ2VyKTsKICAgICAgICAgICAgICAgICAgICByZXR1cm4gdHJ1ZTsKICAgICAgICAgICAgICAgIH0KICAgICAgICAgICAgfQogICAgICAgICAgICAvL3dlIGtub3cgbm90aGluZyBhYm91dCB0aGlzIFhBUmVzb3VyY2Ugb3IgcmVzb3VyY2UgbWFuYWdlcgogICAgICAgICAgICBYaWQgYnJhbmNoSWQgPSB4aWRGYWN0b3J5LmNyZWF0ZUJyYW5jaCh4aWQsIHJlc291cmNlTWFuYWdlcnMuc2l6ZSgpICsgMSk7CiAgICAgICAgICAgIHhhUmVzLnN0YXJ0KGJyYW5jaElkLCBYQVJlc291cmNlLlRNTk9GTEFHUyk7CiAgICAgICAgICAgIGFjdGl2ZVhhUmVzb3VyY2VzLnB1dCh4YVJlcywgYWRkQnJhbmNoWGlkKHhhUmVzLCBicmFuY2hJZCkpOwogICAgICAgICAgICByZXR1cm4gdHJ1ZTsKICAgICAgICB9IGNhdGNoIChYQUV4Y2VwdGlvbiBlKSB7CiAgICAgICAgICAgIGxvZy53YXJuKCJVbmFibGUgdG8gZW5saXN0IFhBUmVzb3VyY2UgIiArIHhhUmVzICsgIiwgZXJyb3JDb2RlOiAiICsgZS5lcnJvckNvZGUsIGUpOwogICAgICAgICAgICAvLyBtYXJrIHN0YXR1cyBhcyByb2xsYmFjayBvbmx5IGJlY2F1c2UgZW5saXN0IHJlc291cmNlIGZhaWxlZAogICAgICAgICAgICBzZXRSb2xsYmFja09ubHkoKTsKICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwogICAgICAgIH0KICAgIH0KCiAgICBwdWJsaWMgc3luY2hyb25pemVkIGJvb2xlYW4gZGVsaXN0UmVzb3VyY2UoWEFSZXNvdXJjZSB4YVJlcywgaW50IGZsYWcpIHRocm93cyBJbGxlZ2FsU3RhdGVFeGNlcHRpb24sIFN5c3RlbUV4Y2VwdGlvbiB7CiAgICAgICAgaWYgKCEoZmxhZyA9PSBYQVJlc291cmNlLlRNRkFJTCB8fCBmbGFnID09IFhBUmVzb3VyY2UuVE1TVUNDRVNTIHx8IGZsYWcgPT0gWEFSZXNvdXJjZS5UTVNVU1BFTkQpKSB7CiAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsU3RhdGVFeGNlcHRpb24oImludmFsaWQgZmxhZyBmb3IgZGVsaXN0UmVzb3VyY2U6ICIgKyBmbGFnKTsKICAgICAgICB9CiAgICAgICAgaWYgKHhhUmVzID09IG51bGwpIHsKICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxBcmd1bWVudEV4Y2VwdGlvbigiWEFSZXNvdXJjZSBpcyBudWxsIik7CiAgICAgICAgfQogICAgICAgIHN3aXRjaCAoc3RhdHVzKSB7CiAgICAgICAgICAgIGNhc2UgU3RhdHVzLlNUQVRVU19BQ1RJVkU6CiAgICAgICAgICAgIGNhc2UgU3RhdHVzLlNUQVRVU19NQVJLRURfUk9MTEJBQ0s6CiAgICAgICAgICAgICAgICBicmVhazsKICAgICAgICAgICAgZGVmYXVsdDoKICAgICAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsU3RhdGVFeGNlcHRpb24oIlN0YXR1cyBpcyAiICsgZ2V0U3RhdGVTdHJpbmcoc3RhdHVzKSk7CiAgICAgICAgfQogICAgICAgIFRyYW5zYWN0aW9uQnJhbmNoIG1hbmFnZXIgPSAoVHJhbnNhY3Rpb25CcmFuY2gpIGFjdGl2ZVhhUmVzb3VyY2VzLnJlbW92ZSh4YVJlcyk7CiAgICAgICAgaWYgKG1hbmFnZXIgPT0gbnVsbCkgewogICAgICAgICAgICBpZiAoZmxhZyA9PSBYQVJlc291cmNlLlRNU1VTUEVORCkgewogICAgICAgICAgICAgICAgdGhyb3cgbmV3IElsbGVnYWxTdGF0ZUV4Y2VwdGlvbigidHJ5aW5nIHRvIHN1c3BlbmQgYW4gaW5hY3RpdmUgeGFyZXNvdXJjZTogIiArIHhhUmVzKTsKICAgICAgICAgICAgfQogICAgICAgICAgICAvL25vdCBhY3RpdmUsIGFuZCB3ZSBhcmUgbm90IHRyeWluZyB0byBzdXNwZW5kLiAgV2UgbXVzdCBiZSBlbmRpbmcgdHguCiAgICAgICAgICAgIG1hbmFnZXIgPSAoVHJhbnNhY3Rpb25CcmFuY2gpIHN1c3BlbmRlZFhhUmVzb3VyY2VzLnJlbW92ZSh4YVJlcyk7CiAgICAgICAgICAgIGlmIChtYW5hZ2VyID09IG51bGwpIHsKICAgICAgICAgICAgICAgIHRocm93IG5ldyBJbGxlZ2FsU3RhdGVFeGNlcHRpb24oIlJlc291cmNlIG5vdCBrbm93biB0byB0cmFuc2FjdGlvbjogIiArIHhhUmVzKTsKICAgICAgICAgICAgfQogICAgICAgIH0KCiAgICAgICAgdHJ5IHsKICAgICAgICAgICAgeGFSZXMuZW5kKG1hbmFnZXIuZ2V0QnJhbmNoSWQoKSwgZmxhZyk7CiAgICAgICAgICAgIGlmIChmbGFnID09IFhBUmVzb3VyY2UuVE1TVVNQRU5EKSB7CiAgICAgICAgICAgICAgICBzdXNwZW5kZWRYYVJlc291cmNlcy5wdXQoeGFSZXMsIG1hbmFnZXIpOwogICAgICAgICAgICB9CiAgICAgICAgICAgIHJldHVybiB0cnVlOwogICAgICAgIH0gY2F0Y2ggKFhBRXhjZXB0aW9uIGUpIHsKICAgICAgICAgICAgbG9nLndhcm4oIlVuYWJsZSB0byBkZWxpc3QgWEFSZXNvdXJjZSAiICsgeGFSZXMgKyAiLCBlcnJvciBjb2RlOiAiICsgZS5lcnJvckNvZGUsIGUpOwogICAgICAgICAgICByZXR1cm4gZmFsc2U7CiAgICAgICAgfQogICAgfQoKICAgIC8vVHJhbnNhY3Rpb24gbWV0aG9kLCBkb2VzIDJwYwogICAgcHVibGljIHZvaWQgY29tbWl0KCkgdGhyb3dzIEhldXJpc3RpY01peGVkRXhjZXB0aW9uLCBIZXVyaXN0aWNSb2xsYmFja0V4Y2VwdGlvbiwgUm9sbGJhY2tFeGNlcHRpb24sIFNlY3VyaXR5RXhjZXB0aW9uLCBTeXN0ZW1FeGNlcHRpb24gewogICAgICAgIGJlZm9yZVByZXBhcmUoKTsKCiAgICAgICAgdHJ5IHsKICAgICAgICAgICAgYm9vbGVhbiB0aW1lZG91dCA9IGZhbHNlOwogICAgICAgICAgICBpZiAoVHJhbnNhY3Rpb25UaW1lci5nZXRDdXJyZW50VGltZSgpID4gdGltZW91dCkgewogICAgICAgICAgICAgICAgc3RhdHVzID0gU3RhdHVzLlNUQVRVU19NQVJLRURfUk9MTEJBQ0s7CiAgICAgICAgICAgICAgICB0aW1lZG91dCA9IHRydWU7CiAgICAgICAgICAgIH0gCgogICAgICAgICAgICBpZiAoc3RhdHVzID09IFN0YXR1cy5TVEFUVVNfTUFSS0VEX1JPTExCQUNLKSB7CiAgICAgICAgICAgICAgICByb2xsYmFja1Jlc291cmNlc0R1cmluZ0NvbW1pdChyZXNvdXJjZU1hbmFnZXJzLCBmYWxzZSk7CiAgICAgICAgICAgICAgICBpZiAodGltZWRvdXQpIHsKICAgICAgICAgICAgICAgICAgICB0aHJvdyBuZXcgUm9sbGJhY2tFeGNlcHRpb24oIlVuYWJsZSB0byBjb21taXQ6IFRyYW5zYWN0aW9uIHRpbWVvdXQiKTsKICAgICAgICAgICAgICAgIH0gZWxzZSB7CiAgICAgICAgICAgICAgICAgICAgdGhyb3cgbmV3IFJvbGxiYWNrRXhjZXB0aW9uKCJVbmFibGUgdG8gY29tbWl0OiB0cmFuc2FjdGlvbiBtYXJrZWQgZm9yIHJvbGxiYWNrIik7CiAgICAgICAgICAgICAgICB9CiAgICAgICAgICAgIH0KICAgICAgICAgICAgc3luY2hyb25pemVkICh0aGlzKSB7CiAgICAgICAgICAgICAgICBpZiAoc3RhdHVzID09IFN0YXR1cy5TVEFUVVNfQUNUSVZFKSB7CiAgICAgICAgICAgICAgICAgICAgaWYgKHRoaXMucmVzb3VyY2VNYW5hZ2Vycy5zaXplKCkgPT0gMCkgewogICAgICAgICAgICAgICAgICAgICAgICAvLyBub3RoaW5nIHRvIGNvbW1pdAogICAgICAgICAgICAgICAgICAgICAgICBzdGF0dXMgPSBTdGF0dXMuU1RBVFVTX0NPTU1JVFRFRDsKICAgICAgICAgICAgICAgICAgICB9IGVsc2UgaWYgKHRoaXMucmVzb3VyY2VNYW5hZ2Vycy5zaXplKCkgPT0gMSkgewogICAgICAgICAgICAgICAgICAgICAgICAvLyBvbmUtcGhhc2UgY29tbWl0IGRlY2lzaW9uCiAgICAgICAgICAgICAgICAgICAgICAgIHN0YXR1cyA9IFN0YXR1cy5TVEFUVVNfQ09NTUlUVElORzsKICAgICAgICAgICAgICAgICAgICB9IGVsc2UgewogICAgICAgICAgICAgICAgICAgICAgICAvLyBzdGFydCBwcmVwYXJlIHBhcnQgb2YgdHdvLXBoYXNlCiAgICAgICAgICAgICAgICAgICAgICAgIHN0YXR1cyA9IFN0YXR1cy5TVEFUVVNfUFJFUEFSSU5HOwogICAgICAgICAgICAgICAgICAgIH0KICAgICAgICAgICAgICAgIH0KICAgICAgICAgICAgICAgIC8vIHJlc291cmNlTWFuYWdlcnMgaXMgbm93IGltbXV0YWJsZQogICAgICAgICAgICB9CgogICAgICAgICAgICAvLyBuby1waGFzZQogICAgICAgICAgICBpZiAocmVzb3VyY2VNYW5hZ2Vycy5zaXplKCkgPT0gMCkgewogICAgICAgICAgICAgICAgc3luY2hyb25pemVkICh0aGlzKSB7CiAgICAgICAgICAgICAgICAgICAgc3RhdHVzID0gU3RhdHVzLlNUQVRVU19DT01NSVRURUQ7CiAgICAgICAgICAgICAgICB9CiAgICAgICAgICAgICAgICByZXR1cm47CiAgICAgICAgICAgIH0KCiAgICAgICAgICAgIC8vIG9uZS1waGFzZQogICAgICAgICAgICBpZiAocmVzb3VyY2VNYW5hZ2Vycy5zaXplKCkgPT0gMSkgewogICAgICAgICAgICAgICAgVHJhbnNhY3Rpb25CcmFuY2ggbWFuYWdlciA9IChUcmFuc2FjdGlvbkJyYW5jaCkgcmVzb3VyY2VNYW5hZ2Vycy5nZXRGaXJzdCgpOwogICAgICAgICAgICAgICAgY29tbWl0UmVzb3VyY2UobWFuYWdlcik7CiAgICAgICAgICAgICAgICByZXR1cm47CiAgICAgICAgICAgIH0gCiAgICAgICAgICAgIAogICAgICAgICAgICBib29sZWFuIHdpbGxDb21taXQgPSBmYWxzZTsKICAgICAgICAgICAgdHJ5IHsKICAgICAgICAgICAgICAgIC8vIHR3by1waGFzZQogICAgICAgICAgICAgICAgd2lsbENvbW1pdCA9IGludGVybmFsUHJlcGFyZSgpOwogICAgICAgICAgICB9IGNhdGNoIChTeXN0ZW1FeGNlcHRpb24gZSkgewogICAgICAgICAgICAgICAgcm9sbGJhY2tSZXNvdXJjZXMocmVzb3VyY2VNYW5hZ2Vycyk7CiAgICAgICAgICAgICAgICB0aHJvdyBlOwogICAgICAgICAgICB9CiAgICAgICAgICAgIAogICAgICAgICAgICAvLyBub3RpZnkgdGhlIFJNcwogICAgICAgICAgICBpZiAod2lsbENvbW1pdCkgewogICAgICAgICAgICAgICAgY29tbWl0UmVzb3VyY2VzKHJlc291cmNlTWFuYWdlcnMpOwogICAgICAgICAgICB9IGVsc2UgewogICAgICAgICAgICAgICAgLy8gc2V0IGV2ZXJSb2xsYmFjayB0byB0cnVlIGhlcmUgYmVjYXVzZSB0aGUgcm9sbGJhY2sgaGVyZSBpcyBjYXVzZWQgYnkgCiAgICAgICAgICAgICAgICAvLyBYQUV4Y2VwdGlvbiBkdXJpbmcgdGhlIGFib3ZlIGludGVybmFsUHJlcGFyZQogICAgICAgICAgICAgICAgcm9sbGJhY2tSZXNvdXJjZXNEdXJpbmdDb21taXQocmVzb3VyY2VNYW5hZ2VycywgdHJ1ZSk7CiAgICAgICAgICAgIH0KICAgICAgICB9IGZpbmFsbHkgewogICAgICAgICAgICBhZnRlckNvbXBsZXRpb24oKTsKICAgICAgICAgICAgc3luY2hyb25pemVkICh0aGlzKSB7CiAgICAgICAgICAgICAgICBzdGF0dXMgPSBTdGF0dXMuU1RBVFVTX05PX1RSQU5TQUNUSU9OOwogICAgICAgICAgICB9CiAgICAgICAgfQogICAgfQoKICAgIC8vVXNlZCBmcm9tIFhBVGVybWluYXRvciBmb3IgZmlyc3QgcGhhc2UgaW4gYSByZW1vdGVseSBjb250cm9sbGVkIHR4LgogICAgaW50IHByZXBhcmUoKSB0aHJvd3MgU3lzdGVtRXhjZXB0aW9uLCBSb2xsYmFja0V4Y2VwdGlvbiB7CiAgICAgICAgYmVmb3JlUHJlcGFyZSgpOwogICAgICAgIGludCByZXN1bHQgPSBYQVJlc291cmNlLlhBX1JET05MWTsKICAgICAgICB0cnkgewogICAgICAgICAgICBMaW5rZWRMaXN0IHJtczsKICAgICAgICAgICAgc3luY2hyb25pemVkICh0aGlzKSB7CiAgICAgICAgICAgICAgICBpZiAoc3RhdHVzID09IFN0YXR1cy5TVEFUVVNfQUNUSVZFKSB7CiAgICAgICAgICAgICAgICAgICAgaWYgKHJlc291cmNlTWFuYWdlcnMuc2l6ZSgpID09IDApIHsKICAgICAgICAgICAgICAgICAgICAgICAgLy8gbm90aGluZyB0byBjb21taXQKICAgICAgICAgICAgICAgICAgICAgICAgc3RhdHVzID0gU3RhdHVzLlNUQVRVU19DT01NSVRURUQ7CiAgICAgICAgICAgICAgICAgICAgICAgIHJldHVybiByZXN1bHQ7CiAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHsKICAgICAgICAgICAgICAgICAgICAgICAgLy8gc3RhcnQgcHJlcGFyZSBwYXJ0IG9mIHR3by1waGFzZQogICAgICAgICAgICAgICAgICAgICAgICBzdGF0dXMgPSBTdGF0dXMuU1RBVFVTX1BSRVBBUklORzsKICAgICAgICAgICAgICAgICAgICB9CiAgICAgICAgICAgICAgICB9CiAgICAgICAgICAgICAgICAvLyByZXNvdXJjZU1hbmFnZXJzIGlzIG5vdyBpbW11dGFibGUKICAgICAgICAgICAgICAgIHJtcyA9IHJlc291cmNlTWFuYWdlcnM7CiAgICAgICAgICAgIH0KCiAgICAgICAgICAgIGJvb2xlYW4gd2lsbENvbW1pdCA9IGludGVybmFsUHJlcGFyZSgpOwoKICAgICAgICAgICAgLy8gbm90aWZ5IHRoZSBSTXMKICAgICAgICAgICAgaWYgKHdpbGxDb21taXQpIHsKICAgICAgICAgICAgICAgIGlmICghcm1zLmlzRW1wdHkoKSkgewogICAgICAgICAgICAgICAgICAgIHJlc3VsdCA9IFhBUmVzb3VyY2UuWEFfT0s7CiAgICAgICAgICAgICAgICB9CiAgICAgICAgICAgIH0gZWxzZSB7CiAgICAgICAgICAgICAgICByb2xsYmFja1Jlc291cmNlcyhybXMpOwogICAgICAgICAgICAgICAgdGhyb3cgbmV3IFJvbGxiYWNrRXhjZXB0aW9uKCJVbmFibGUgdG8gY29tbWl0Iik7CiAgICAgICAgICAgIH0KICAgICAgICB9IGZpbmFsbHkgewogICAgICAgICAgICBpZiAocmVzdWx0ID09IFhBUmVzb3VyY2UuWEFfUkRPTkxZKSB7CiAgICAgICAgICAgICAgICBhZnRlckNvbXBsZXRpb24oKTsKICAgICAgICAgICAgICAgIHN5bmNocm9uaXplZCAodGhpcykgewogICAgICAgICAgICAgICAgICAgIHN0YXR1cyA9IFN0YXR1cy5TVEFUVVNfTk9fVFJBTlNBQ1RJT047CiAgICAgICAgICAgICAgICB9CiAgICAgICAgICAgIH0KICAgICAgICB9CiAgICAgICAgcmV0dXJuIHJlc3VsdDsKICAgIH0KCiAgICAvL3VzZWQgZnJvbSBYQVRlcm1pbmF0b3IgZm9yIGNvbW1pdCBwaGFzZSBvZiBub24tcmVhZG9ubHkgcmVtb3RlbHkgY29udHJvbGxlZCB0eC4KICAgIHZvaWQgcHJlcGFyZWRDb21taXQoKSB0aHJvd3MgSGV1cmlzdGljUm9sbGJhY2tFeGNlcHRpb24sIEhldXJpc3RpY01peGVkRXhjZXB0aW9uLCBTeXN0ZW1FeGNlcHRpb24gewogICAgICAgIHRyeSB7CiAgICAgICAgICAgIGNvbW1pdFJlc291cmNlcyhyZXNvdXJjZU1hbmFnZXJzKTsKICAgICAgICB9IGZpbmFsbHkgewogICAgICAgICAgICBhZnRlckNvbXBsZXRpb24oKTsKICAgICAgICAgICAgc3luY2hyb25pemVkICh0aGlzKSB7CiAgICAgICAgICAgICAgICBzdGF0dXMgPSBTdGF0dXMuU1RBVFVTX05PX1RSQU5TQUNUSU9OOwogICAgICAgICAgICB9CiAgICAgICAgfQogICAgfQoKICAgIC8vaGVscGVyIG1ldGhvZCB1c2VkIGJ5IFRyYW5zYWN0aW9uLmNvbW1pdCBhbmQgWEFUZXJtaW5hdG9yIHByZXBhcmUuCiAgICBwcml2YXRlIHZvaWQgYmVmb3JlUHJlcGFyZSgpIHsKICAgICAgICBzeW5jaHJvbml6ZWQgKHRoaXMpIHsKICAgICAgICAgICAgc3dpdGNoIChzdGF0dXMpIHsKICAgICAgICAgICAgICAgIGNhc2UgU3RhdHVzLlNUQVRVU19BQ1RJVkU6CiAgICAgICAgICAgICAgICBjYXNlIFN0YXR1cy5TVEFUVVNfTUFSS0VEX1JPTExCQUNLOgogICAgICAgICAgICAgICAgICAgIGJyZWFrOwogICAgICAgICAgICAgICAgZGVmYXVsdDoKICAgICAgICAgICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbFN0YXRlRXhjZXB0aW9uKCJTdGF0dXMgaXMgIiArIGdldFN0YXRlU3RyaW5nKHN0YXR1cykpOwogICAgICAgICAgICB9CiAgICAgICAgfQoKICAgICAgICBiZWZvcmVDb21wbGV0aW9uKCk7CiAgICAgICAgZW5kUmVzb3VyY2VzKCk7CiAgICB9CgoKICAgIC8vaGVscGVyIG1ldGhvZCB1c2VkIGJ5IFRyYW5zYWN0aW9uLmNvbW1pdCBhbmQgWEFUZXJtaW5hdG9yIHByZXBhcmUuCiAgICBwcml2YXRlIGJvb2xlYW4gaW50ZXJuYWxQcmVwYXJlKCkgdGhyb3dzIFN5c3RlbUV4Y2VwdGlvbiB7CiAgICAgICAgZm9yIChJdGVyYXRvciBybXMgPSByZXNvdXJjZU1hbmFnZXJzLml0ZXJhdG9yKCk7IHJtcy5oYXNOZXh0KCk7KSB7CiAgICAgICAgICAgIHN5bmNocm9uaXplZCAodGhpcykgewogICAgICAgICAgICAgICAgaWYgKHN0YXR1cyAhPSBTdGF0dXMuU1RBVFVTX1BSRVBBUklORykgewogICAgICAgICAgICAgICAgICAgIC8vIHdlIHdlcmUgbWFya2VkIGZvciByb2xsYmFjawogICAgICAgICAgICAgICAgICAgIGJyZWFrOwogICAgICAgICAgICAgICAgfQogICAgICAgICAgICB9CiAgICAgICAgICAgIFRyYW5zYWN0aW9uQnJhbmNoIG1hbmFnZXIgPSAoVHJhbnNhY3Rpb25CcmFuY2gpIHJtcy5uZXh0KCk7CiAgICAgICAgICAgIHRyeSB7CiAgICAgICAgICAgICAgICBpbnQgdm90ZSA9IG1hbmFnZXIuZ2V0Q29tbWl0dGVyKCkucHJlcGFyZShtYW5hZ2VyLmdldEJyYW5jaElkKCkpOwogICAgICAgICAgICAgICAgaWYgKHZvdGUgPT0gWEFSZXNvdXJjZS5YQV9SRE9OTFkpIHsKICAgICAgICAgICAgICAgICAgICAvLyB3ZSBkb24ndCBuZWVkIHRvIGNvbnNpZGVyIHRoaXMgUk0gYW55IG1vcmUKICAgICAgICAgICAgICAgICAgICBybXMucmVtb3ZlKCk7CiAgICAgICAgICAgICAgICB9CiAgICAgICAgICAgIH0gY2F0Y2ggKFhBRXhjZXB0aW9uIGUpIHsKICAgICAgICAgICAgICAgIGlmIChlLmVycm9yQ29kZSA9PSBYQUV4Y2VwdGlvbi5YQUVSX1JNRVJSIAogICAgICAgICAgICAgICAgICAgICAgICB8fCBlLmVycm9yQ29kZSA9PSBYQUV4Y2VwdGlvbi5YQUVSX1BST1RPCiAgICAgICAgICAgICAgICAgICAgICAgIHx8IGUuZXJyb3JDb2RlID09IFhBRXhjZXB0aW9uLlhBRVJfSU5WQUwpIHsKICAgICAgICAgICAgICAgICAgICB0aHJvdyAoU3lzdGVtRXhjZXB0aW9uKSBuZXcgU3lzdGVtRXhjZXB0aW9uKCJFcnJvciBkdXJpbmcgcHJlcGFyZTsgdHJhbnNhY3Rpb24gd2FzIHJvbGxlZCBiYWNrIikuaW5pdENhdXNlKGUpOwogICAgICAgICAgICAgICAgfQogICAgICAgICAgICAgICAgc3luY2hyb25pemVkICh0aGlzKSB7CiAgICAgICAgICAgICAgICAgICAgc3RhdHVzID0gU3RhdHVzLlNUQVRVU19NQVJLRURfUk9MTEJBQ0s7CiAgICAgICAgICAgICAgICAgICAgLyogUGVyIEpUQSBzcGVjLCAgSWYgdGhlIHJlc291cmNlIG1hbmFnZXIgd2FudHMgdG8gcm9sbCBiYWNrIHRoZSB0cmFuc2FjdGlvbiwgCiAgICAgICAgICAgICAgICAgICAgaXQgc2hvdWxkIGRvIHNvIGJ5IHRocm93aW5nIGFuIGFwcHJvcHJpYXRlIFhBRXhjZXB0aW9uIGluIHRoZSBwcmVwYXJlIG1ldGhvZC4KICAgICAgICAgICAgICAgICAgICBBbHNvIHBlciBPVFMgc3BlYzoKICAgICAgICAgICAgICAgICAgICBUaGUgcmVzb3VyY2UgY2FuIHJldHVybiBWb3RlUm9sbGJhY2sgdW5kZXIgYW55IGNpcmN1bXN0YW5jZXMsIGluY2x1ZGluZyBub3QgaGF2aW5nIAogICAgICAgICAgICAgICAgICAgIGFueSBrbm93bGVkZ2UgYWJvdXQgdGhlIHRyYW5zYWN0aW9uICh3aGljaCBtaWdodCBoYXBwZW4gYWZ0ZXIgYSBjcmFzaCkuIElmIHRoaXMgCiAgICAgICAgICAgICAgICAgICAgcmVzcG9uc2UgaXMgcmV0dXJuZWQsIHRoZSB0cmFuc2FjdGlvbiBtdXN0IGJlIHJvbGxlZCBiYWNrLiBGdXJ0aGVybW9yZSwgdGhlIFRyYW5zYWN0aW9uIAogICAgICAgICAgICAgICAgICAgIFNlcnZpY2UgaXMgbm90IHJlcXVpcmVkIHRvIHBlcmZvcm0gYW55IGFkZGl0aW9uYWwgb3BlcmF0aW9ucyBvbiB0aGlzIHJlc291cmNlLiovCiAgICAgICAgICAgICAgICAgICAgLy9ybXMucmVtb3ZlKCk7CiAgICAgICAgICAgICAgICAgICAgYnJlYWs7CiAgICAgICAgICAgICAgICB9CiAgICAgICAgICAgIH0KICAgICAgICB9CgogICAgICAgIC8vIGRlY2lzaW9uIHRpbWUuLi4KICAgICAgICBib29sZWFuIHdpbGxDb21taXQ7CiAgICAgICAgc3luY2hyb25pemVkICh0aGlzKSB7CiAgICAgICAgICAgIHdpbGxDb21taXQgPSAoc3RhdHVzICE9IFN0YXR1cy5TVEFUVVNfTUFSS0VEX1JPTExCQUNLKTsKICAgICAgICAgICAgaWYgKHdpbGxDb21taXQpIHsKICAgICAgICAgICAgICAgIHN0YXR1cyA9IFN0YXR1cy5TVEFUVVNfUFJFUEFSRUQ7CiAgICAgICAgICAgIH0KICAgICAgICB9CiAgICAgICAgLy8gbG9nIG91ciBkZWNpc2lvbgogICAgICAgIGlmICh3aWxsQ29tbWl0ICYmICFyZXNvdXJjZU1hbmFnZXJzLmlzRW1wdHkoKSkgewogICAgICAgICAgICB0cnkgewogICAgICAgICAgICAgICAgbG9nTWFyayA9IHR4bkxvZy5wcmVwYXJlKHhpZCwgcmVzb3VyY2VNYW5hZ2Vycyk7CiAgICAgICAgICAgIH0gY2F0Y2ggKExvZ0V4Y2VwdGlvbiBlKSB7CiAgICAgICAgICAgICAgICB0cnkgewogICAgICAgICAgICAgICAgICAgIHJvbGxiYWNrUmVzb3VyY2VzKHJlc291cmNlTWFuYWdlcnMpOwogICAgICAgICAgICAgICAgfSBjYXRjaCAoRXhjZXB0aW9uIHNlKSB7CiAgICAgICAgICAgICAgICAgICAgbG9nLmVycm9yKCJVbmFibGUgdG8gcm9sbGJhY2sgYWZ0ZXIgZmFpbHVyZSB0byBsb2cgcHJlcGFyZSIsIHNlLmdldENhdXNlKCkpOwogICAgICAgICAgICAgICAgfQogICAgICAgICAgICAgICAgdGhyb3cgKFN5c3RlbUV4Y2VwdGlvbikgbmV3IFN5c3RlbUV4Y2VwdGlvbigiRXJyb3IgbG9nZ2luZyBwcmVwYXJlOyB0cmFuc2FjdGlvbiB3YXMgcm9sbGVkIGJhY2spIikuaW5pdENhdXNlKGUpOwogICAgICAgICAgICB9CiAgICAgICAgfQogICAgICAgIHJldHVybiB3aWxsQ29tbWl0OwogICAgfQoKICAgIHB1YmxpYyB2b2lkIHJvbGxiYWNrKCkgdGhyb3dzIElsbGVnYWxTdGF0ZUV4Y2VwdGlvbiwgU3lzdGVtRXhjZXB0aW9uIHsKICAgICAgICBMaXN0IHJtczsKICAgICAgICBzeW5jaHJvbml6ZWQgKHRoaXMpIHsKICAgICAgICAgICAgc3dpdGNoIChzdGF0dXMpIHsKICAgICAgICAgICAgICAgIGNhc2UgU3RhdHVzLlNUQVRVU19BQ1RJVkU6CiAgICAgICAgICAgICAgICAgICAgc3RhdHVzID0gU3RhdHVzLlNUQVRVU19NQVJLRURfUk9MTEJBQ0s7CiAgICAgICAgICAgICAgICAgICAgYnJlYWs7CiAgICAgICAgICAgICAgICBjYXNlIFN0YXR1cy5TVEFUVVNfTUFSS0VEX1JPTExCQUNLOgogICAgICAgICAgICAgICAgICAgIGJyZWFrOwogICAgICAgICAgICAgICAgZGVmYXVsdDoKICAgICAgICAgICAgICAgICAgICB0aHJvdyBuZXcgSWxsZWdhbFN0YXRlRXhjZXB0aW9uKCJTdGF0dXMgaXMgIiArIGdldFN0YXRlU3RyaW5nKHN0YXR1cykpOwogICAgICAgICAgICB9CiAgICAgICAgfQoKICAgICAgICBlbmRSZXNvdXJjZXMoKTsKICAgICAgICBybXMgPSByZXNvdXJjZU1hbmFnZXJzOwogICAgICAgIHRyeSB7CiAgICAgICAgICAgIHJvbGxiYWNrUmVzb3VyY2VzKHJtcyk7CiAgICAgICAgICAgIC8vb25seSB3cml0ZSByb2xsYmFjayByZWNvcmQgaWYgd2UgaGF2ZSBhbHJlYWR5IHdyaXR0ZW4gcHJlcGFyZSByZWNvcmQuCiAgICAgICAgICAgIGlmIChsb2dNYXJrICE9IG51bGwpIHsKICAgICAgICAgICAgICAgIHRyeSB7CiAgICAgICAgICAgICAgICAgICAgdHhuTG9nLnJvbGxiYWNrKHhpZCwgbG9nTWFyayk7CiAgICAgICAgICAgICAgICB9IGNhdGNoIChMb2dFeGNlcHRpb24gZSkgewogICAgICAgICAgICAgICAgICAgIHRyeSB7CiAgICAgICAgICAgICAgICAgICAgICAgIHJvbGxiYWNrUmVzb3VyY2VzKHJtcyk7CiAgICAgICAgICAgICAgICAgICAgfSBjYXRjaCAoRXhjZXB0aW9uIHNlKSB7CiAgICAgICAgICAgICAgICAgICAgICAgIGxvZy5lcnJvcigiVW5hYmxlIHRvIHJvbGxiYWNrIGFmdGVyIGZhaWx1cmUgdG8gbG9nIGRlY2lzaW9uIiwgc2UuZ2V0Q2F1c2UoKSk7CiAgICAgICAgICAgICAgICAgICAgfQogICAgICAgICAgICAgICAgICAgIHRocm93IChTeXN0ZW1FeGNlcHRpb24pIG5ldyBTeXN0ZW1FeGNlcHRpb24oIkVycm9yIGxvZ2dpbmcgcm9sbGJhY2siKS5pbml0Q2F1c2UoZSk7CiAgICAgICAgICAgICAgICB9CiAgICAgICAgICAgIH0KICAgICAgICB9IGZpbmFsbHkgewogICAgICAgICAgICBhZnRlckNvbXBsZXRpb24oKTsKICAgICAgICAgICAgc3luY2hyb25pemVkICh0aGlzKSB7CiAgICAgICAgICAgICAgICBzdGF0dXMgPSBTdGF0dXMuU1RBVFVTX05PX1RSQU5TQUNUSU9OOwogICAgICAgICAgICB9CiAgICAgICAgfQogICAgfQoKICAgIHByaXZhdGUgdm9pZCBiZWZvcmVDb21wbGV0aW9uKCkgewogICAgICAgIGJlZm9yZUNvbXBsZXRpb24oc3luY0xpc3QpOwogICAgICAgIGJlZm9yZUNvbXBsZXRpb24oaW50ZXJwb3NlZFN5bmNMaXN0KTsKICAgIH0KCiAgICBwcml2YXRlIHZvaWQgYmVmb3JlQ29tcGxldGlvbihMaXN0IHN5bmNzKSB7CiAgICAgICAgaW50IGkgPSAwOwogICAgICAgIHdoaWxlICh0cnVlKSB7CiAgICAgICAgICAgIFN5bmNocm9uaXphdGlvbiBzeW5jaDsKICAgICAgICAgICAgc3luY2hyb25pemVkICh0aGlzKSB7CiAgICAgICAgICAgICAgICBpZiAoaSA9PSBzeW5jcy5zaXplKCkpIHsKICAgICAgICAgICAgICAgICAgICByZXR1cm47CiAgICAgICAgICAgICAgICB9IGVsc2UgewogICAgICAgICAgICAgICAgICAgIHN5bmNoID0gKFN5bmNocm9uaXphdGlvbikgc3luY3MuZ2V0KGkrKyk7CiAgICAgICAgICAgICAgICB9CiAgICAgICAgICAgIH0KICAgICAgICAgICAgdHJ5IHsKICAgICAgICAgICAgICAgIHN5bmNoLmJlZm9yZUNvbXBsZXRpb24oKTsKICAgICAgICAgICAgfSBjYXRjaCAoRXhjZXB0aW9uIGUpIHsKICAgICAgICAgICAgICAgIGxvZy53YXJuKCJVbmV4cGVjdGVkIGV4Y2VwdGlvbiBmcm9tIGJlZm9yZUNvbXBsZXRpb247IHRyYW5zYWN0aW9uIHdpbGwgcm9sbCBiYWNrIiwgZSk7CiAgICAgICAgICAgICAgICBzeW5jaHJvbml6ZWQgKHRoaXMpIHsKICAgICAgICAgICAgICAgICAgICBzdGF0dXMgPSBTdGF0dXMuU1RBVFVTX01BUktFRF9ST0xMQkFDSzsKICAgICAgICAgICAgICAgIH0KICAgICAgICAgICAgfQogICAgICAgIH0KICAgIH0KCiAgICBwcml2YXRlIHZvaWQgYWZ0ZXJDb21wbGV0aW9uKCkgewogICAgICAgIC8vIHRoaXMgZG9lcyBub3Qgc3luY2hyb25pemUgYmVjYXVzZSBub3RoaW5nIGNhbiBtb2RpZnkgb3VyIHN0YXRlIGF0IHRoaXMgdGltZQogICAgICAgIGFmdGVyQ29tcGxldGlvbihpbnRlcnBvc2VkU3luY0xpc3QpOwogICAgICAgIGFmdGVyQ29tcGxldGlvbihzeW5jTGlzdCk7CiAgICB9CgogICAgcHJpdmF0ZSB2b2lkIGFmdGVyQ29tcGxldGlvbihMaXN0IHN5bmNzKSB7CiAgICAgICAgZm9yIChJdGVyYXRvciBpID0gc3luY3MuaXRlcmF0b3IoKTsgaS5oYXNOZXh0KCk7KSB7CiAgICAgICAgICAgIFN5bmNocm9uaXphdGlvbiBzeW5jaCA9IChTeW5jaHJvbml6YXRpb24pIGkubmV4dCgpOwogICAgICAgICAgICB0cnkgewogICAgICAgICAgICAgICAgc3luY2guYWZ0ZXJDb21wbGV0aW9uKHN0YXR1cyk7CiAgICAgICAgICAgIH0gY2F0Y2ggKEV4Y2VwdGlvbiBlKSB7CiAgICAgICAgICAgICAgICBsb2cud2FybigiVW5leHBlY3RlZCBleGNlcHRpb24gZnJvbSBhZnRlckNvbXBsZXRpb247IGNvbnRpbnVpbmciLCBlKTsKICAgICAgICAgICAgfQogICAgICAgIH0KICAgIH0KCiAgICBwcml2YXRlIHZvaWQgZW5kUmVzb3VyY2VzKCkgewogICAgICAgIGVuZFJlc291cmNlcyhhY3RpdmVYYVJlc291cmNlcyk7CiAgICAgICAgZW5kUmVzb3VyY2VzKHN1c3BlbmRlZFhhUmVzb3VyY2VzKTsKICAgIH0KCiAgICBwcml2YXRlIHZvaWQgZW5kUmVzb3VyY2VzKElkZW50aXR5SGFzaE1hcCByZXNvdXJjZU1hcCkgewogICAgICAgIHdoaWxlICh0cnVlKSB7CiAgICAgICAgICAgIFhBUmVzb3VyY2UgeGFSZXM7CiAgICAgICAgICAgIFRyYW5zYWN0aW9uQnJhbmNoIG1hbmFnZXI7CiAgICAgICAgICAgIGludCBmbGFnczsKICAgICAgICAgICAgc3luY2hyb25pemVkICh0aGlzKSB7CiAgICAgICAgICAgICAgICBTZXQgZW50cnlTZXQgPSByZXNvdXJjZU1hcC5lbnRyeVNldCgpOwogICAgICAgICAgICAgICAgaWYgKGVudHJ5U2V0LmlzRW1wdHkoKSkgewogICAgICAgICAgICAgICAgICAgIHJldHVybjsKICAgICAgICAgICAgICAgIH0KICAgICAgICAgICAgICAgIE1hcC5FbnRyeSBlbnRyeSA9IChNYXAuRW50cnkpIGVudHJ5U2V0Lml0ZXJhdG9yKCkubmV4dCgpOwogICAgICAgICAgICAgICAgeGFSZXMgPSAoWEFSZXNvdXJjZSkgZW50cnkuZ2V0S2V5KCk7CiAgICAgICAgICAgICAgICBtYW5hZ2VyID0gKFRyYW5zYWN0aW9uQnJhbmNoKSBlbnRyeS5nZXRWYWx1ZSgpOwogICAgICAgICAgICAgICAgZmxhZ3MgPSAoc3RhdHVzID09IFN0YXR1cy5TVEFUVVNfTUFSS0VEX1JPTExCQUNLKSA/IFhBUmVzb3VyY2UuVE1GQUlMIDogWEFSZXNvdXJjZS5UTVNVQ0NFU1M7CiAgICAgICAgICAgICAgICByZXNvdXJjZU1hcC5yZW1vdmUoeGFSZXMpOwogICAgICAgICAgICB9CiAgICAgICAgICAgIHRyeSB7CiAgICAgICAgICAgICAgICB4YVJlcy5lbmQobWFuYWdlci5nZXRCcmFuY2hJZCgpLCBmbGFncyk7CiAgICAgICAgICAgIH0gY2F0Y2ggKFhBRXhjZXB0aW9uIGUpIHsKICAgICAgICAgICAgICAgIGxvZy53YXJuKCJFcnJvciBlbmRpbmcgYXNzb2NpYXRpb24gZm9yIFhBUmVzb3VyY2UgIiArIHhhUmVzICsgIjsgdHJhbnNhY3Rpb24gd2lsbCByb2xsIGJhY2suIFhBIGVycm9yIGNvZGU6ICIgKyBlLmVycm9yQ29kZSwgZSk7CiAgICAgICAgICAgICAgICBzeW5jaHJvbml6ZWQgKHRoaXMpIHsKICAgICAgICAgICAgICAgICAgICBzdGF0dXMgPSBTdGF0dXMuU1RBVFVTX01BUktFRF9ST0xMQkFDSzsKICAgICAgICAgICAgICAgICAgICAvLyB3aGVuIFhBRXhjZXB0aW9uIGlzIFhBX1JCUk9MTEJBQ0ssIHdoaWNoIGluZGljYXRlcyB0aGF0IFhBUmVzb3VyY2UgaGFzIGFscmVhZHkgcm9sbGVkIGJhY2sgdGhlIHRyYW5zYWN0aW9uLCBubyBuZWVkCiAgICAgICAgICAgICAgICAgICAgLy8gdG8gc2VuZCByb2xsYmFjayByZXF1ZXN0IHRvIHRoZSBYQVJlc291cmNlLCB0aHVzIHJlbW92ZSBpdCBvZmYgcmVzb3VyY2VNYW5hZ2VycyBsaXN0LgogICAgICAgICAgICAgICAgICAgIGlmIChlLmVycm9yQ29kZSA9PSBYQUV4Y2VwdGlvbi5YQV9SQlJPTExCQUNLKSB7CiAgICAgICAgICAgICAgICAgICAgICAgIHJlc291cmNlTWFuYWdlcnMucmVtb3ZlKG1hbmFnZXIpOwogICAgICAgICAgICAgICAgICAgIH0KICAgICAgICAgICAgICAgIH0KICAgICAgICAgICAgfQogICAgICAgIH0KICAgIH0KCiAgICBwcml2YXRlIHZvaWQgcm9sbGJhY2tSZXNvdXJjZXMoTGlzdCBybXMpIHRocm93cyBTeXN0ZW1FeGNlcHRpb24gewogICAgICAgIFN5c3RlbUV4Y2VwdGlvbiBjYXVzZSA9IG51bGw7CiAgICAgICAgc3luY2hyb25pemVkICh0aGlzKSB7CiAgICAgICAgICAgIHN0YXR1cyA9IFN0YXR1cy5TVEFUVVNfUk9MTElOR19CQUNLOwogICAgICAgIH0KICAgICAgICB0cnkgewogICAgICAgICAgICBmb3IgKEl0ZXJhdG9yIGkgPSBybXMuaXRlcmF0b3IoKTsgaS5oYXNOZXh0KCk7KSB7CiAgICAgICAgICAgICAgICBUcmFuc2FjdGlvbkJyYW5jaCBtYW5hZ2VyID0gKFRyYW5zYWN0aW9uQnJhbmNoKSBpLm5leHQoKTsKICAgICAgICAgICAgICAgIHRyeSB7CiAgICAgICAgICAgICAgICAgICAgbWFuYWdlci5nZXRDb21taXR0ZXIoKS5yb2xsYmFjayhtYW5hZ2VyLmdldEJyYW5jaElkKCkpOwogICAgICAgICAgICAgICAgfSBjYXRjaCAoWEFFeGNlcHRpb24gZSkgewogICAgICAgICAgICAgICAgICAgIGxvZy5lcnJvcigiVW5leHBlY3RlZCBleGNlcHRpb24gcm9sbGluZyBiYWNrICIgKyBtYW5hZ2VyLmdldENvbW1pdHRlcigpICsgIjsgY29udGludWluZyB3aXRoIHJvbGxiYWNrIiwgZSk7CiAgICAgICAgICAgICAgICAgICAgaWYgKGUuZXJyb3JDb2RlID09IFhBRXhjZXB0aW9uLlhBX0hFVVJSQikgewogICAgICAgICAgICAgICAgICAgICAgICAvLyBsZXQncyBub3Qgc2V0IHRoZSBjYXVzZSBoZXJlCiAgICAgICAgICAgICAgICAgICAgICAgIGxvZy5pbmZvKCJUcmFuc2FjdGlvbiBoYXMgYmVlbiBoZXVyaXN0aWNhbGx5IHJvbGxlZCBiYWNrICIgKyBtYW5hZ2VyLmdldENvbW1pdHRlcigpICsgIjsgY29udGludWluZyB3aXRoIHJvbGxiYWNrIiwgZSk7CiAgICAgICAgICAgICAgICAgICAgICAgIG1hbmFnZXIuZ2V0Q29tbWl0dGVyKCkuZm9yZ2V0KG1hbmFnZXIuZ2V0QnJhbmNoSWQoKSk7CiAgICAgICAgICAgICAgICAgICAgfSBlbHNlIGlmIChlLmVycm9yQ29kZSA9PSBYQUV4Y2VwdGlvbi5YQV9SQlJPTExCQUNLIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgfHwgZS5lcnJvckNvZGUgPT0gWEFFeGNlcHRpb24uWEFFUl9STUVSUiAKICAgICAgICAgICAgICAgICAgICAgICAgICAgIHx8IGUuZXJyb3JDb2RlID09IFhBRXhjZXB0aW9uLlhBRVJfTk9UQQogICAgICAgICAgICAgICAgICAgICAgICAgICAgfHwgZS5lcnJvckNvZGUgPT0gWEFFeGNlcHRpb24uWEFFUl9STUZBSUwpIHsKICAgICAgICAgICAgICAgICAgICAgICAgLy8gbGV0J3Mgbm90IHNldCB0aGUgY2F1c2UgaGVyZSBiZWNhdXNlIHdlIGV4cGVjdCB0aGUgdHJhbnNhY3Rpb24gdG8gYmUgcm9sbGVkIGJhY2sgZXZlbnR1YWxseQogICAgICAgICAgICAgICAgICAgICAgICAvLyBUT0RPOiBmb3IgUk1GQUlMLCBpdCBtZWFucyByZXNvdXJjZSB1bmF2YWlsYWJsZSB0ZW1wb3JhcmlseS4gIAogICAgICAgICAgICAgICAgICAgICAgICAvLyBkbyB3ZSBuZWVkIGtlZXAgc2VuZGluZyByZXF1ZXN0IHRvIHJlc291cmNlIHRvIG1ha2Ugc3VyZSB0aGUgcm9sbCBiYWNrIGNvbXBsZXRlcz8KICAgICAgICAgICAgICAgICAgICB9IGVsc2UgaWYgKGNhdXNlID09IG51bGwpIHsKICAgICAgICAgICAgICAgICAgICAgICAgY2F1c2UgPSBuZXcgU3lzdGVtRXhjZXB0aW9uKGUuZXJyb3JDb2RlKTsKICAgICAgICAgICAgICAgICAgICB9CiAgICAgICAgICAgICAgICB9CiAgICAgICAgICAgIH0KICAgICAgICB9IGNhdGNoIChYQUV4Y2VwdGlvbiBlKSB7CiAgICAgICAgICAgIHRocm93IChTeXN0ZW1FeGNlcHRpb24pIG5ldyBTeXN0ZW1FeGNlcHRpb24oIkVycm9yIGR1cmluZyByb2xsaW5nIGJhY2siKS5pbml0Q2F1c2UoZSk7ICAgICAgICAgICAgCiAgICAgICAgfSAgICAgICAgICAgIAoKICAgICAgICBzeW5jaHJvbml6ZWQgKHRoaXMpIHsKICAgICAgICAgICAgc3RhdHVzID0gU3RhdHVzLlNUQVRVU19ST0xMRURCQUNLOwogICAgICAgIH0KICAgICAgICBpZiAoY2F1c2UgIT0gbnVsbCkgewogICAgICAgICAgICB0aHJvdyBjYXVzZTsKICAgICAgICB9CiAgICB9CgogICAgcHJpdmF0ZSB2b2lkIHJvbGxiYWNrUmVzb3VyY2VzRHVyaW5nQ29tbWl0KExpc3Qgcm1zLCBib29sZWFuIGV2ZXJSYikgdGhyb3dzIEhldXJpc3RpY01peGVkRXhjZXB0aW9uLCBSb2xsYmFja0V4Y2VwdGlvbiwgU3lzdGVtRXhjZXB0aW9uIHsKICAgICAgICBYQUV4Y2VwdGlvbiBjYXVzZSA9IG51bGw7CiAgICAgICAgYm9vbGVhbiBldmVyUm9sbGVkYmFjayA9IGV2ZXJSYjsKICAgICAgICBzeW5jaHJvbml6ZWQgKHRoaXMpIHsKICAgICAgICAgICAgc3RhdHVzID0gU3RhdHVzLlNUQVRVU19ST0xMSU5HX0JBQ0s7CiAgICAgICAgfQogICAgICAgIHRyeSB7CiAgICAgICAgICAgIGZvciAoSXRlcmF0b3IgaSA9IHJtcy5pdGVyYXRvcigpOyBpLmhhc05leHQoKTspIHsKICAgICAgICAgICAgICAgIFRyYW5zYWN0aW9uQnJhbmNoIG1hbmFnZXIgPSAoVHJhbnNhY3Rpb25CcmFuY2gpIGkubmV4dCgpOwogICAgICAgICAgICAgICAgdHJ5IHsKICAgICAgICAgICAgICAgICAgICBtYW5hZ2VyLmdldENvbW1pdHRlcigpLnJvbGxiYWNrKG1hbmFnZXIuZ2V0QnJhbmNoSWQoKSk7CiAgICAgICAgICAgICAgICAgICAgZXZlclJvbGxlZGJhY2sgPSB0cnVlOwogICAgICAgICAgICAgICAgfSBjYXRjaCAoWEFFeGNlcHRpb24gZSkgewogICAgICAgICAgICAgICAgICAgIGlmIChlLmVycm9yQ29kZSA9PSBYQUV4Y2VwdGlvbi5YQV9IRVVSUkIpIHsKICAgICAgICAgICAgICAgICAgICAgICAgLy8gbGV0J3Mgbm90IHNldCB0aGUgY2F1c2UgaGVyZSBhcyB0aGUgcmVzdWx0aW5nIGJlaGF2aW9yIGlzIHNhbWUgYXMgcmVxdWVzdGVkIGJlaGF2aW9yCiAgICAgICAgICAgICAgICAgICAgICAgIGxvZy5lcnJvcigiVHJhbnNhY3Rpb24gaGFzIGJlZW4gaGV1cmlzdGljYWxseSByb2xsZWQgYmFjayAiICsgbWFuYWdlci5nZXRDb21taXR0ZXIoKSArICI7IGNvbnRpbnVpbmcgd2l0aCByb2xsYmFjayIsIGUpOwogICAgICAgICAgICAgICAgICAgICAgICBldmVyUm9sbGVkYmFjayA9IHRydWU7CiAgICAgICAgICAgICAgICAgICAgICAgIG1hbmFnZXIuZ2V0Q29tbWl0dGVyKCkuZm9yZ2V0KG1hbmFnZXIuZ2V0QnJhbmNoSWQoKSk7CiAgICAgICAgICAgICAgICAgICAgfSBlbHNlIGlmIChlLmVycm9yQ29kZSA9PSBYQUV4Y2VwdGlvbi5YQV9IRVVSTUlYKSB7CiAgICAgICAgICAgICAgICAgICAgICAgIGxvZy5lcnJvcigiVHJhbnNhY3Rpb24gaGFzIGJlZW4gaGV1cmlzdGljYWxseSBjb21taXR0ZWQgYW5kIHJvbGxlZCBiYWNrICIgKyBtYW5hZ2VyLmdldENvbW1pdHRlcigpICsgIjsgY29udGludWluZyB3aXRoIHJvbGxiYWNrIiwgZSk7CiAgICAgICAgICAgICAgICAgICAgICAgIGNhdXNlID0gZTsKICAgICAgICAgICAgICAgICAgICAgICAgZXZlclJvbGxlZGJhY2sgPSB0cnVlOwogICAgICAgICAgICAgICAgICAgICAgICBtYW5hZ2VyLmdldENvbW1pdHRlcigpLmZvcmdldChtYW5hZ2VyLmdldEJyYW5jaElkKCkpOwogICAgICAgICAgICAgICAgICAgIH0gZWxzZSBpZiAoZS5lcnJvckNvZGUgPT0gWEFFeGNlcHRpb24uWEFfSEVVUkNPTSkgewogICAgICAgICAgICAgICAgICAgICAgICBsb2cuZXJyb3IoIlRyYW5zYWN0aW9uIGhhcyBiZWVuIGhldXJpc3RpY2FsbHkgY29tbWl0dGVkICIgKyBtYW5hZ2VyLmdldENvbW1pdHRlcigpICsgIjsgY29udGludWluZyB3aXRoIHJvbGxiYWNrIiwgZSk7CiAgICAgICAgICAgICAgICAgICAgICAgIGNhdXNlID0gZTsKICAgICAgICAgICAgICAgICAgICAgICAgbWFuYWdlci5nZXRDb21taXR0ZXIoKS5mb3JnZXQobWFuYWdlci5nZXRCcmFuY2hJZCgpKTsKICAgICAgICAgICAgICAgICAgICB9IGVsc2UgaWYgKGUuZXJyb3JDb2RlID09IFhBRXhjZXB0aW9uLlhBX1JCUk9MTEJBQ0sgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICB8fCBlLmVycm9yQ29kZSA9PSBYQUV4Y2VwdGlvbi5YQUVSX1JNRVJSCiAgICAgICAgICAgICAgICAgICAgICAgICAgICB8fCBlLmVycm9yQ29kZSA9PSBYQUV4Y2VwdGlvbi5YQUVSX05PVEEKICAgICAgICAgICAgICAgICAgICAgICAgICAgIHx8IGUuZXJyb3JDb2RlID09IFhBRXhjZXB0aW9uLlhBRVJfUk1GQUlMKSB7CiAgICAgICAgICAgICAgICAgICAgICAgIC8vIFhBRXhjZXB0aW9uLlhBX1JCUk9MTEJBQ0sgZHVyaW5nIGNvbW1pdC9yb2xsYmFjaywgdGh1cyBSb2xsYmFja0V4Y2VwdGlvbiBpcyBleHBlY3RlZAogICAgICAgICAgICAgICAgICAgICAgICAvLyBYQUV4Y2VwdGlvbi5YQUVSX1JNRVJSIG1lYW5zIHRyYW5zYWN0aW9uIGJyYW5jaCBlcnJvciBhbmQgdHJhbnNhY3Rpb24gaGFzIGJlZW4gcm9sbGVkIGJhY2sKICAgICAgICAgICAgICAgICAgICAgICAgLy8gbGV0J3Mgbm90IHNldCB0aGUgY2F1c2UgaGVyZSBiZWNhdXNlIHdlIGV4cGVjdCB0aGUgdHJhbnNhY3Rpb24gdG8gYmUgcm9sbGVkIGJhY2sgZXZlbnR1YWxseQogICAgICAgICAgICAgICAgICAgICAgICAvLyBUT0RPOiBmb3IgUk1GQUlMLCBpdCBtZWFucyByZXNvdXJjZSB1bmF2YWlsYWJsZSB0ZW1wb3JhcmlseS4gIAogICAgICAgICAgICAgICAgICAgICAgICAvLyBkbyB3ZSBuZWVkIGtlZXAgc2VuZGluZyByZXF1ZXN0IHRvIHJlc291cmNlIHRvIG1ha2Ugc3VyZSB0aGUgcm9sbCBiYWNrIGNvbXBsZXRlcz8KICAgICAgICAgICAgICAgICAgICB9IGVsc2UgaWYgKGNhdXNlID09IG51bGwpIHsKICAgICAgICAgICAgICAgICAgICAgICAgY2F1c2UgPSBlOwogICAgICAgICAgICAgICAgICAgIH0KICAgICAgICAgICAgICAgIH0KICAgICAgICAgICAgfQogICAgICAgIH0gY2F0Y2ggKFhBRXhjZXB0aW9uIGUpIHsKICAgICAgICAgICAgdGhyb3cgKFN5c3RlbUV4Y2VwdGlvbikgbmV3IFN5c3RlbUV4Y2VwdGlvbigiU3lzdGVtIGVycm9yIGR1cmluZyBjb21taXQvcm9sbGluZyBiYWNrIikuaW5pdENhdXNlKGUpOyAgICAgICAgICAgIAogICAgICAgIH0KICAgICAgICAKICAgICAgICBzeW5jaHJvbml6ZWQgKHRoaXMpIHsKICAgICAgICAgICAgc3RhdHVzID0gU3RhdHVzLlNUQVRVU19ST0xMRURCQUNLOwogICAgICAgIH0KICAgICAgICAKICAgICAgICBpZiAoY2F1c2UgPT0gbnVsbCkgewogICAgICAgICAgICB0aHJvdyAoUm9sbGJhY2tFeGNlcHRpb24pIG5ldyBSb2xsYmFja0V4Y2VwdGlvbigiVW5hYmxlIHRvIGNvbW1pdDogdHJhbnNhY3Rpb24gbWFya2VkIGZvciByb2xsYmFjayIpLmluaXRDYXVzZShjYXVzZSk7CiAgICAgICAgfSBlbHNlIHsKICAgICAgICAgICAgaWYgKGNhdXNlLmVycm9yQ29kZSA9PSBYQUV4Y2VwdGlvbi5YQV9IRVVSQ09NICYmIGV2ZXJSb2xsZWRiYWNrKSB7CiAgICAgICAgICAgICAgICB0aHJvdyAoSGV1cmlzdGljTWl4ZWRFeGNlcHRpb24pIG5ldyBIZXVyaXN0aWNNaXhlZEV4Y2VwdGlvbigiSGV1cmlzdGljTWl4ZWQgZXJyb3IgZHVyaW5nIGNvbW1pdC9yb2xsaW5nIGJhY2siKS5pbml0Q2F1c2UoY2F1c2UpOwogICAgICAgICAgICB9IGVsc2UgaWYgKGNhdXNlLmVycm9yQ29kZSA9PSBYQUV4Y2VwdGlvbi5YQV9IRVVSTUlYKSB7CiAgICAgICAgICAgICAgICB0aHJvdyAoSGV1cmlzdGljTWl4ZWRFeGNlcHRpb24pIG5ldyBIZXVyaXN0aWNNaXhlZEV4Y2VwdGlvbigiSGV1cmlzdGljTWl4ZWQgZXJyb3IgZHVyaW5nIGNvbW1pdC9yb2xsaW5nIGJhY2siKS5pbml0Q2F1c2UoY2F1c2UpOwogICAgICAgICAgICB9IGVsc2UgeyAgICAgICAgICAgIAogICAgICAgICAgICAgICAgdGhyb3cgKFN5c3RlbUV4Y2VwdGlvbikgbmV3IFN5c3RlbUV4Y2VwdGlvbigiU3lzdGVtIEVycm9yIGR1cmluZyBjb21taXQvcm9sbGluZyBiYWNrIikuaW5pdENhdXNlKGNhdXNlKTsKICAgICAgICAgICAgfSAKICAgICAgICB9CiAgICB9CiAgICAKICAgIHByaXZhdGUgdm9pZCBjb21taXRSZXNvdXJjZShUcmFuc2FjdGlvbkJyYW5jaCBtYW5hZ2VyKSB0aHJvd3MgUm9sbGJhY2tFeGNlcHRpb24sIEhldXJpc3RpY1JvbGxiYWNrRXhjZXB0aW9uLCBIZXVyaXN0aWNNaXhlZEV4Y2VwdGlvbiwgU3lzdGVtRXhjZXB0aW9uewogICAgICAgIFhBRXhjZXB0aW9uIGNhdXNlID0gbnVsbDsKICAgICAgICB0cnkgewogICAgICAgICAgICB0cnkgewoKICAgICAgICAgICAgICAgIG1hbmFnZXIuZ2V0Q29tbWl0dGVyKCkuY29tbWl0KG1hbmFnZXIuZ2V0QnJhbmNoSWQoKSwgdHJ1ZSk7CiAgICAgICAgICAgICAgICBzeW5jaHJvbml6ZWQgKHRoaXMpIHsKICAgICAgICAgICAgICAgICAgICBzdGF0dXMgPSBTdGF0dXMuU1RBVFVTX0NPTU1JVFRFRDsKICAgICAgICAgICAgICAgIH0KICAgICAgICAgICAgICAgIHJldHVybjsKICAgICAgICAgICAgfSBjYXRjaCAoWEFFeGNlcHRpb24gZSkgewogICAgICAgICAgICAgICAgc3luY2hyb25pemVkICh0aGlzKSB7CiAgICAgICAgICAgICAgICAgICAgc3RhdHVzID0gU3RhdHVzLlNUQVRVU19ST0xMRURCQUNLOwogICAgICAgICAgICAgICAgfQogICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICBpZiAoZS5lcnJvckNvZGUgPT0gWEFFeGNlcHRpb24uWEFfSEVVUlJCKSB7CiAgICAgICAgICAgICAgICAgICAgY2F1c2UgPSBlOwogICAgICAgICAgICAgICAgICAgIG1hbmFnZXIuZ2V0Q29tbWl0dGVyKCkuZm9yZ2V0KG1hbmFnZXIuZ2V0QnJhbmNoSWQoKSk7CiAgICAgICAgICAgICAgICAgICAgLy90aHJvdyAoSGV1cmlzdGljUm9sbGJhY2tFeGNlcHRpb24pIG5ldyBIZXVyaXN0aWNSb2xsYmFja0V4Y2VwdGlvbigiRXJyb3IgZHVyaW5nIG9uZS1waGFzZSBjb21taXQiKS5pbml0Q2F1c2UoZSk7CiAgICAgICAgICAgICAgICB9IGVsc2UgaWYgKGUuZXJyb3JDb2RlID09IFhBRXhjZXB0aW9uLlhBX0hFVVJNSVgpIHsKICAgICAgICAgICAgICAgICAgICBjYXVzZSA9IGU7CiAgICAgICAgICAgICAgICAgICAgbWFuYWdlci5nZXRDb21taXR0ZXIoKS5mb3JnZXQobWFuYWdlci5nZXRCcmFuY2hJZCgpKTsKICAgICAgICAgICAgICAgICAgICB0aHJvdyAoSGV1cmlzdGljTWl4ZWRFeGNlcHRpb24pIG5ldyBIZXVyaXN0aWNNaXhlZEV4Y2VwdGlvbigiRXJyb3IgZHVyaW5nIG9uZS1waGFzZSBjb21taXQiKS5pbml0Q2F1c2UoZSk7CiAgICAgICAgICAgICAgICB9IGVsc2UgaWYgKGUuZXJyb3JDb2RlID09IFhBRXhjZXB0aW9uLlhBX0hFVVJDT00pIHsKICAgICAgICAgICAgICAgICAgICAvLyBsZXQncyBub3QgdGhyb3cgYW4gZXhjZXB0aW9uIGFzIHRoZSB0cmFuc2FjdGlvbiBoYXMgYmVlbiBjb21taXR0ZWQKICAgICAgICAgICAgICAgICAgICBsb2cuaW5mbygiVHJhbnNhY3Rpb24gaGFzIGJlZW4gaGV1cmlzdGljYWxseSBjb21taXR0ZWQiKTsKICAgICAgICAgICAgICAgICAgICBtYW5hZ2VyLmdldENvbW1pdHRlcigpLmZvcmdldChtYW5hZ2VyLmdldEJyYW5jaElkKCkpOwogICAgICAgICAgICAgICAgfSBlbHNlIGlmIChlLmVycm9yQ29kZSA9PSBYQUV4Y2VwdGlvbi5YQV9SQlJPTExCQUNLIAogICAgICAgICAgICAgICAgICAgICAgICB8fCBlLmVycm9yQ29kZSA9PSBYQUV4Y2VwdGlvbi5YQUVSX1JNRVJSIAogICAgICAgICAgICAgICAgICAgICAgICB8fCBlLmVycm9yQ29kZSA9PSBYQUV4Y2VwdGlvbi5YQUVSX05PVEEpIHsKICAgICAgICAgICAgICAgICAgICAvLyBQZXIgWEEgc3BlYywgWEFFeGNlcHRpb24uWEFFUl9STUVSUiBmcm9tIGNvbW1pdCBtZWFucyBBbiBlcnJvciBvY2N1cnJlZCBpbiAKICAgICAgICAgICAgICAgICAgICAvLyBjb21taXR0aW5nIHRoZSB3b3JrIHBlcmZvcm1lZCBvbiBiZWhhbGYgb2YgdGhlIHRyYW5zYWN0aW9uIGJyYW5jaCAKICAgICAgICAgICAgICAgICAgICAvLyBhbmQgdGhlIGJyYW5jaNVzIHdvcmsgaGFzIGJlZW4gcm9sbGVkIGJhY2suIAogICAgICAgICAgICAgICAgICAgIC8vIFhBRXhjZXB0aW9uLlhBRVJfTk9UQTogIHNzdW1lIHRoZSBEQiB0b29rIGEgdW5pbGF0ZXJhbCByb2xsYmFjayBkZWNpc2lvbiBhbmQgZm9yZ290IHRoZSB0cmFuc2FjdGlvbgogICAgICAgICAgICAgICAgICAgIGxvZy5pbmZvKCJUcmFuc2FjdGlvbiBoYXMgYmVlbiByb2xsZWQgYmFjayIpOwogICAgICAgICAgICAgICAgICAgIGNhdXNlID0gZTsKICAgICAgICAgICAgICAgICAgICAvLyB0aHJvdyAoUm9sbGJhY2tFeGNlcHRpb24pIG5ldyBSb2xsYmFja0V4Y2VwdGlvbigiRXJyb3IgZHVyaW5nIG9uZS1waGFzZSBjb21taXQiKS5pbml0Q2F1c2UoZSk7ICAgICAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICB9IGVsc2UgewogICAgICAgICAgICAgICAgICAgIGNhdXNlID0gZTsKICAgICAgICAgICAgICAgICAgICAvL3Rocm93IChTeXN0ZW1FeGNlcHRpb24pIG5ldyBTeXN0ZW1FeGNlcHRpb24oIkVycm9yIGR1cmluZyBvbmUtcGhhc2UgY29tbWl0IikuaW5pdENhdXNlKGUpOwogICAgICAgICAgICAgICAgfSAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgIH0KICAgICAgICB9IGNhdGNoIChYQUV4Y2VwdGlvbiBlKSB7CiAgICAgICAgICAgIGlmIChlLmVycm9yQ29kZSA9PSBYQUV4Y2VwdGlvbi5YQUVSX05PVEEpIHsKICAgICAgICAgICAgICAgIC8vIE5PVEEgaW4gcmVzcG9uc2UgdG8gZm9yZ2V0LCBtZWFucyB0aGUgcmVzb3VyY2UgYWxyZWFkeSBmb3Jnb3QgdGhlIHRyYW5zYWN0aW9uCiAgICAgICAgICAgICAgICAvLyBpZ25vcmUKICAgICAgICAgICAgfSBlbHNlIHsKICAgICAgICAgICAgICAgIHRocm93IChTeXN0ZW1FeGNlcHRpb24pIG5ldyBTeXN0ZW1FeGNlcHRpb24oIkVycm9yIGR1cmluZyBvbmUgcGhhc2UgY29tbWl0IikuaW5pdENhdXNlKGUpOwogICAgICAgICAgICB9CiAgICAgICAgfQogICAgICAgIAogICAgICAgIGlmIChjYXVzZSAhPSBudWxsKSB7CiAgICAgICAgICAgIGlmIChjYXVzZS5lcnJvckNvZGUgPT0gWEFFeGNlcHRpb24uWEFfSEVVUlJCKSB7CiAgICAgICAgICAgICAgICB0aHJvdyAoSGV1cmlzdGljUm9sbGJhY2tFeGNlcHRpb24pIG5ldyBIZXVyaXN0aWNSb2xsYmFja0V4Y2VwdGlvbigiRXJyb3IgZHVyaW5nIHR3byBwaGFzZSBjb21taXQiKS5pbml0Q2F1c2UoY2F1c2UpOwogICAgICAgICAgICB9IGVsc2UgaWYgKGNhdXNlLmVycm9yQ29kZSA9PSBYQUV4Y2VwdGlvbi5YQV9IRVVSTUlYKSB7CiAgICAgICAgICAgICAgICB0aHJvdyAoSGV1cmlzdGljTWl4ZWRFeGNlcHRpb24pIG5ldyBIZXVyaXN0aWNNaXhlZEV4Y2VwdGlvbigiRXJyb3IgZHVyaW5nIHR3byBwaGFzZSBjb21taXQiKS5pbml0Q2F1c2UoY2F1c2UpOwogICAgICAgICAgICB9IGVsc2UgaWYgKGNhdXNlLmVycm9yQ29kZSA9PSBYQUV4Y2VwdGlvbi5YQV9SQlJPTExCQUNLIAogICAgICAgICAgICAgICAgICAgIHx8IGNhdXNlLmVycm9yQ29kZSA9PSBYQUV4Y2VwdGlvbi5YQUVSX1JNRVJSIAogICAgICAgICAgICAgICAgICAgIHx8IGNhdXNlLmVycm9yQ29kZSA9PSBYQUV4Y2VwdGlvbi5YQUVSX05PVEEpIHsKICAgICAgICAgICAgICAgIHRocm93IChSb2xsYmFja0V4Y2VwdGlvbikgbmV3IFJvbGxiYWNrRXhjZXB0aW9uKCJFcnJvciBkdXJpbmcgdHdvIHBoYXNlIGNvbW1pdCIpLmluaXRDYXVzZShjYXVzZSk7CiAgICAgICAgICAgIH0gZWxzZSB7CiAgICAgICAgICAgICAgICB0aHJvdyAoU3lzdGVtRXhjZXB0aW9uKSBuZXcgU3lzdGVtRXhjZXB0aW9uKCJFcnJvciBkdXJpbmcgdHdvIHBoYXNlIGNvbW1pdCIpLmluaXRDYXVzZShjYXVzZSk7CiAgICAgICAgICAgIH0gCiAgICAgICAgfQogICAgfQogICAgCiAgICBwcml2YXRlIHZvaWQgY29tbWl0UmVzb3VyY2VzKExpc3Qgcm1zKSB0aHJvd3MgSGV1cmlzdGljUm9sbGJhY2tFeGNlcHRpb24sIEhldXJpc3RpY01peGVkRXhjZXB0aW9uLCBTeXN0ZW1FeGNlcHRpb24gewogICAgICAgIFhBRXhjZXB0aW9uIGNhdXNlID0gbnVsbDsKICAgICAgICBib29sZWFuIGV2ZXJjb21taXQgPSBmYWxzZTsKICAgICAgICBzeW5jaHJvbml6ZWQgKHRoaXMpIHsKICAgICAgICAgICAgc3RhdHVzID0gU3RhdHVzLlNUQVRVU19DT01NSVRUSU5HOwogICAgICAgIH0KICAgICAgICB0cnkgewogICAgICAgICAgICBmb3IgKEl0ZXJhdG9yIGkgPSBybXMuaXRlcmF0b3IoKTsgaS5oYXNOZXh0KCk7KSB7CiAgICAgICAgICAgICAgICBUcmFuc2FjdGlvbkJyYW5jaCBtYW5hZ2VyID0gKFRyYW5zYWN0aW9uQnJhbmNoKSBpLm5leHQoKTsKICAgICAgICAgICAgICAgIHRyeSB7CiAgICAgICAgICAgICAgICAgICAgbWFuYWdlci5nZXRDb21taXR0ZXIoKS5jb21taXQobWFuYWdlci5nZXRCcmFuY2hJZCgpLCBmYWxzZSk7CiAgICAgICAgICAgICAgICAgICAgZXZlcmNvbW1pdCA9IHRydWU7CiAgICAgICAgICAgICAgICB9IGNhdGNoIChYQUV4Y2VwdGlvbiBlKSB7CiAgICAgICAgICAgICAgICAgICAgbG9nLmVycm9yKCJVbmV4cGVjdGVkIGV4Y2VwdGlvbiBjb21taXR0aW5nICIgKyBtYW5hZ2VyLmdldENvbW1pdHRlcigpICsgIjsgY29udGludWluZyB0byBjb21taXQgb3RoZXIgUk1zIiwgZSk7CiAgICAgICAgICAgICAgICAgICAgCiAgICAgICAgICAgICAgICAgICAgaWYgKGUuZXJyb3JDb2RlID09IFhBRXhjZXB0aW9uLlhBX0hFVVJSQikgewogICAgICAgICAgICAgICAgICAgICAgICBsb2cuaW5mbygiVHJhbnNhY3Rpb24gaGFzIGJlZW4gaGV1cmlzdGljYWxseSByb2xsZWQgYmFjayIpOwogICAgICAgICAgICAgICAgICAgICAgICBjYXVzZSA9IGU7CiAgICAgICAgICAgICAgICAgICAgICAgIG1hbmFnZXIuZ2V0Q29tbWl0dGVyKCkuZm9yZ2V0KG1hbmFnZXIuZ2V0QnJhbmNoSWQoKSk7CiAgICAgICAgICAgICAgICAgICAgfSBlbHNlIGlmIChlLmVycm9yQ29kZSA9PSBYQUV4Y2VwdGlvbi5YQV9IRVVSTUlYKSB7CiAgICAgICAgICAgICAgICAgICAgICAgIGxvZy5pbmZvKCJUcmFuc2FjdGlvbiBoYXMgYmVlbiBoZXVyaXN0aWNhbGx5IGNvbW1pdHRlZCBhbmQgcm9sbGVkIGJhY2siKTsKICAgICAgICAgICAgICAgICAgICAgICAgY2F1c2UgPSBlOwogICAgICAgICAgICAgICAgICAgICAgICBldmVyY29tbWl0ID0gdHJ1ZTsKICAgICAgICAgICAgICAgICAgICAgICAgbWFuYWdlci5nZXRDb21taXR0ZXIoKS5mb3JnZXQobWFuYWdlci5nZXRCcmFuY2hJZCgpKTsKICAgICAgICAgICAgICAgICAgICB9IGVsc2UgaWYgKGUuZXJyb3JDb2RlID09IFhBRXhjZXB0aW9uLlhBX0hFVVJDT00pIHsKICAgICAgICAgICAgICAgICAgICAgICAgLy8gbGV0J3Mgbm90IHRocm93IGFuIGV4Y2VwdGlvbiBhcyB0aGUgdHJhbnNhY3Rpb24gaGFzIGJlZW4gY29tbWl0dGVkCiAgICAgICAgICAgICAgICAgICAgICAgIGxvZy5pbmZvKCJUcmFuc2FjdGlvbiBoYXMgYmVlbiBoZXVyaXN0aWNhbGx5IGNvbW1pdHRlZCIpOwogICAgICAgICAgICAgICAgICAgICAgICBldmVyY29tbWl0ID0gdHJ1ZTsKICAgICAgICAgICAgICAgICAgICAgICAgbWFuYWdlci5nZXRDb21taXR0ZXIoKS5mb3JnZXQobWFuYWdlci5nZXRCcmFuY2hJZCgpKTsKICAgICAgICAgICAgICAgICAgICB9IGVsc2UgewogICAgICAgICAgICAgICAgICAgICAgICBjYXVzZSA9IGU7CiAgICAgICAgICAgICAgICAgICAgfQogICAgICAgICAgICAgICAgfQogICAgICAgICAgICB9CiAgICAgICAgfSBjYXRjaCAoWEFFeGNlcHRpb24gZSkgewogICAgICAgICAgICBpZiAoZS5lcnJvckNvZGUgPT0gWEFFeGNlcHRpb24uWEFFUl9OT1RBKSB7CiAgICAgICAgICAgICAgICAvLyBOT1RBIGluIHJlc3BvbnNlIHRvIGZvcmdldCwgbWVhbnMgdGhlIHJlc291cmNlIGFscmVhZHkgZm9yZ290IHRoZSB0cmFuc2FjdGlvbgogICAgICAgICAgICAgICAgLy8gaWdub3JlCiAgICAgICAgICAgIH0gZWxzZSB7CiAgICAgICAgICAgICAgICB0aHJvdyAoU3lzdGVtRXhjZXB0aW9uKSBuZXcgU3lzdGVtRXhjZXB0aW9uKCJFcnJvciBkdXJpbmcgdHdvIHBoYXNlIGNvbW1pdCIpLmluaXRDYXVzZShlKTsKICAgICAgICAgICAgfQogICAgICAgIH0KICAgICAgICAvL2lmIGFsbCByZXNvdXJjZXMgd2VyZSByZWFkIG9ubHksIHdlIGRpZG4ndCB3cml0ZSBhIHByZXBhcmUgcmVjb3JkLgogICAgICAgIGlmICghcm1zLmlzRW1wdHkoKSkgewogICAgICAgICAgICB0cnkgewogICAgICAgICAgICAgICAgdHhuTG9nLmNvbW1pdCh4aWQsIGxvZ01hcmspOwogICAgICAgICAgICB9IGNhdGNoIChMb2dFeGNlcHRpb24gZSkgewogICAgICAgICAgICAgICAgbG9nLmVycm9yKCJVbmV4cGVjdGVkIGV4Y2VwdGlvbiBsb2dnaW5nIGNvbW1pdCBjb21wbGV0aW9uIGZvciB4aWQgIiArIHhpZCwgZSk7CiAgICAgICAgICAgICAgICB0aHJvdyAoU3lzdGVtRXhjZXB0aW9uKSBuZXcgU3lzdGVtRXhjZXB0aW9uKCJVbmV4cGVjdGVkIGVycm9yIGxvZ2dpbmcgY29tbWl0IGNvbXBsZXRpb24gZm9yIHhpZCAiICsgeGlkKS5pbml0Q2F1c2UoZSk7CiAgICAgICAgICAgIH0KICAgICAgICB9CiAgICAgICAgc3luY2hyb25pemVkICh0aGlzKSB7CiAgICAgICAgICAgIHN0YXR1cyA9IFN0YXR1cy5TVEFUVVNfQ09NTUlUVEVEOwogICAgICAgIH0KICAgICAgICBpZiAoY2F1c2UgIT0gbnVsbCkgewogICAgICAgICAgICBpZiAoY2F1c2UuZXJyb3JDb2RlID09IFhBRXhjZXB0aW9uLlhBX0hFVVJSQiAmJiAhZXZlcmNvbW1pdCkgewogICAgICAgICAgICAgICAgdGhyb3cgKEhldXJpc3RpY1JvbGxiYWNrRXhjZXB0aW9uKSBuZXcgSGV1cmlzdGljUm9sbGJhY2tFeGNlcHRpb24oIkVycm9yIGR1cmluZyB0d28gcGhhc2UgY29tbWl0IikuaW5pdENhdXNlKGNhdXNlKTsKICAgICAgICAgICAgfSBlbHNlIGlmIChjYXVzZS5lcnJvckNvZGUgPT0gWEFFeGNlcHRpb24uWEFfSEVVUlJCICYmIGV2ZXJjb21taXQpIHsKICAgICAgICAgICAgICAgIHRocm93IChIZXVyaXN0aWNNaXhlZEV4Y2VwdGlvbikgbmV3IEhldXJpc3RpY01peGVkRXhjZXB0aW9uKCJFcnJvciBkdXJpbmcgdHdvIHBoYXNlIGNvbW1pdCIpLmluaXRDYXVzZShjYXVzZSk7CiAgICAgICAgICAgIH0gZWxzZSBpZiAoY2F1c2UuZXJyb3JDb2RlID09IFhBRXhjZXB0aW9uLlhBX0hFVVJNSVgpIHsKICAgICAgICAgICAgICAgIHRocm93IChIZXVyaXN0aWNNaXhlZEV4Y2VwdGlvbikgbmV3IEhldXJpc3RpY01peGVkRXhjZXB0aW9uKCJFcnJvciBkdXJpbmcgdHdvIHBoYXNlIGNvbW1pdCIpLmluaXRDYXVzZShjYXVzZSk7CiAgICAgICAgICAgIH0gZWxzZSB7CiAgICAgICAgICAgICAgICB0aHJvdyAoU3lzdGVtRXhjZXB0aW9uKSBuZXcgU3lzdGVtRXhjZXB0aW9uKCJFcnJvciBkdXJpbmcgdHdvIHBoYXNlIGNvbW1pdCIpLmluaXRDYXVzZShjYXVzZSk7CiAgICAgICAgICAgIH0gCiAgICAgICAgfQogICAgfQoKICAgIHByaXZhdGUgc3RhdGljIFN0cmluZyBnZXRTdGF0ZVN0cmluZyhpbnQgc3RhdHVzKSB7CiAgICAgICAgc3dpdGNoIChzdGF0dXMpIHsKICAgICAgICAgICAgY2FzZSBTdGF0dXMuU1RBVFVTX0FDVElWRToKICAgICAgICAgICAgICAgIHJldHVybiAiU1RBVFVTX0FDVElWRSI7CiAgICAgICAgICAgIGNhc2UgU3RhdHVzLlNUQVRVU19QUkVQQVJJTkc6CiAgICAgICAgICAgICAgICByZXR1cm4gIlNUQVRVU19QUkVQQVJJTkciOwogICAgICAgICAgICBjYXNlIFN0YXR1cy5TVEFUVVNfUFJFUEFSRUQ6CiAgICAgICAgICAgICAgICByZXR1cm4gIlNUQVRVU19QUkVQQVJFRCI7CiAgICAgICAgICAgIGNhc2UgU3RhdHVzLlNUQVRVU19NQVJLRURfUk9MTEJBQ0s6CiAgICAgICAgICAgICAgICByZXR1cm4gIlNUQVRVU19NQVJLRURfUk9MTEJBQ0siOwogICAgICAgICAgICBjYXNlIFN0YXR1cy5TVEFUVVNfUk9MTElOR19CQUNLOgogICAgICAgICAgICAgICAgcmV0dXJuICJTVEFUVVNfUk9MTElOR19CQUNLIjsKICAgICAgICAgICAgY2FzZSBTdGF0dXMuU1RBVFVTX0NPTU1JVFRJTkc6CiAgICAgICAgICAgICAgICByZXR1cm4gIlNUQVRVU19DT01NSVRUSU5HIjsKICAgICAgICAgICAgY2FzZSBTdGF0dXMuU1RBVFVTX0NPTU1JVFRFRDoKICAgICAgICAgICAgICAgIHJldHVybiAiU1RBVFVTX0NPTU1JVFRFRCI7CiAgICAgICAgICAgIGNhc2UgU3RhdHVzLlNUQVRVU19ST0xMRURCQUNLOgogICAgICAgICAgICAgICAgcmV0dXJuICJTVEFUVVNfUk9MTEVEQkFDSyI7CiAgICAgICAgICAgIGNhc2UgU3RhdHVzLlNUQVRVU19OT19UUkFOU0FDVElPTjoKICAgICAgICAgICAgICAgIHJldHVybiAiU1RBVFVTX05PX1RSQU5TQUNUSU9OIjsKICAgICAgICAgICAgY2FzZSBTdGF0dXMuU1RBVFVTX1VOS05PV046CiAgICAgICAgICAgICAgICByZXR1cm4gIlNUQVRVU19VTktOT1dOIjsKICAgICAgICAgICAgZGVmYXVsdDoKICAgICAgICAgICAgICAgIHRocm93IG5ldyBBc3NlcnRpb25FcnJvcigpOwogICAgICAgIH0KICAgIH0KCiAgICBwdWJsaWMgYm9vbGVhbiBlcXVhbHMoT2JqZWN0IG9iaikgewogICAgICAgIGlmIChvYmogaW5zdGFuY2VvZiBUcmFuc2FjdGlvbkltcGwpIHsKICAgICAgICAgICAgVHJhbnNhY3Rpb25JbXBsIG90aGVyID0gKFRyYW5zYWN0aW9uSW1wbCkgb2JqOwogICAgICAgICAgICByZXR1cm4geGlkLmVxdWFscyhvdGhlci54aWQpOwogICAgICAgIH0gZWxzZSB7CiAgICAgICAgICAgIHJldHVybiBmYWxzZTsKICAgICAgICB9CiAgICB9CgogICAgLy93aGVuIHVzZWQgZnJvbSByZWNvdmVyeSwgZG8gbm90IGFkZCBtYW5hZ2VyIHRvIGFjdGl2ZSBvciBzdXNwZW5kZWQgcmVzb3VyY2UgbWFwcy4KICAgIC8vIFRoZSB4YXJlc291cmNlcyBoYXZlIGFscmVhZHkgYmVlbiBlbmRlZCB3aXRoIFRNU1VDQ0VTUy4KICAgIHB1YmxpYyBUcmFuc2FjdGlvbkJyYW5jaCBhZGRCcmFuY2hYaWQoWEFSZXNvdXJjZSB4YVJlcywgWGlkIGJyYW5jaElkKSB7CiAgICAgICAgVHJhbnNhY3Rpb25CcmFuY2ggbWFuYWdlciA9IG5ldyBUcmFuc2FjdGlvbkJyYW5jaCh4YVJlcywgYnJhbmNoSWQpOwogICAgICAgIHJlc291cmNlTWFuYWdlcnMuYWRkKG1hbmFnZXIpOwogICAgICAgIHJldHVybiBtYW5hZ2VyOwogICAgfQoKICAgIHByaXZhdGUgc3RhdGljIGNsYXNzIFRyYW5zYWN0aW9uQnJhbmNoIGltcGxlbWVudHMgVHJhbnNhY3Rpb25CcmFuY2hJbmZvIHsKICAgICAgICBwcml2YXRlIGZpbmFsIFhBUmVzb3VyY2UgY29tbWl0dGVyOwogICAgICAgIHByaXZhdGUgZmluYWwgWGlkIGJyYW5jaElkOwoKICAgICAgICBwdWJsaWMgVHJhbnNhY3Rpb25CcmFuY2goWEFSZXNvdXJjZSB4YVJlcywgWGlkIGJyYW5jaElkKSB7CiAgICAgICAgICAgIGNvbW1pdHRlciA9IHhhUmVzOwogICAgICAgICAgICB0aGlzLmJyYW5jaElkID0gYnJhbmNoSWQ7CiAgICAgICAgfQoKICAgICAgICBwdWJsaWMgWEFSZXNvdXJjZSBnZXRDb21taXR0ZXIoKSB7CiAgICAgICAgICAgIHJldHVybiBjb21taXR0ZXI7CiAgICAgICAgfQoKICAgICAgICBwdWJsaWMgWGlkIGdldEJyYW5jaElkKCkgewogICAgICAgICAgICByZXR1cm4gYnJhbmNoSWQ7CiAgICAgICAgfQoKICAgICAgICBwdWJsaWMgU3RyaW5nIGdldFJlc291cmNlTmFtZSgpIHsKICAgICAgICAgICAgaWYgKGNvbW1pdHRlciBpbnN0YW5jZW9mIE5hbWVkWEFSZXNvdXJjZSkgewogICAgICAgICAgICAgICAgcmV0dXJuICgoTmFtZWRYQVJlc291cmNlKSBjb21taXR0ZXIpLmdldE5hbWUoKTsKICAgICAgICAgICAgfSBlbHNlIHsKICAgICAgICAgICAgICAgIC8vIGlmIGl0IGlzbid0IGEgbmFtZWQgcmVzb3VyY2Ugc2hvdWxkIHdlIHJlYWxseSBzdG9wIGFsbCBwcm9jZXNzaW5nIGhlcmUhCiAgICAgICAgICAgICAgICAvLyBNYXliZSB0aGlzIHdvdWxkIGJlIGJldHRlciB0byBoYW5kbGUgZWxzZSB3aGVyZSBhbmQgZG8gd2UgcmVhbGx5IHdhbnQgdG8gcHJldmVudCBhbGwgcHJvY2Vzc2luZyBvZiB0cmFuc2FjdGlvbnM/CiAgICAgICAgICAgICAgICBsb2cuZXJyb3IoIlBsZWFzZSBjb3JyZWN0IHRoZSBpbnRlZ3JhdGlvbiBhbmQgc3VwcGx5IGEgTmFtZWRYQVJlc291cmNlIiwgbmV3IElsbGVnYWxTdGF0ZUV4Y2VwdGlvbigiQ2Fubm90IGxvZyB0cmFuc2FjdGlvbnMgYXMgIiArIGNvbW1pdHRlciArICIgaXMgbm90IGEgTmFtZWRYQVJlc291cmNlLiIpKTsKICAgICAgICAgICAgICAgIHJldHVybiBjb21taXR0ZXIudG9TdHJpbmcoKTsKICAgICAgICAgICAgfQogICAgICAgIH0KCiAgICAgICAgcHVibGljIFhpZCBnZXRCcmFuY2hYaWQoKSB7CiAgICAgICAgICAgIHJldHVybiBicmFuY2hJZDsKICAgICAgICB9CiAgICB9CgoKfQo=