{"id":473023,"date":"2025-09-02T15:12:47","date_gmt":"2025-09-02T15:12:47","guid":{"rendered":"http:\/\/savepearlharbor.com\/?p=473023"},"modified":"-0001-11-30T00:00:00","modified_gmt":"-0001-11-29T21:00:00","slug":"","status":"publish","type":"post","link":"https:\/\/savepearlharbor.com\/?p=473023","title":{"rendered":"<span>PEP 723 + uv: \u043e\u0434\u043d\u043e\u0444\u0430\u0439\u043b\u043e\u0432\u044b\u0435 \u0441\u043a\u0440\u0438\u043f\u0442\u044b \u0441 \u0437\u0430\u0432\u0438\u0441\u0438\u043c\u043e\u0441\u0442\u044f\u043c\u0438<\/span>"},"content":{"rendered":"<div><!--[--><!--]--><\/div>\n<div id=\"post-content-body\">\n<div>\n<div class=\"article-formatted-body article-formatted-body article-formatted-body_version-2\">\n<div xmlns=\"http:\/\/www.w3.org\/1999\/xhtml\">\n<p>\u041f\u0440\u0438\u0432\u0435\u0442, \u0425\u0430\u0431\u0440!<\/p>\n<p>\u0415\u0441\u043b\u0438 \u0432\u044b \u0441\u043e\u0431\u0438\u0440\u0430\u0435\u0442\u0435 \u043f\u0440\u043e\u0442\u043e\u0442\u0438\u043f \u043d\u0430\u00a0C++, \u0442\u043e \u043e\u0434\u0438\u043d \u0444\u0430\u0439\u043b \u0441 <code>main.cpp<\/code> \u0438\u043d\u043e\u0433\u0434\u0430 \u0440\u0435\u0430\u043b\u044c\u043d\u043e \u043a\u043e\u043c\u043f\u0438\u043b\u0438\u0440\u0443\u0435\u0442\u0441\u044f \u0432\u00a0\u0440\u0430\u0431\u043e\u0447\u0443\u044e \u0443\u0442\u0438\u043b\u0438\u0442\u0443. \u0411\u0438\u0431\u043b\u0438\u043e\u0442\u0435\u043a\u0438\u00a0\u043b\u0438\u0431\u043e \u0437\u0430\u0432\u043e\u0437\u044f\u0442\u0441\u044f \u043f\u0430\u043a\u0435\u0442\u043d\u044b\u043c \u043c\u0435\u043d\u0435\u0434\u0436\u0435\u0440\u043e\u043c \u0437\u0430\u0440\u0430\u043d\u0435\u0435,\u00a0\u043b\u0438\u0431\u043e \u0443\u00a0\u0432\u0430\u0441 \u0435\u0441\u0442\u044c header\u2011only \u0437\u0430\u0432\u0438\u0441\u0438\u043c\u043e\u0441\u0442\u044c \u0438 \u0432\u0441\u0451 \u0432\u0437\u043b\u0435\u0442\u0430\u0435\u0442. \u0412\u00a0Python \u0434\u043e\u043b\u0433\u043e\u0435 \u0432\u0440\u0435\u043c\u044f \u044d\u0442\u043e\u00a0\u0431\u044b\u043b\u043e \u0431\u043e\u043b\u044c\u044e: \u043b\u044e\u0431\u043e\u0439 \u043e\u0434\u043d\u043e\u0444\u0430\u0439\u043b\u043e\u0432\u044b\u0439 \u0441\u043a\u0440\u0438\u043f\u0442, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u0442\u0440\u0435\u0431\u0443\u0435\u0442 <code>requests<\/code> \u0438\u043b\u0438 <code>rich<\/code>, \u0443\u0436\u0435 \u0442\u044f\u043d\u0435\u0442 \u0437\u0430\u00a0\u0441\u043e\u0431\u043e\u0439 \u0432\u0438\u0440\u0442\u0443\u0430\u043b\u044c\u043d\u044b\u0435 \u043e\u043a\u0440\u0443\u0436\u0435\u043d\u0438\u044f, \u0438\u043d\u0441\u0442\u0440\u0443\u043a\u0446\u0438\u0438 \u0432\u00a0README \u0438 \u043b\u043e\u043a\u0430\u043b\u044c\u043d\u044b\u0435 \u0444\u0438\u0447\u0438. <\/p>\n<p>\u0415\u0441\u0442\u044c \u0440\u0430\u0431\u043e\u0447\u0438\u0439 \u0441\u0442\u0430\u043d\u0434\u0430\u0440\u0442 \u0434\u043b\u044f\u00a0\u043d\u043e\u0440\u043c\u0430\u043b\u044c\u043d\u044b\u0445 \u043e\u0434\u043d\u043e\u0444\u0430\u0439\u043b\u043e\u0432\u044b\u0445 \u0441\u0446\u0435\u043d\u0430\u0440\u0438\u0435\u0432 \u0441\u00a0\u0437\u0430\u0432\u0438\u0441\u0438\u043c\u043e\u0441\u0442\u044f\u043c\u0438\u00a0\u2014 PEP 723: \u0432\u044b \u043e\u0431\u044a\u044f\u0432\u043b\u044f\u0435\u0442\u0435 \u0437\u0430\u0432\u0438\u0441\u0438\u043c\u043e\u0441\u0442\u0438 \u043f\u0440\u044f\u043c\u043e \u0432\u00a0\u043a\u043e\u043c\u043c\u0435\u043d\u0442\u0430\u0440\u0438\u044f\u0445, \u0430\u00a0\u0440\u0430\u043d\u043d\u0435\u0440 \u0441\u0442\u0430\u0432\u0438\u0442 \u0432\u0441\u0451 \u0441\u0430\u043c \u0438 \u0437\u0430\u043f\u0443\u0441\u043a\u0430\u0435\u0442 \u0432\u00a0\u0438\u0437\u043e\u043b\u0438\u0440\u043e\u0432\u0430\u043d\u043d\u043e\u0439 \u0441\u0440\u0435\u0434\u0435. \u0412\u00a0\u0441\u0432\u044f\u0437\u043a\u0435 \u0441 <code>uv<\/code> \u043f\u043e\u043b\u0443\u0447\u0430\u0435\u0442\u0441\u044f \u043d\u0435\u043f\u043b\u043e\u0445\u043e\u0439 \u0442\u0430\u043a\u043e\u0439 \u0441\u043f\u043e\u0441\u043e\u0431 \u0434\u0435\u043b\u0438\u0442\u044c\u0441\u044f \u0441\u043a\u0440\u0438\u043f\u0442\u0430\u043c\u0438, \u0432\u00a0\u0442\u043e\u043c \u0447\u0438\u0441\u043b\u0435 \u0434\u043b\u044f\u00a0\u043f\u0432\u0441\u043f\u043e\u043c\u043e\u0433\u0430\u0442\u0435\u043b\u044c\u043d\u044b\u0445 \u0437\u0430\u0434\u0430\u0447. \u0418 \u0434\u0430, \u0443\u00a0\u044d\u0442\u043e\u0439 \u043a\u0440\u0430\u0441\u043e\u0442\u044b \u0435\u0441\u0442\u044c \u043d\u044e\u0430\u043d\u0441\u044b \u0431\u0435\u0437\u043e\u043f\u0430\u0441\u043d\u043e\u0441\u0442\u0438, \u043e\u00a0\u043d\u0438\u0445 \u043f\u043e\u0433\u043e\u0432\u043e\u0440\u0438\u043c \u043e\u0442\u0434\u0435\u043b\u044c\u043d\u043e.<\/p>\n<h3>\u0427\u0442\u043e \u0442\u0430\u043a\u043e\u0435 PEP 723 \u0438 \u043a\u0430\u043a \u044d\u0442\u043e \u0432\u044b\u0433\u043b\u044f\u0434\u0438\u0442 <\/h3>\n<p>PEP 723\u00a0\u043e\u043f\u0440\u0435\u0434\u0435\u043b\u044f\u0435\u0442 \u0431\u043b\u043e\u043a \u043c\u0435\u0442\u0430\u0434\u0430\u043d\u043d\u044b\u0445 \u0432\u00a0\u043a\u043e\u043c\u043c\u0435\u043d\u0442\u0430\u0440\u0438\u044f\u0445, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u043f\u0430\u0440\u0441\u0438\u0442\u0441\u044f \u0432\u043d\u0435\u0448\u043d\u0438\u043c\u0438 \u0438\u043d\u0441\u0442\u0440\u0443\u043c\u0435\u043d\u0442\u0430\u043c\u0438. \u0424\u043e\u0440\u043c\u0430\u0442 \u043f\u0440\u043e\u0441\u0442: \u0441\u0432\u0435\u0440\u0445\u0443 \u0438 \u0441\u043d\u0438\u0437\u0443 \u043c\u0430\u0440\u043a\u0435\u0440\u044b, \u0432\u043d\u0443\u0442\u0440\u0438 TOML c \u043f\u043e\u043b\u044f\u043c\u0438 <code>dependencies<\/code> \u0438 <code>requires-python<\/code>. <\/p>\n<p>\u041f\u0440\u0438\u043c\u0435\u0440 \u043c\u0438\u043d\u0438\u043c\u0430\u043b\u044c\u043d\u043e\u0433\u043e \u0441\u043a\u0440\u0438\u043f\u0442\u0430:<\/p>\n<pre><code class=\"yaml\"># \/\/\/ script # requires-python = \"&gt;=3.11\" # dependencies = [ #   \"httpx&lt;1.0\", #   \"rich&gt;=13.7\", # ] # \/\/\/ import httpx from rich import print  def main() -&gt; None:     r = httpx.get(\"https:\/\/httpbin.org\/json\", timeout=10)     r.raise_for_status()     print({\"status\": r.status_code, \"len\": len(r.text)})  if __name__ == \"__main__\":     main()<\/code><\/pre>\n<p>\u0417\u0430\u043f\u0443\u0441\u043a \u0447\u0435\u0440\u0435\u0437 <code>uv<\/code>:<\/p>\n<pre><code class=\"bash\">uv run example.py<\/code><\/pre>\n<p><code>uv<\/code> \u0441\u043e\u0437\u0434\u0430\u0441\u0442 \u0438\u0437\u043e\u043b\u0438\u0440\u043e\u0432\u0430\u043d\u043d\u0443\u044e \u0441\u0440\u0435\u0434\u0443 \u0432\u00a0\u043a\u0435\u0448\u0435, \u0443\u0441\u0442\u0430\u043d\u043e\u0432\u0438\u0442 \u0437\u0430\u0432\u0438\u0441\u0438\u043c\u043e\u0441\u0442\u0438 \u0438 \u0432\u044b\u043f\u043e\u043b\u043d\u0438\u0442 \u043a\u043e\u0434. \u041d\u0438\u043a\u0430\u043a\u0438\u0445 \u0440\u0443\u0447\u043d\u044b\u0445 <code>venv<\/code>.<\/p>\n<h3>\u0414\u043e\u0431\u0430\u0432\u043b\u044f\u0435\u043c \u0438 \u0443\u043f\u0440\u0430\u0432\u043b\u044f\u0435\u043c \u0437\u0430\u0432\u0438\u0441\u0438\u043c\u043e\u0441\u0442\u044f\u043c\u0438 \u043f\u0440\u044f\u043c\u043e \u0438\u0437 CLI<\/h3>\n<p>PEP 723\u00a0\u2014 \u044d\u0442\u043e \u043f\u0440\u043e\u00a0\u0444\u043e\u0440\u043c\u0430\u0442, \u043d\u043e\u00a0\u043f\u0440\u0430\u0432\u0438\u0442\u044c TOML \u0440\u0443\u043a\u0430\u043c\u0438\u00a0\u0431\u044b\u0441\u0442\u0440\u043e \u043d\u0430\u0434\u043e\u0435\u0434\u0430\u0435\u0442. <code>uv<\/code> \u0443\u043c\u0435\u0435\u0442 \u0440\u0435\u0434\u0430\u043a\u0442\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u0431\u043b\u043e\u043a \u0437\u0430\u0432\u0438\u0441\u0438\u043c\u043e\u0441\u0442\u0435\u0439 \u0432\u00a0\u0444\u0430\u0439\u043b\u0435:<\/p>\n<pre><code class=\"bash\">uv add --script example.py \"pydantic&gt;=2.8\" uv remove --script example.py \"httpx\" uv run example.py<\/code><\/pre>\n<p>\u0415\u0441\u043b\u0438 \u043d\u0443\u0436\u0435\u043d \u0438\u0441\u043f\u043e\u043b\u043d\u044f\u0435\u043c\u044b\u0439 \u0444\u0430\u0439\u043b \u0431\u0435\u0437\u00a0\u044f\u0432\u043d\u043e\u0433\u043e \u0432\u044b\u0437\u043e\u0432\u0430 <code>uv run<\/code>, \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u0435\u043c shebang:<\/p>\n<pre><code class=\"bash\">#!\/usr\/bin\/env -S uv run --script # \/\/\/ script # dependencies = [\"rich\"] # \/\/\/ from rich import print print(\"hello\")<\/code><\/pre>\n<p>\u0421\u0442\u0430\u0432\u0438\u043c \u043f\u0440\u0430\u0432\u0430 \u0438 \u0437\u0430\u043f\u0443\u0441\u043a\u0430\u0435\u043c \u0438\u0437\u00a0\u043a\u0430\u0442\u0430\u043b\u043e\u0433\u0430:<\/p>\n<pre><code class=\"bash\">chmod +x greet .\/greet<\/code><\/pre>\n<h3>\u041b\u043e\u0447\u0438\u043c \u0437\u0430\u0432\u0438\u0441\u0438\u043c\u043e\u0441\u0442\u0438 \u0438 \u0434\u0435\u043b\u0430\u0435\u043c \u0441\u043a\u0440\u0438\u043f\u0442 \u0432\u043e\u0441\u043f\u0440\u043e\u0438\u0437\u0432\u043e\u0434\u0438\u043c\u044b\u043c<\/h3>\n<p>\u041e\u0434\u043d\u043e \u0434\u0435\u043b\u043e \u0443\u0441\u0442\u0430\u043d\u043e\u0432\u0438\u0442\u044c \u0430\u043a\u0442\u0443\u0430\u043b\u044c\u043d\u044b\u0435 \u0432\u0435\u0440\u0441\u0438\u0438, \u0434\u0440\u0443\u0433\u043e\u0435\u00a0\u2014 \u0437\u0430\u0444\u0438\u043a\u0441\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u0438\u0445. <code>uv<\/code> \u043f\u043e\u0434\u0434\u0435\u0440\u0436\u0438\u0432\u0430\u0435\u0442 lock\u2011\u0444\u0430\u0439\u043b \u0434\u043b\u044f\u00a0\u0441\u043a\u0440\u0438\u043f\u0442\u043e\u0432. <\/p>\n<pre><code class=\"bash\">uv lock --script example.py # \u043f\u043e\u044f\u0432\u0438\u0442\u0441\u044f example.py.lock  # \u0442\u0435\u043f\u0435\u0440\u044c \u043b\u044e\u0431\u044b\u0435 \u043e\u043f\u0435\u0440\u0430\u0446\u0438\u0438 \u0431\u0443\u0434\u0443\u0442 \u0443\u0447\u0438\u0442\u044b\u0432\u0430\u0442\u044c lock: uv run --script example.py uv add --script example.py \"rich\"      # \u043e\u0431\u043d\u043e\u0432\u0438\u0442 lock uv export --script example.py -o req.txt<\/code><\/pre>\n<p>\u041c\u043e\u0436\u043d\u043e \u043e\u0433\u0440\u0430\u043d\u0438\u0447\u0438\u0442\u044c \u0441\u0432\u0435\u0436\u0435\u0441\u0442\u044c \u043f\u0430\u043a\u0435\u0442\u043e\u0432 \u0434\u0430\u0442\u043e\u0439, \u0447\u0442\u043e\u0431\u044b \u0438\u0437\u0431\u0435\u0436\u0430\u0442\u044c \u043d\u0435\u043e\u0436\u0438\u0434\u0430\u043d\u043d\u044b\u0445 \u0430\u043f\u0434\u0435\u0439\u0442\u043e\u0432 \u0447\u0435\u0440\u0435\u0437 \u0433\u043e\u0434. \u0412\u00a0\u0431\u043b\u043e\u043a\u0435 <code>[tool.uv]<\/code> \u0432\u043d\u0443\u0442\u0440\u0438 \u043c\u0435\u0442\u0430\u0434\u0430\u043d\u043d\u044b\u0445:<\/p>\n<pre><code class=\"python\"># \/\/\/ script # requires-python = \"&gt;=3.11\" # dependencies = [\"httpx&lt;1.0\"] # [tool.uv] # exclude-newer = \"2025-08-01T00:00:00Z\" # \/\/\/<\/code><\/pre>\n<p><code>exclude-newer<\/code> \u0437\u0430\u0441\u0442\u0430\u0432\u0438\u0442 \u0440\u0435\u0437\u043e\u043b\u0432\u0435\u0440 \u0438\u0433\u043d\u043e\u0440\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u0440\u0435\u043b\u0438\u0437\u044b, \u0432\u044b\u0448\u0435\u0434\u0448\u0438\u0435 \u043f\u043e\u0441\u043b\u0435 \u0443\u043a\u0430\u0437\u0430\u043d\u043d\u043e\u0439 \u0434\u0430\u0442\u044b. <\/p>\n<p>\u0410\u00a0\u0435\u0441\u043b\u0438 \u043d\u0443\u0436\u043d\u043e \u0438\u043c\u0435\u043d\u043d\u043e pip\u2011\u0441\u043e\u0432\u043c\u0435\u0441\u0442\u0438\u043c\u043e\u0435 \u0437\u0430\u0444\u0438\u043a\u0441\u0438\u0440\u043e\u0432\u0430\u043d\u043d\u043e\u0435 \u0434\u0435\u0440\u0435\u0432\u043e \u0434\u043b\u044f\u00a0\u0434\u043b\u0438\u0442\u0435\u043b\u044c\u043d\u044b\u0445 CI\u2011\u043f\u0430\u0439\u043f\u043b\u0430\u0439\u043d\u043e\u0432, \u0443 <code>uv<\/code> \u0435\u0441\u0442\u044c \u0438\u043d\u0441\u0442\u0440\u0443\u043c\u0435\u043d\u0442\u044b \u0443\u0440\u043e\u0432\u043d\u044f <code>pip-tools<\/code>: <code>uv pip compile<\/code> \u0438 <code>uv pip sync<\/code>. \u041e\u043d\u0438 \u043f\u043e\u0437\u0432\u043e\u043b\u044f\u044e\u0442 \u0441\u0433\u0435\u043d\u0435\u0440\u0438\u0440\u043e\u0432\u0430\u0442\u044c <code>requirements.txt<\/code> \u0438\u0437\u00a0\u0438\u0441\u0445\u043e\u0434\u043d\u0438\u043a\u0430\u2011\u0434\u0435\u043a\u043b\u0430\u0440\u0430\u0446\u0438\u0438 \u0438 \u0441\u0438\u043d\u0445\u0440\u043e\u043d\u0438\u0437\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u043e\u043a\u0440\u0443\u0436\u0435\u043d\u0438\u0435 \u043e\u0434\u0438\u043d\u2011\u0432\u2011\u043e\u0434\u0438\u043d. <\/p>\n<h3>\u041a\u043e\u043d\u0442\u0440\u043e\u043b\u044c \u0432\u0435\u0440\u0441\u0438\u0438 Python \u043d\u0430 \u0440\u0430\u043d\u0442\u0430\u0439\u043c\u0435<\/h3>\n<p>\u0421\u043a\u0440\u0438\u043f\u0442 \u043c\u043e\u0436\u0435\u0442 \u0442\u0440\u0435\u0431\u043e\u0432\u0430\u0442\u044c \u043a\u043e\u043d\u043a\u0440\u0435\u0442\u043d\u0443\u044e \u0432\u0435\u0442\u043a\u0443 \u0438\u043d\u0442\u0435\u0440\u043f\u0440\u0435\u0442\u0430\u0442\u043e\u0440\u0430. \u0412\u00a0\u044d\u0442\u043e\u043c \u0441\u043b\u0443\u0447\u0430\u0435 <code>uv<\/code> \u043f\u043e\u0437\u0432\u043e\u043b\u044f\u0435\u0442 \u0437\u0430\u043f\u0440\u043e\u0441\u0438\u0442\u044c \u043d\u0443\u0436\u043d\u0443\u044e \u0432\u0435\u0440\u0441\u0438\u044e \u043f\u0440\u0438\u00a0\u0437\u0430\u043f\u0443\u0441\u043a\u0435:<\/p>\n<pre><code class=\"bash\">uv run --python 3.10 example.py uv run --python 3.12 example.py<\/code><\/pre>\n<h3>\u0411\u0435\u0437\u043e\u043f\u0430\u0441\u043d\u043e\u0441\u0442\u044c<\/h3>\n<p>\u041e\u0434\u043d\u043e\u0444\u0430\u0439\u043b\u043e\u0432\u044b\u0435 \u0441\u043a\u0440\u0438\u043f\u0442\u044b \u0443\u0434\u043e\u0431\u043d\u043e \u0437\u0430\u043f\u0443\u0441\u043a\u0430\u0442\u044c \u043a\u0430\u043a\u00a0\u0435\u0441\u0442\u044c, \u0438 \u0438\u043c\u0435\u043d\u043d\u043e \u043f\u043e\u044d\u0442\u043e\u043c\u0443 \u043d\u0430\u0434\u043e\u00a0\u0431\u044b\u0442\u044c \u0430\u043a\u043a\u0443\u0440\u0430\u0442\u043d\u0435\u0435 \u043e\u0431\u044b\u0447\u043d\u043e\u0433\u043e. \u0420\u0438\u0441\u043a\u0438:<\/p>\n<ol>\n<li>\n<p>\u041f\u043e\u0434\u043c\u0435\u043d\u0430 \u0437\u0430\u0432\u0438\u0441\u0438\u043c\u043e\u0441\u0442\u0435\u0439. \u0422\u0440\u0438\u0432\u0438\u0430\u043b\u044c\u043d\u044b\u0435 \u0430\u0442\u0430\u043a\u0438 \u043d\u0430\u00a0\u0446\u0435\u043f\u043e\u0447\u043a\u0443 \u043f\u043e\u0441\u0442\u0430\u0432\u043e\u043a \u0447\u0435\u0440\u0435\u0437 \u0437\u0430\u0440\u0435\u0433\u0438\u0441\u0442\u0440\u0438\u0440\u043e\u0432\u0430\u043d\u043d\u044b\u0435 \u043f\u0430\u043a\u0435\u0442\u044b \u0441\u00a0\u043f\u043e\u0445\u043e\u0436\u0438\u043c\u0438 \u0438\u043c\u0435\u043d\u0430\u043c\u0438.<\/p>\n<\/li>\n<li>\n<p>\u041d\u0435\u043f\u0440\u0435\u0434\u0441\u043a\u0430\u0437\u0443\u0435\u043c\u044b\u0435 \u0430\u043f\u0433\u0440\u0435\u0439\u0434\u044b. \u0421\u0435\u0433\u043e\u0434\u043d\u044f \u0432\u0441\u0451 \u0447\u0438\u0441\u0442\u043e, \u0437\u0430\u0432\u0442\u0440\u0430 \u043d\u043e\u0432\u044b\u0439 \u0440\u0435\u043b\u0438\u0437 \u0442\u0440\u0430\u043d\u0437\u0438\u0442\u0438\u0432\u043d\u043e\u0439 \u0437\u0430\u0432\u0438\u0441\u0438\u043c\u043e\u0441\u0442\u0438 \u043b\u043e\u043c\u0430\u0435\u0442 \u0438\u043d\u0432\u0430\u0440\u0438\u0430\u043d\u0442\u044b.<\/p>\n<\/li>\n<li>\n<p>\u0418\u0441\u0442\u043e\u0447\u043d\u0438\u043a\u0438 \u043f\u0430\u043a\u0435\u0442\u043e\u0432. \u041f\u0435\u0440\u0435\u043e\u043f\u0440\u0435\u0434\u0435\u043b\u0435\u043d\u0438\u0435 \u0438\u043d\u0434\u0435\u043a\u0441\u043e\u0432, \u043e\u0442\u043a\u043b\u044e\u0447\u0435\u043d\u0438\u0435 TLS\u2011\u043f\u0440\u043e\u0432\u0435\u0440\u043e\u043a, \u0432\u043d\u0443\u0442\u0440\u0435\u043d\u043d\u0438\u0435 \u0437\u0435\u0440\u043a\u0430\u043b\u0430 \u0431\u0435\u0437\u00a0\u043f\u043e\u043b\u0438\u0442\u0438\u043a\u0438.<\/p>\n<\/li>\n<li>\n<p>\u0423\u044f\u0437\u0432\u0438\u043c\u043e\u0441\u0442\u0438 \u0432\u00a0\u0443\u0436\u0435 \u0432\u044b\u0431\u0440\u0430\u043d\u043d\u044b\u0445 \u0432\u0435\u0440\u0441\u0438\u044f\u0445.<\/p>\n<\/li>\n<\/ol>\n<p>\u041a\u043e\u043d\u043a\u0440\u0435\u0442\u043d\u044b\u0435 \u043c\u0435\u0440\u044b:<\/p>\n<p><strong>\u0424\u0438\u043a\u0441\u0430\u0446\u0438\u044f \u0438 \u0430\u0443\u0434\u0438\u0442. <\/strong>\u0414\u043b\u044f\u00a0\u0441\u043a\u0440\u0438\u043f\u0442\u043e\u0432 \u0441\u043e\u0437\u0434\u0430\u0451\u043c <code>example.py.lock<\/code> \u0438 \u0440\u0435\u0433\u0443\u043b\u044f\u0440\u043d\u043e \u043f\u0440\u043e\u0433\u043e\u043d\u044f\u0435\u043c \u0430\u0443\u0434\u0438\u0442 \u0437\u0430\u0432\u0438\u0441\u0438\u043c\u043e\u0441\u0442\u0435\u0439. \u0412\u00a0\u044d\u043a\u043e\u0441\u0438\u0441\u0442\u0435\u043c\u0435 PyPA \u0434\u043b\u044f\u00a0\u044d\u0442\u043e\u0433\u043e \u0435\u0441\u0442\u044c <code>pip-audit<\/code>, \u043e\u043d \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u0435\u0442 \u0431\u0430\u0437\u0443 \u0443\u044f\u0437\u0432\u0438\u043c\u043e\u0441\u0442\u0435\u0439 PyPI \u0438 \u043c\u043e\u0436\u0435\u0442 \u0440\u0430\u0431\u043e\u0442\u0430\u0442\u044c \u043f\u043e\u0432\u0435\u0440\u0445 \u0432\u0430\u0448\u0435\u0433\u043e requirements \u0438\u043b\u0438\u00a0\u0443\u0441\u0442\u0430\u043d\u043e\u0432\u043b\u0435\u043d\u043d\u043e\u0439 \u0441\u0440\u0435\u0434\u044b. \u041c\u043e\u0436\u043d\u043e \u0437\u0430\u043f\u0443\u0441\u043a\u0430\u0442\u044c \u0447\u0435\u0440\u0435\u0437 <code>uvx<\/code> \u0432\u00a0\u0438\u0437\u043e\u043b\u0438\u0440\u043e\u0432\u0430\u043d\u043d\u043e\u043c \u043e\u043a\u0440\u0443\u0436\u0435\u043d\u0438\u0438:<\/p>\n<pre><code class=\"bash\">uvx pip-audit -r req.txt # \u0438\u043b\u0438 \u043f\u0440\u043e\u0441\u043a\u0430\u043d\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u043b\u043e\u043a\u0430\u043b\u044c\u043d\u0443\u044e \u0441\u0440\u0435\u0434\u0443 uvx pip-audit --local<\/code><\/pre>\n<p>\u041f\u0440\u043e\u0435\u043a\u0442 \u043f\u043e\u0434\u0434\u0435\u0440\u0436\u0438\u0432\u0430\u0435\u0442\u0441\u044f PyPA, \u0435\u0441\u0442\u044c GitHub Action. <\/p>\n<p><strong>\u0425\u0435\u0448\u0438 \u0438 \u0441\u0438\u043d\u0445\u0440\u043e\u043d\u0438\u0437\u0430\u0446\u0438\u044f<\/strong>. \u0414\u043b\u044f\u00a0\u0431\u043e\u043b\u044c\u0448\u0438\u0445 \u043f\u0430\u0439\u043f\u043b\u0430\u0439\u043d\u043e\u0432 \u043c\u043e\u0436\u043d\u043e \u0433\u0435\u043d\u0435\u0440\u0438\u0440\u043e\u0432\u0430\u0442\u044c requirements \u0441\u00a0\u0445\u0435\u0448\u0430\u043c\u0438, \u0430\u00a0\u0443\u0441\u0442\u0430\u043d\u0430\u0432\u043b\u0438\u0432\u0430\u0442\u044c \u0447\u0435\u0440\u0435\u0437 <code>uv pip sync<\/code>, \u0447\u0442\u043e\u0431\u044b \u043e\u043a\u0440\u0443\u0436\u0435\u043d\u0438\u0435 \u0441\u043e\u043e\u0442\u0432\u0435\u0442\u0441\u0442\u0432\u043e\u0432\u0430\u043b\u043e \u0444\u0430\u0439\u043b\u0443 \u043e\u0434\u0438\u043d \u043a\u00a0\u043e\u0434\u043d\u043e\u043c\u0443. \u0423\u043f\u0440\u0430\u0432\u043b\u0435\u043d\u0438\u0435 \u0445\u0435\u0448\u0430\u043c\u0438 \u0438 \u043f\u0430\u0440\u0430\u043c\u0435\u0442\u0440\u044b \u043a\u043e\u043c\u043f\u0438\u043b\u044f\u0446\u0438\u0438 \u043a\u043e\u043d\u0444\u0438\u0433\u0443\u0440\u0438\u0440\u0443\u044e\u0442\u0441\u044f \u0432 <code>uv.toml<\/code>\/<code>pyproject.toml<\/code> \u0434\u043b\u044f <code>uv pip compile<\/code>. <\/p>\n<p><strong>\u0418\u043d\u0434\u0435\u043a\u0441\u044b \u0438 TLS. <\/strong>\u041d\u0435\u00a0\u0438\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u0435\u043c \u0434\u043e\u0432\u0435\u0440\u0435\u043d\u043d\u044b\u0435 \u043d\u0435\u0431\u0435\u0437\u043e\u043f\u0430\u0441\u043d\u044b\u0435 \u0445\u043e\u0441\u0442\u044b. \u0412\u00a0\u0441\u043f\u0440\u0430\u0432\u043e\u0447\u043d\u0438\u043a\u0435 CLI \u043f\u0440\u044f\u043c\u043e \u0435\u0441\u0442\u044c \u043f\u0440\u0435\u0434\u0443\u043f\u0440\u0435\u0436\u0434\u0435\u043d\u0438\u0435: \u0444\u043b\u0430\u0433 <code>--allow-insecure-host<\/code> \u043e\u0442\u043a\u043b\u044e\u0447\u0430\u0435\u0442 \u0432\u0435\u0440\u0438\u0444\u0438\u043a\u0430\u0446\u0438\u044e \u0446\u0435\u043f\u043e\u0447\u043a\u0438 \u0441\u0435\u0440\u0442\u0438\u0444\u0438\u043a\u0430\u0442\u043e\u0432, \u0447\u0442\u043e\u00a0\u0434\u0435\u043b\u0430\u0435\u0442 \u0432\u0430\u0441 \u0443\u044f\u0437\u0432\u0438\u043c\u044b\u043c\u0438 \u0434\u043b\u044f\u00a0MITM. <\/p>\n<p><strong>\u0412\u043e\u0441\u043f\u0440\u043e\u0438\u0437\u0432\u043e\u0434\u0438\u043c\u043e\u0441\u0442\u044c \u043f\u043e\u00a0\u0432\u0440\u0435\u043c\u0435\u043d\u0438.<\/strong> \u0414\u043e\u0431\u0430\u0432\u043b\u044f\u0435\u043c <code>exclude-newer<\/code> \u043a\u00a0\u0441\u043a\u0440\u0438\u043f\u0442\u0443, \u0434\u0435\u0440\u0436\u0438\u0442\u0435 lock \u0440\u044f\u0434\u043e\u043c \u0441\u00a0\u043d\u0438\u043c, \u0430\u00a0CI \u0437\u0430\u043f\u0443\u0441\u043a\u0430\u0439\u0442\u0435 \u0441 <code>uv run --script<\/code> \u0438\u043b\u0438\u00a0\u0441\u00a0\u044d\u043a\u0441\u043f\u043e\u0440\u0442\u043e\u043c \u0432 <code>requirements.txt<\/code> \u0438 \u043f\u043e\u0441\u043b\u0435\u0434\u0443\u044e\u0449\u0438\u043c <code>uv pip sync<\/code>. <\/p>\n<h4>\u0421\u0446\u0435\u043d\u0430\u0440\u0438\u0439: \u043c\u0430\u043b\u0435\u043d\u044c\u043a\u0438\u0439 CLI-\u0438\u043d\u0442\u0435\u0433\u0440\u0430\u0446\u0438\u043e\u043d\u043d\u044b\u0439 \u0441\u043a\u0440\u0438\u043f\u0442<\/h4>\n<p>\u0423\u0441\u043b\u043e\u0432\u0438\u0435: \u043d\u0443\u0436\u0435\u043d \u043e\u0434\u043d\u043e\u0444\u0430\u0439\u043b\u043e\u0432\u044b\u0439 \u0438\u043d\u0441\u0442\u0440\u0443\u043c\u0435\u043d\u0442 \u0434\u043b\u044f\u00a0\u0432\u044b\u0433\u0440\u0443\u0437\u043a\u0438 \u0434\u0430\u043d\u043d\u044b\u0445 \u0438\u0437\u00a0API, \u0441\u00a0\u0440\u0435\u0442\u0440\u0430\u044f\u043c\u0438 \u0438 \u043b\u043e\u0433\u0430\u043c\u0438, \u0431\u0435\u0437\u00a0\u0443\u0441\u0442\u0430\u043d\u043e\u0432\u043a\u0438 \u043f\u0440\u043e\u0435\u043a\u0442\u0430 \u0438 \u0441\u00a0\u0432\u043e\u0441\u043f\u0440\u043e\u0438\u0437\u0432\u043e\u0434\u0438\u043c\u044b\u043c\u0438 \u0432\u0435\u0440\u0441\u0438\u044f\u043c\u0438.<\/p>\n<p><code>fetch_users.py<\/code>:<\/p>\n<pre><code class=\"python\"># \/\/\/ script # requires-python = \"&gt;=3.11\" # dependencies = [ #   \"httpx==0.27.2\", #   \"tenacity==9.0.0\", #   \"structlog==24.4.0\", # ] # [tool.uv] # exclude-newer = \"2025-08-01T00:00:00Z\" # \/\/\/ from __future__ import annotations  import os import sys import json import time import httpx import structlog from tenacity import retry, stop_after_attempt, wait_exponential, retry_if_exception_type  log = structlog.get_logger()  API_URL = os.environ.get(\"API_URL\", \"https:\/\/httpbin.org\/json\") TIMEOUT = httpx.Timeout(10.0, connect=5.0)  class FetchError(RuntimeError):     pass  @retry(     reraise=True,     stop=stop_after_attempt(4),     wait=wait_exponential(multiplier=0.5, min=0.5, max=5),     retry=retry_if_exception_type((httpx.HTTPError, FetchError)), ) def fetch_json(client: httpx.Client, url: str) -&gt; dict:     r = client.get(url)     if r.status_code &gt;= 500:         # \u0441\u0435\u0440\u0432\u0435\u0440\u043d\u0430\u044f \u043e\u0448\u0438\u0431\u043a\u0430 \u2014 \u043f\u0440\u043e\u0431\u0443\u0435\u043c \u043f\u043e\u0432\u0442\u043e\u0440\u0438\u0442\u044c         raise FetchError(f\"server_error={r.status_code}\")     r.raise_for_status()     return r.json()  def main() -&gt; int:     structlog.configure(processors=[structlog.processors.add_log_level, structlog.processors.JSONRenderer()])     with httpx.Client(timeout=TIMEOUT, headers={\"User-Agent\": \"fetch-users\/1.0\"}) as client:         t0 = time.perf_counter()         data = fetch_json(client, API_URL)         dt = time.perf_counter() - t0         log.info(\"fetched\", bytes=len(json.dumps(data).encode(\"utf-8\")), seconds=round(dt, 3))         print(json.dumps(data, ensure_ascii=False))     return 0  if __name__ == \"__main__\":     try:         sys.exit(main())     except httpx.RequestError as e:         log.error(\"http_error\", error=str(e))         sys.exit(2)     except Exception as e:         log.error(\"unexpected\", error=str(e.__class__.__name__))         sys.exit(3)<\/code><\/pre>\n<p>\u0417\u0430\u043f\u0443\u0441\u043a:<\/p>\n<pre><code class=\"bash\">uv run fetch_users.py # \u0437\u0430\u043a\u0440\u0435\u043f\u043b\u044f\u0435\u043c \u0432\u0435\u0440\u0441\u0438\u0438 \u0440\u044f\u0434\u043e\u043c \u0441\u043e \u0441\u043a\u0440\u0438\u043f\u0442\u043e\u043c uv lock --script fetch_users.py # \u044d\u043a\u0441\u043f\u043e\u0440\u0442 \u0434\u043b\u044f CI uv export --script fetch_users.py -o requirements.txt uvx pip-audit -r requirements.txt<\/code><\/pre>\n<p><code>uv<\/code> \u0437\u0430\u0434\u043e\u043a\u0443\u043c\u0435\u043d\u0442\u0438\u0440\u043e\u0432\u0430\u043b \u0438 lock \u0434\u043b\u044f\u00a0\u0441\u043a\u0440\u0438\u043f\u0442\u043e\u0432, \u0438 \u044d\u043a\u0441\u043f\u043e\u0440\u0442. <\/p>\n<h3>\u0418\u043d\u0441\u043f\u0435\u043a\u0446\u0438\u044f \u0434\u0435\u0440\u0435\u0432\u0430 \u0437\u0430\u0432\u0438\u0441\u0438\u043c\u043e\u0441\u0442\u0435\u0439 \u0438 \u043f\u043e\u043b\u0438\u0442\u0438\u043a\u0430 \u0438\u0441\u0442\u043e\u0447\u043d\u0438\u043a\u043e\u0432<\/h3>\n<p>\u041f\u0435\u0440\u0435\u0434 \u0432\u044b\u043a\u043b\u0430\u0434\u043a\u043e\u0439 \u0432\u00a0\u0445\u0443\u043a pre\u2011commit \u043c\u043e\u0436\u043d\u043e \u043e\u0431\u043e\u0437\u0440\u0435\u0442\u044c \u0434\u0435\u0440\u0435\u0432\u043e:<\/p>\n<pre><code class=\"bash\">uv tree --script fetch_users.py<\/code><\/pre>\n<p>\u0414\u043b\u044f\u00a0\u0447\u0430\u0441\u0442\u043d\u044b\u0445 \u0438\u043d\u0434\u0435\u043a\u0441\u043e\u0432 \u0438 \u0437\u0435\u0440\u043a\u0430\u043b \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u0435\u043c \u043a\u043e\u043d\u0444\u0438\u0433\u0443\u0440\u0430\u0446\u0438\u043e\u043d\u043d\u044b\u0435 \u0444\u0430\u0439\u043b\u044b <code>uv.toml<\/code> \u0438\u043b\u0438 <code>pyproject.toml<\/code> \u0434\u043b\u044f <code>uv pip<\/code> \u0438 \u043d\u0435\u00a0\u0440\u0430\u0437\u0434\u0430\u0435\u043c <code>--allow-insecure-host<\/code>. \u041a\u043e\u043d\u0444\u0438\u0433\u0438 \u043f\u043e\u0434\u0434\u0435\u0440\u0436\u0438\u0432\u0430\u044e\u0442\u0441\u044f \u043d\u0430\u00a0\u0443\u0440\u043e\u0432\u043d\u0435 \u043f\u0440\u043e\u0435\u043a\u0442\u0430 \u0438 \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044f; \u0435\u0441\u0442\u044c \u0438 \u0441\u0438\u0441\u0442\u0435\u043c\u043d\u044b\u0439 \u0443\u0440\u043e\u0432\u0435\u043d\u044c. \u041f\u0443\u0442\u044c \u043a\u00a0\u043a\u0435\u0448\u0443 \u0438 \u043f\u0430\u0440\u0430\u043c\u0435\u0442\u0440\u044b \u043c\u043e\u0436\u043d\u043e \u043a\u043e\u043d\u0442\u0440\u043e\u043b\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u0447\u0435\u0440\u0435\u0437 \u043a\u043e\u043c\u0430\u043d\u0434\u044b \u0438 \u043f\u0435\u0440\u0435\u043c\u0435\u043d\u043d\u044b\u0435 \u043e\u043a\u0440\u0443\u0436\u0435\u043d\u0438\u044f. <\/p>\n<hr\/>\n<h3>\u0418\u0442\u043e\u0433<\/h3>\n<p>PEP 723\u00a0\u0440\u0435\u0448\u0430\u0435\u0442 \u0440\u0443\u0442\u0438\u043d\u0443: \u043e\u0434\u043d\u043e\u0444\u0430\u0439\u043b\u043e\u0432\u044b\u0435 \u0441\u043a\u0440\u0438\u043f\u0442\u044b \u043c\u043e\u0433\u0443\u0442\u00a0\u0431\u044b\u0442\u044c \u0441\u0430\u043c\u043e\u0434\u043e\u0441\u0442\u0430\u0442\u043e\u0447\u043d\u044b\u043c\u0438, \u0431\u0435\u0437\u00a0\u0432\u043e\u043f\u0440\u043e\u0441\u043e\u0432 \u043e\u00a0\u0442\u043e\u043c, \u043a\u0430\u043a\u00a0\u0443\u0441\u0442\u0430\u043d\u043e\u0432\u0438\u0442\u044c \u0437\u0430\u0432\u0438\u0441\u0438\u043c\u043e\u0441\u0442\u0438. <code>uv<\/code> \u0441\u0434\u0435\u043b\u0430\u043b \u044d\u0442\u043e\u00a0\u0431\u044b\u0441\u0442\u0440\u044b\u043c \u0438 \u043e\u043f\u0435\u0440\u0430\u0446\u0438\u043e\u043d\u043d\u043e \u0443\u0434\u043e\u0431\u043d\u044b\u043c: \u0440\u0435\u0434\u0430\u043a\u0442\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u0435 \u0437\u0430\u0432\u0438\u0441\u0438\u043c\u043e\u0441\u0442\u0435\u0439, shebang, lock \u0440\u044f\u0434\u043e\u043c \u0441\u043e \u0441\u043a\u0440\u0438\u043f\u0442\u043e\u043c, \u044d\u043a\u0441\u043f\u043e\u0440\u0442 \u0434\u043b\u044f\u00a0CI, \u043a\u043e\u043d\u0442\u0440\u043e\u043b\u044c \u0432\u0435\u0440\u0441\u0438\u0438 \u0438\u043d\u0442\u0435\u0440\u043f\u0440\u0435\u0442\u0430\u0442\u043e\u0440\u0430. <\/p>\n<blockquote>\n<p>\u041a\u0430\u043a \u0438 \u0432\u00a0\u0441\u043b\u0443\u0447\u0430\u0435 \u0441\u00a0\u043e\u0434\u043d\u043e\u0444\u0430\u0439\u043b\u043e\u0432\u044b\u043c\u0438 \u0441\u043a\u0440\u0438\u043f\u0442\u0430\u043c\u0438 \u043f\u043e\u00a0PEP 723\u00a0\u0438 uv, \u043a\u043e\u0442\u043e\u0440\u044b\u0435 \u043f\u043e\u0437\u0432\u043e\u043b\u044f\u044e\u0442\u00a0\u0431\u044b\u0441\u0442\u0440\u043e \u043f\u0440\u043e\u0432\u0435\u0440\u0438\u0442\u044c \u0438\u0434\u0435\u044e \u0431\u0435\u0437\u00a0\u043b\u0438\u0448\u043d\u0438\u0445 \u043f\u043e\u0434\u0433\u043e\u0442\u043e\u0432\u0438\u0442\u0435\u043b\u044c\u043d\u044b\u0445 \u0448\u0430\u0433\u043e\u0432, \u0443\u00a0\u0432\u0430\u0441 \u0435\u0441\u0442\u044c \u0432\u043e\u0437\u043c\u043e\u0436\u043d\u043e\u0441\u0442\u044c \u0442\u0430\u043a\u00a0\u0436\u0435 \u043f\u043e\u0437\u043d\u0430\u043a\u043e\u043c\u0438\u0442\u044c\u0441\u044f \u0441\u00a0\u043a\u0443\u0440\u0441\u043e\u043c Python Developer. Professional\u00a0\u2014 \u0447\u0435\u0440\u0435\u0437 <strong>\u0431\u0435\u0441\u043f\u043b\u0430\u0442\u043d\u044b\u0435<\/strong> <a href=\"https:\/\/otus.pw\/Chl4o\/\" rel=\"noopener noreferrer nofollow\">\u043e\u0442\u043a\u0440\u044b\u0442\u044b\u0435 \u0443\u0440\u043e\u043a\u0438<\/a>, \u0434\u043e\u0441\u0442\u0443\u043f\u043d\u044b\u0435 \u043f\u043e\u00a0\u0441\u0441\u044b\u043b\u043a\u0435.<\/p>\n<p>\u041a\u0440\u043e\u043c\u0435 \u0442\u043e\u0433\u043e, \u0432\u044b \u043c\u043e\u0436\u0435\u0442\u0435 \u043f\u0440\u043e\u0439\u0442\u0438 <strong>\u0431\u0435\u0441\u043f\u043b\u0430\u0442\u043d\u043e\u0435<\/strong> <a href=\"https:\/\/otus.pw\/5ofm\/\" rel=\"noopener noreferrer nofollow\">\u0432\u0441\u0442\u0443\u043f\u0438\u0442\u0435\u043b\u044c\u043d\u043e\u0435 \u0442\u0435\u0441\u0442\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u0435<\/a>, \u043a\u043e\u0442\u043e\u0440\u043e\u0435 \u043f\u043e\u0437\u0432\u043e\u043b\u0438\u0442 \u043e\u0446\u0435\u043d\u0438\u0442\u044c \u0432\u0430\u0448\u0438 \u0437\u043d\u0430\u043d\u0438\u044f \u0438 \u043d\u0430\u0432\u044b\u043a\u0438.<\/p>\n<p>\u0410\u00a0\u0435\u0441\u043b\u0438 \u0445\u043e\u0442\u0438\u0442\u0435 \u0443\u0437\u043d\u0430\u0442\u044c \u0431\u043e\u043b\u044c\u0448\u0435 \u043e\u00a0\u0441\u0430\u043c\u043e\u043c \u043a\u0443\u0440\u0441\u0435 \u0438 \u0432\u043f\u0435\u0447\u0430\u0442\u043b\u0435\u043d\u0438\u044f\u0445 \u0443\u0447\u0430\u0441\u0442\u043d\u0438\u043a\u043e\u0432, \u0437\u0430\u0433\u043b\u044f\u043d\u0438\u0442\u0435 \u0432 <a href=\"https:\/\/otus.pw\/2zTI\/\" rel=\"noopener noreferrer nofollow\">\u0441\u0435\u043a\u0446\u0438\u044e \u0441\u00a0\u043e\u0442\u0437\u044b\u0432\u0430\u043c\u0438<\/a>.<\/p>\n<\/blockquote>\n<\/div>\n<\/div>\n<\/div>\n<p><!----><!----><\/div>\n<p><!----><!----><br \/> \u0441\u0441\u044b\u043b\u043a\u0430 \u043d\u0430 \u043e\u0440\u0438\u0433\u0438\u043d\u0430\u043b \u0441\u0442\u0430\u0442\u044c\u0438 <a href=\"https:\/\/habr.com\/ru\/articles\/940674\/\"> https:\/\/habr.com\/ru\/articles\/940674\/<\/a><\/p>\n","protected":false},"excerpt":{"rendered":"<div><!--[--><!--]--><\/div>\n<div id=\"post-content-body\">\n<div>\n<div class=\"article-formatted-body article-formatted-body article-formatted-body_version-2\">\n<div xmlns=\"http:\/\/www.w3.org\/1999\/xhtml\">\n<p>\u041f\u0440\u0438\u0432\u0435\u0442, \u0425\u0430\u0431\u0440!<\/p>\n<p>\u0415\u0441\u043b\u0438 \u0432\u044b \u0441\u043e\u0431\u0438\u0440\u0430\u0435\u0442\u0435 \u043f\u0440\u043e\u0442\u043e\u0442\u0438\u043f \u043d\u0430\u00a0C++, \u0442\u043e \u043e\u0434\u0438\u043d \u0444\u0430\u0439\u043b \u0441 <code>main.cpp<\/code> \u0438\u043d\u043e\u0433\u0434\u0430 \u0440\u0435\u0430\u043b\u044c\u043d\u043e \u043a\u043e\u043c\u043f\u0438\u043b\u0438\u0440\u0443\u0435\u0442\u0441\u044f \u0432\u00a0\u0440\u0430\u0431\u043e\u0447\u0443\u044e \u0443\u0442\u0438\u043b\u0438\u0442\u0443. \u0411\u0438\u0431\u043b\u0438\u043e\u0442\u0435\u043a\u0438\u00a0\u043b\u0438\u0431\u043e \u0437\u0430\u0432\u043e\u0437\u044f\u0442\u0441\u044f \u043f\u0430\u043a\u0435\u0442\u043d\u044b\u043c \u043c\u0435\u043d\u0435\u0434\u0436\u0435\u0440\u043e\u043c \u0437\u0430\u0440\u0430\u043d\u0435\u0435,\u00a0\u043b\u0438\u0431\u043e \u0443\u00a0\u0432\u0430\u0441 \u0435\u0441\u0442\u044c header\u2011only \u0437\u0430\u0432\u0438\u0441\u0438\u043c\u043e\u0441\u0442\u044c \u0438 \u0432\u0441\u0451 \u0432\u0437\u043b\u0435\u0442\u0430\u0435\u0442. \u0412\u00a0Python \u0434\u043e\u043b\u0433\u043e\u0435 \u0432\u0440\u0435\u043c\u044f \u044d\u0442\u043e\u00a0\u0431\u044b\u043b\u043e \u0431\u043e\u043b\u044c\u044e: \u043b\u044e\u0431\u043e\u0439 \u043e\u0434\u043d\u043e\u0444\u0430\u0439\u043b\u043e\u0432\u044b\u0439 \u0441\u043a\u0440\u0438\u043f\u0442, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u0442\u0440\u0435\u0431\u0443\u0435\u0442 <code>requests<\/code> \u0438\u043b\u0438 <code>rich<\/code>, \u0443\u0436\u0435 \u0442\u044f\u043d\u0435\u0442 \u0437\u0430\u00a0\u0441\u043e\u0431\u043e\u0439 \u0432\u0438\u0440\u0442\u0443\u0430\u043b\u044c\u043d\u044b\u0435 \u043e\u043a\u0440\u0443\u0436\u0435\u043d\u0438\u044f, \u0438\u043d\u0441\u0442\u0440\u0443\u043a\u0446\u0438\u0438 \u0432\u00a0README \u0438 \u043b\u043e\u043a\u0430\u043b\u044c\u043d\u044b\u0435 \u0444\u0438\u0447\u0438. <\/p>\n<p>\u0415\u0441\u0442\u044c \u0440\u0430\u0431\u043e\u0447\u0438\u0439 \u0441\u0442\u0430\u043d\u0434\u0430\u0440\u0442 \u0434\u043b\u044f\u00a0\u043d\u043e\u0440\u043c\u0430\u043b\u044c\u043d\u044b\u0445 \u043e\u0434\u043d\u043e\u0444\u0430\u0439\u043b\u043e\u0432\u044b\u0445 \u0441\u0446\u0435\u043d\u0430\u0440\u0438\u0435\u0432 \u0441\u00a0\u0437\u0430\u0432\u0438\u0441\u0438\u043c\u043e\u0441\u0442\u044f\u043c\u0438\u00a0\u2014 PEP 723: \u0432\u044b \u043e\u0431\u044a\u044f\u0432\u043b\u044f\u0435\u0442\u0435 \u0437\u0430\u0432\u0438\u0441\u0438\u043c\u043e\u0441\u0442\u0438 \u043f\u0440\u044f\u043c\u043e \u0432\u00a0\u043a\u043e\u043c\u043c\u0435\u043d\u0442\u0430\u0440\u0438\u044f\u0445, \u0430\u00a0\u0440\u0430\u043d\u043d\u0435\u0440 \u0441\u0442\u0430\u0432\u0438\u0442 \u0432\u0441\u0451 \u0441\u0430\u043c \u0438 \u0437\u0430\u043f\u0443\u0441\u043a\u0430\u0435\u0442 \u0432\u00a0\u0438\u0437\u043e\u043b\u0438\u0440\u043e\u0432\u0430\u043d\u043d\u043e\u0439 \u0441\u0440\u0435\u0434\u0435. \u0412\u00a0\u0441\u0432\u044f\u0437\u043a\u0435 \u0441 <code>uv<\/code> \u043f\u043e\u043b\u0443\u0447\u0430\u0435\u0442\u0441\u044f \u043d\u0435\u043f\u043b\u043e\u0445\u043e\u0439 \u0442\u0430\u043a\u043e\u0439 \u0441\u043f\u043e\u0441\u043e\u0431 \u0434\u0435\u043b\u0438\u0442\u044c\u0441\u044f \u0441\u043a\u0440\u0438\u043f\u0442\u0430\u043c\u0438, \u0432\u00a0\u0442\u043e\u043c \u0447\u0438\u0441\u043b\u0435 \u0434\u043b\u044f\u00a0\u043f\u0432\u0441\u043f\u043e\u043c\u043e\u0433\u0430\u0442\u0435\u043b\u044c\u043d\u044b\u0445 \u0437\u0430\u0434\u0430\u0447. \u0418 \u0434\u0430, \u0443\u00a0\u044d\u0442\u043e\u0439 \u043a\u0440\u0430\u0441\u043e\u0442\u044b \u0435\u0441\u0442\u044c \u043d\u044e\u0430\u043d\u0441\u044b \u0431\u0435\u0437\u043e\u043f\u0430\u0441\u043d\u043e\u0441\u0442\u0438, \u043e\u00a0\u043d\u0438\u0445 \u043f\u043e\u0433\u043e\u0432\u043e\u0440\u0438\u043c \u043e\u0442\u0434\u0435\u043b\u044c\u043d\u043e.<\/p>\n<h3>\u0427\u0442\u043e \u0442\u0430\u043a\u043e\u0435 PEP 723 \u0438 \u043a\u0430\u043a \u044d\u0442\u043e \u0432\u044b\u0433\u043b\u044f\u0434\u0438\u0442 <\/h3>\n<p>PEP 723\u00a0\u043e\u043f\u0440\u0435\u0434\u0435\u043b\u044f\u0435\u0442 \u0431\u043b\u043e\u043a \u043c\u0435\u0442\u0430\u0434\u0430\u043d\u043d\u044b\u0445 \u0432\u00a0\u043a\u043e\u043c\u043c\u0435\u043d\u0442\u0430\u0440\u0438\u044f\u0445, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u043f\u0430\u0440\u0441\u0438\u0442\u0441\u044f \u0432\u043d\u0435\u0448\u043d\u0438\u043c\u0438 \u0438\u043d\u0441\u0442\u0440\u0443\u043c\u0435\u043d\u0442\u0430\u043c\u0438. \u0424\u043e\u0440\u043c\u0430\u0442 \u043f\u0440\u043e\u0441\u0442: \u0441\u0432\u0435\u0440\u0445\u0443 \u0438 \u0441\u043d\u0438\u0437\u0443 \u043c\u0430\u0440\u043a\u0435\u0440\u044b, \u0432\u043d\u0443\u0442\u0440\u0438 TOML c \u043f\u043e\u043b\u044f\u043c\u0438 <code>dependencies<\/code> \u0438 <code>requires-python<\/code>. <\/p>\n<p>\u041f\u0440\u0438\u043c\u0435\u0440 \u043c\u0438\u043d\u0438\u043c\u0430\u043b\u044c\u043d\u043e\u0433\u043e \u0441\u043a\u0440\u0438\u043f\u0442\u0430:<\/p>\n<pre><code class=\"yaml\"># \/\/\/ script # requires-python = \"&gt;=3.11\" # dependencies = [ #   \"httpx&lt;1.0\", #   \"rich&gt;=13.7\", # ] # \/\/\/ import httpx from rich import print  def main() -&gt; None:     r = httpx.get(\"https:\/\/httpbin.org\/json\", timeout=10)     r.raise_for_status()     print({\"status\": r.status_code, \"len\": len(r.text)})  if __name__ == \"__main__\":     main()<\/code><\/pre>\n<p>\u0417\u0430\u043f\u0443\u0441\u043a \u0447\u0435\u0440\u0435\u0437 <code>uv<\/code>:<\/p>\n<pre><code class=\"bash\">uv run example.py<\/code><\/pre>\n<p><code>uv<\/code> \u0441\u043e\u0437\u0434\u0430\u0441\u0442 \u0438\u0437\u043e\u043b\u0438\u0440\u043e\u0432\u0430\u043d\u043d\u0443\u044e \u0441\u0440\u0435\u0434\u0443 \u0432\u00a0\u043a\u0435\u0448\u0435, \u0443\u0441\u0442\u0430\u043d\u043e\u0432\u0438\u0442 \u0437\u0430\u0432\u0438\u0441\u0438\u043c\u043e\u0441\u0442\u0438 \u0438 \u0432\u044b\u043f\u043e\u043b\u043d\u0438\u0442 \u043a\u043e\u0434. \u041d\u0438\u043a\u0430\u043a\u0438\u0445 \u0440\u0443\u0447\u043d\u044b\u0445 <code>venv<\/code>.<\/p>\n<h3>\u0414\u043e\u0431\u0430\u0432\u043b\u044f\u0435\u043c \u0438 \u0443\u043f\u0440\u0430\u0432\u043b\u044f\u0435\u043c \u0437\u0430\u0432\u0438\u0441\u0438\u043c\u043e\u0441\u0442\u044f\u043c\u0438 \u043f\u0440\u044f\u043c\u043e \u0438\u0437 CLI<\/h3>\n<p>PEP 723\u00a0\u2014 \u044d\u0442\u043e \u043f\u0440\u043e\u00a0\u0444\u043e\u0440\u043c\u0430\u0442, \u043d\u043e\u00a0\u043f\u0440\u0430\u0432\u0438\u0442\u044c TOML \u0440\u0443\u043a\u0430\u043c\u0438\u00a0\u0431\u044b\u0441\u0442\u0440\u043e \u043d\u0430\u0434\u043e\u0435\u0434\u0430\u0435\u0442. <code>uv<\/code> \u0443\u043c\u0435\u0435\u0442 \u0440\u0435\u0434\u0430\u043a\u0442\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u0431\u043b\u043e\u043a \u0437\u0430\u0432\u0438\u0441\u0438\u043c\u043e\u0441\u0442\u0435\u0439 \u0432\u00a0\u0444\u0430\u0439\u043b\u0435:<\/p>\n<pre><code class=\"bash\">uv add --script example.py \"pydantic&gt;=2.8\" uv remove --script example.py \"httpx\" uv run example.py<\/code><\/pre>\n<p>\u0415\u0441\u043b\u0438 \u043d\u0443\u0436\u0435\u043d \u0438\u0441\u043f\u043e\u043b\u043d\u044f\u0435\u043c\u044b\u0439 \u0444\u0430\u0439\u043b \u0431\u0435\u0437\u00a0\u044f\u0432\u043d\u043e\u0433\u043e \u0432\u044b\u0437\u043e\u0432\u0430 <code>uv run<\/code>, \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u0435\u043c shebang:<\/p>\n<pre><code class=\"bash\">#!\/usr\/bin\/env -S uv run --script # \/\/\/ script # dependencies = [\"rich\"] # \/\/\/ from rich import print print(\"hello\")<\/code><\/pre>\n<p>\u0421\u0442\u0430\u0432\u0438\u043c \u043f\u0440\u0430\u0432\u0430 \u0438 \u0437\u0430\u043f\u0443\u0441\u043a\u0430\u0435\u043c \u0438\u0437\u00a0\u043a\u0430\u0442\u0430\u043b\u043e\u0433\u0430:<\/p>\n<pre><code class=\"bash\">chmod +x greet .\/greet<\/code><\/pre>\n<h3>\u041b\u043e\u0447\u0438\u043c \u0437\u0430\u0432\u0438\u0441\u0438\u043c\u043e\u0441\u0442\u0438 \u0438 \u0434\u0435\u043b\u0430\u0435\u043c \u0441\u043a\u0440\u0438\u043f\u0442 \u0432\u043e\u0441\u043f\u0440\u043e\u0438\u0437\u0432\u043e\u0434\u0438\u043c\u044b\u043c<\/h3>\n<p>\u041e\u0434\u043d\u043e \u0434\u0435\u043b\u043e \u0443\u0441\u0442\u0430\u043d\u043e\u0432\u0438\u0442\u044c \u0430\u043a\u0442\u0443\u0430\u043b\u044c\u043d\u044b\u0435 \u0432\u0435\u0440\u0441\u0438\u0438, \u0434\u0440\u0443\u0433\u043e\u0435\u00a0\u2014 \u0437\u0430\u0444\u0438\u043a\u0441\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u0438\u0445. <code>uv<\/code> \u043f\u043e\u0434\u0434\u0435\u0440\u0436\u0438\u0432\u0430\u0435\u0442 lock\u2011\u0444\u0430\u0439\u043b \u0434\u043b\u044f\u00a0\u0441\u043a\u0440\u0438\u043f\u0442\u043e\u0432. <\/p>\n<pre><code class=\"bash\">uv lock --script example.py # \u043f\u043e\u044f\u0432\u0438\u0442\u0441\u044f example.py.lock  # \u0442\u0435\u043f\u0435\u0440\u044c \u043b\u044e\u0431\u044b\u0435 \u043e\u043f\u0435\u0440\u0430\u0446\u0438\u0438 \u0431\u0443\u0434\u0443\u0442 \u0443\u0447\u0438\u0442\u044b\u0432\u0430\u0442\u044c lock: uv run --script example.py uv add --script example.py \"rich\"      # \u043e\u0431\u043d\u043e\u0432\u0438\u0442 lock uv export --script example.py -o req.txt<\/code><\/pre>\n<p>\u041c\u043e\u0436\u043d\u043e \u043e\u0433\u0440\u0430\u043d\u0438\u0447\u0438\u0442\u044c \u0441\u0432\u0435\u0436\u0435\u0441\u0442\u044c \u043f\u0430\u043a\u0435\u0442\u043e\u0432 \u0434\u0430\u0442\u043e\u0439, \u0447\u0442\u043e\u0431\u044b \u0438\u0437\u0431\u0435\u0436\u0430\u0442\u044c \u043d\u0435\u043e\u0436\u0438\u0434\u0430\u043d\u043d\u044b\u0445 \u0430\u043f\u0434\u0435\u0439\u0442\u043e\u0432 \u0447\u0435\u0440\u0435\u0437 \u0433\u043e\u0434. \u0412\u00a0\u0431\u043b\u043e\u043a\u0435 <code>[tool.uv]<\/code> \u0432\u043d\u0443\u0442\u0440\u0438 \u043c\u0435\u0442\u0430\u0434\u0430\u043d\u043d\u044b\u0445:<\/p>\n<pre><code class=\"python\"># \/\/\/ script # requires-python = \"&gt;=3.11\" # dependencies = [\"httpx&lt;1.0\"] # [tool.uv] # exclude-newer = \"2025-08-01T00:00:00Z\" # \/\/\/<\/code><\/pre>\n<p><code>exclude-newer<\/code> \u0437\u0430\u0441\u0442\u0430\u0432\u0438\u0442 \u0440\u0435\u0437\u043e\u043b\u0432\u0435\u0440 \u0438\u0433\u043d\u043e\u0440\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u0440\u0435\u043b\u0438\u0437\u044b, \u0432\u044b\u0448\u0435\u0434\u0448\u0438\u0435 \u043f\u043e\u0441\u043b\u0435 \u0443\u043a\u0430\u0437\u0430\u043d\u043d\u043e\u0439 \u0434\u0430\u0442\u044b. <\/p>\n<p>\u0410\u00a0\u0435\u0441\u043b\u0438 \u043d\u0443\u0436\u043d\u043e \u0438\u043c\u0435\u043d\u043d\u043e pip\u2011\u0441\u043e\u0432\u043c\u0435\u0441\u0442\u0438\u043c\u043e\u0435 \u0437\u0430\u0444\u0438\u043a\u0441\u0438\u0440\u043e\u0432\u0430\u043d\u043d\u043e\u0435 \u0434\u0435\u0440\u0435\u0432\u043e \u0434\u043b\u044f\u00a0\u0434\u043b\u0438\u0442\u0435\u043b\u044c\u043d\u044b\u0445 CI\u2011\u043f\u0430\u0439\u043f\u043b\u0430\u0439\u043d\u043e\u0432, \u0443 <code>uv<\/code> \u0435\u0441\u0442\u044c \u0438\u043d\u0441\u0442\u0440\u0443\u043c\u0435\u043d\u0442\u044b \u0443\u0440\u043e\u0432\u043d\u044f <code>pip-tools<\/code>: <code>uv pip compile<\/code> \u0438 <code>uv pip sync<\/code>. \u041e\u043d\u0438 \u043f\u043e\u0437\u0432\u043e\u043b\u044f\u044e\u0442 \u0441\u0433\u0435\u043d\u0435\u0440\u0438\u0440\u043e\u0432\u0430\u0442\u044c <code>requirements.txt<\/code> \u0438\u0437\u00a0\u0438\u0441\u0445\u043e\u0434\u043d\u0438\u043a\u0430\u2011\u0434\u0435\u043a\u043b\u0430\u0440\u0430\u0446\u0438\u0438 \u0438 \u0441\u0438\u043d\u0445\u0440\u043e\u043d\u0438\u0437\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u043e\u043a\u0440\u0443\u0436\u0435\u043d\u0438\u0435 \u043e\u0434\u0438\u043d\u2011\u0432\u2011\u043e\u0434\u0438\u043d. <\/p>\n<h3>\u041a\u043e\u043d\u0442\u0440\u043e\u043b\u044c \u0432\u0435\u0440\u0441\u0438\u0438 Python \u043d\u0430 \u0440\u0430\u043d\u0442\u0430\u0439\u043c\u0435<\/h3>\n<p>\u0421\u043a\u0440\u0438\u043f\u0442 \u043c\u043e\u0436\u0435\u0442 \u0442\u0440\u0435\u0431\u043e\u0432\u0430\u0442\u044c \u043a\u043e\u043d\u043a\u0440\u0435\u0442\u043d\u0443\u044e \u0432\u0435\u0442\u043a\u0443 \u0438\u043d\u0442\u0435\u0440\u043f\u0440\u0435\u0442\u0430\u0442\u043e\u0440\u0430. \u0412\u00a0\u044d\u0442\u043e\u043c \u0441\u043b\u0443\u0447\u0430\u0435 <code>uv<\/code> \u043f\u043e\u0437\u0432\u043e\u043b\u044f\u0435\u0442 \u0437\u0430\u043f\u0440\u043e\u0441\u0438\u0442\u044c \u043d\u0443\u0436\u043d\u0443\u044e \u0432\u0435\u0440\u0441\u0438\u044e \u043f\u0440\u0438\u00a0\u0437\u0430\u043f\u0443\u0441\u043a\u0435:<\/p>\n<pre><code class=\"bash\">uv run --python 3.10 example.py uv run --python 3.12 example.py<\/code><\/pre>\n<h3>\u0411\u0435\u0437\u043e\u043f\u0430\u0441\u043d\u043e\u0441\u0442\u044c<\/h3>\n<p>\u041e\u0434\u043d\u043e\u0444\u0430\u0439\u043b\u043e\u0432\u044b\u0435 \u0441\u043a\u0440\u0438\u043f\u0442\u044b \u0443\u0434\u043e\u0431\u043d\u043e \u0437\u0430\u043f\u0443\u0441\u043a\u0430\u0442\u044c \u043a\u0430\u043a\u00a0\u0435\u0441\u0442\u044c, \u0438 \u0438\u043c\u0435\u043d\u043d\u043e \u043f\u043e\u044d\u0442\u043e\u043c\u0443 \u043d\u0430\u0434\u043e\u00a0\u0431\u044b\u0442\u044c \u0430\u043a\u043a\u0443\u0440\u0430\u0442\u043d\u0435\u0435 \u043e\u0431\u044b\u0447\u043d\u043e\u0433\u043e. \u0420\u0438\u0441\u043a\u0438:<\/p>\n<ol>\n<li>\n<p>\u041f\u043e\u0434\u043c\u0435\u043d\u0430 \u0437\u0430\u0432\u0438\u0441\u0438\u043c\u043e\u0441\u0442\u0435\u0439. \u0422\u0440\u0438\u0432\u0438\u0430\u043b\u044c\u043d\u044b\u0435 \u0430\u0442\u0430\u043a\u0438 \u043d\u0430\u00a0\u0446\u0435\u043f\u043e\u0447\u043a\u0443 \u043f\u043e\u0441\u0442\u0430\u0432\u043e\u043a \u0447\u0435\u0440\u0435\u0437 \u0437\u0430\u0440\u0435\u0433\u0438\u0441\u0442\u0440\u0438\u0440\u043e\u0432\u0430\u043d\u043d\u044b\u0435 \u043f\u0430\u043a\u0435\u0442\u044b \u0441\u00a0\u043f\u043e\u0445\u043e\u0436\u0438\u043c\u0438 \u0438\u043c\u0435\u043d\u0430\u043c\u0438.<\/p>\n<\/li>\n<li>\n<p>\u041d\u0435\u043f\u0440\u0435\u0434\u0441\u043a\u0430\u0437\u0443\u0435\u043c\u044b\u0435 \u0430\u043f\u0433\u0440\u0435\u0439\u0434\u044b. \u0421\u0435\u0433\u043e\u0434\u043d\u044f \u0432\u0441\u0451 \u0447\u0438\u0441\u0442\u043e, \u0437\u0430\u0432\u0442\u0440\u0430 \u043d\u043e\u0432\u044b\u0439 \u0440\u0435\u043b\u0438\u0437 \u0442\u0440\u0430\u043d\u0437\u0438\u0442\u0438\u0432\u043d\u043e\u0439 \u0437\u0430\u0432\u0438\u0441\u0438\u043c\u043e\u0441\u0442\u0438 \u043b\u043e\u043c\u0430\u0435\u0442 \u0438\u043d\u0432\u0430\u0440\u0438\u0430\u043d\u0442\u044b.<\/p>\n<\/li>\n<li>\n<p>\u0418\u0441\u0442\u043e\u0447\u043d\u0438\u043a\u0438 \u043f\u0430\u043a\u0435\u0442\u043e\u0432. \u041f\u0435\u0440\u0435\u043e\u043f\u0440\u0435\u0434\u0435\u043b\u0435\u043d\u0438\u0435 \u0438\u043d\u0434\u0435\u043a\u0441\u043e\u0432, \u043e\u0442\u043a\u043b\u044e\u0447\u0435\u043d\u0438\u0435 TLS\u2011\u043f\u0440\u043e\u0432\u0435\u0440\u043e\u043a, \u0432\u043d\u0443\u0442\u0440\u0435\u043d\u043d\u0438\u0435 \u0437\u0435\u0440\u043a\u0430\u043b\u0430 \u0431\u0435\u0437\u00a0\u043f\u043e\u043b\u0438\u0442\u0438\u043a\u0438.<\/p>\n<\/li>\n<li>\n<p>\u0423\u044f\u0437\u0432\u0438\u043c\u043e\u0441\u0442\u0438 \u0432\u00a0\u0443\u0436\u0435 \u0432\u044b\u0431\u0440\u0430\u043d\u043d\u044b\u0445 \u0432\u0435\u0440\u0441\u0438\u044f\u0445.<\/p>\n<\/li>\n<\/ol>\n<p>\u041a\u043e\u043d\u043a\u0440\u0435\u0442\u043d\u044b\u0435 \u043c\u0435\u0440\u044b:<\/p>\n<p><strong>\u0424\u0438\u043a\u0441\u0430\u0446\u0438\u044f \u0438 \u0430\u0443\u0434\u0438\u0442. <\/strong>\u0414\u043b\u044f\u00a0\u0441\u043a\u0440\u0438\u043f\u0442\u043e\u0432 \u0441\u043e\u0437\u0434\u0430\u0451\u043c <code>example.py.lock<\/code> \u0438 \u0440\u0435\u0433\u0443\u043b\u044f\u0440\u043d\u043e \u043f\u0440\u043e\u0433\u043e\u043d\u044f\u0435\u043c \u0430\u0443\u0434\u0438\u0442 \u0437\u0430\u0432\u0438\u0441\u0438\u043c\u043e\u0441\u0442\u0435\u0439. \u0412\u00a0\u044d\u043a\u043e\u0441\u0438\u0441\u0442\u0435\u043c\u0435 PyPA \u0434\u043b\u044f\u00a0\u044d\u0442\u043e\u0433\u043e \u0435\u0441\u0442\u044c <code>pip-audit<\/code>, \u043e\u043d \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u0435\u0442 \u0431\u0430\u0437\u0443 \u0443\u044f\u0437\u0432\u0438\u043c\u043e\u0441\u0442\u0435\u0439 PyPI \u0438 \u043c\u043e\u0436\u0435\u0442 \u0440\u0430\u0431\u043e\u0442\u0430\u0442\u044c \u043f\u043e\u0432\u0435\u0440\u0445 \u0432\u0430\u0448\u0435\u0433\u043e requirements \u0438\u043b\u0438\u00a0\u0443\u0441\u0442\u0430\u043d\u043e\u0432\u043b\u0435\u043d\u043d\u043e\u0439 \u0441\u0440\u0435\u0434\u044b. \u041c\u043e\u0436\u043d\u043e \u0437\u0430\u043f\u0443\u0441\u043a\u0430\u0442\u044c \u0447\u0435\u0440\u0435\u0437 <code>uvx<\/code> \u0432\u00a0\u0438\u0437\u043e\u043b\u0438\u0440\u043e\u0432\u0430\u043d\u043d\u043e\u043c \u043e\u043a\u0440\u0443\u0436\u0435\u043d\u0438\u0438:<\/p>\n<pre><code class=\"bash\">uvx pip-audit -r req.txt # \u0438\u043b\u0438 \u043f\u0440\u043e\u0441\u043a\u0430\u043d\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u043b\u043e\u043a\u0430\u043b\u044c\u043d\u0443\u044e \u0441\u0440\u0435\u0434\u0443 uvx pip-audit --local<\/code><\/pre>\n<p>\u041f\u0440\u043e\u0435\u043a\u0442 \u043f\u043e\u0434\u0434\u0435\u0440\u0436\u0438\u0432\u0430\u0435\u0442\u0441\u044f PyPA, \u0435\u0441\u0442\u044c GitHub Action. <\/p>\n<p><strong>\u0425\u0435\u0448\u0438 \u0438 \u0441\u0438\u043d\u0445\u0440\u043e\u043d\u0438\u0437\u0430\u0446\u0438\u044f<\/strong>. \u0414\u043b\u044f\u00a0\u0431\u043e\u043b\u044c\u0448\u0438\u0445 \u043f\u0430\u0439\u043f\u043b\u0430\u0439\u043d\u043e\u0432 \u043c\u043e\u0436\u043d\u043e \u0433\u0435\u043d\u0435\u0440\u0438\u0440\u043e\u0432\u0430\u0442\u044c requirements \u0441\u00a0\u0445\u0435\u0448\u0430\u043c\u0438, \u0430\u00a0\u0443\u0441\u0442\u0430\u043d\u0430\u0432\u043b\u0438\u0432\u0430\u0442\u044c \u0447\u0435\u0440\u0435\u0437 <code>uv pip sync<\/code>, \u0447\u0442\u043e\u0431\u044b \u043e\u043a\u0440\u0443\u0436\u0435\u043d\u0438\u0435 \u0441\u043e\u043e\u0442\u0432\u0435\u0442\u0441\u0442\u0432\u043e\u0432\u0430\u043b\u043e \u0444\u0430\u0439\u043b\u0443 \u043e\u0434\u0438\u043d \u043a\u00a0\u043e\u0434\u043d\u043e\u043c\u0443. \u0423\u043f\u0440\u0430\u0432\u043b\u0435\u043d\u0438\u0435 \u0445\u0435\u0448\u0430\u043c\u0438 \u0438 \u043f\u0430\u0440\u0430\u043c\u0435\u0442\u0440\u044b \u043a\u043e\u043c\u043f\u0438\u043b\u044f\u0446\u0438\u0438 \u043a\u043e\u043d\u0444\u0438\u0433\u0443\u0440\u0438\u0440\u0443\u044e\u0442\u0441\u044f \u0432 <code>uv.toml<\/code>\/<code>pyproject.toml<\/code> \u0434\u043b\u044f <code>uv pip compile<\/code>. <\/p>\n<p><strong>\u0418\u043d\u0434\u0435\u043a\u0441\u044b \u0438 TLS. <\/strong>\u041d\u0435\u00a0\u0438\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u0435\u043c \u0434\u043e\u0432\u0435\u0440\u0435\u043d\u043d\u044b\u0435 \u043d\u0435\u0431\u0435\u0437\u043e\u043f\u0430\u0441\u043d\u044b\u0435 \u0445\u043e\u0441\u0442\u044b. \u0412\u00a0\u0441\u043f\u0440\u0430\u0432\u043e\u0447\u043d\u0438\u043a\u0435 CLI \u043f\u0440\u044f\u043c\u043e \u0435\u0441\u0442\u044c \u043f\u0440\u0435\u0434\u0443\u043f\u0440\u0435\u0436\u0434\u0435\u043d\u0438\u0435: \u0444\u043b\u0430\u0433 <code>--allow-insecure-host<\/code> \u043e\u0442\u043a\u043b\u044e\u0447\u0430\u0435\u0442 \u0432\u0435\u0440\u0438\u0444\u0438\u043a\u0430\u0446\u0438\u044e \u0446\u0435\u043f\u043e\u0447\u043a\u0438 \u0441\u0435\u0440\u0442\u0438\u0444\u0438\u043a\u0430\u0442\u043e\u0432, \u0447\u0442\u043e\u00a0\u0434\u0435\u043b\u0430\u0435\u0442 \u0432\u0430\u0441 \u0443\u044f\u0437\u0432\u0438\u043c\u044b\u043c\u0438 \u0434\u043b\u044f\u00a0MITM. <\/p>\n<p><strong>\u0412\u043e\u0441\u043f\u0440\u043e\u0438\u0437\u0432\u043e\u0434\u0438\u043c\u043e\u0441\u0442\u044c \u043f\u043e\u00a0\u0432\u0440\u0435\u043c\u0435\u043d\u0438.<\/strong> \u0414\u043e\u0431\u0430\u0432\u043b\u044f\u0435\u043c <code>exclude-newer<\/code> \u043a\u00a0\u0441\u043a\u0440\u0438\u043f\u0442\u0443, \u0434\u0435\u0440\u0436\u0438\u0442\u0435 lock \u0440\u044f\u0434\u043e\u043c \u0441\u00a0\u043d\u0438\u043c, \u0430\u00a0CI \u0437\u0430\u043f\u0443\u0441\u043a\u0430\u0439\u0442\u0435 \u0441 <code>uv run --script<\/code> \u0438\u043b\u0438\u00a0\u0441\u00a0\u044d\u043a\u0441\u043f\u043e\u0440\u0442\u043e\u043c \u0432 <code>requirements.txt<\/code> \u0438 \u043f\u043e\u0441\u043b\u0435\u0434\u0443\u044e\u0449\u0438\u043c <code>uv pip sync<\/code>. <\/p>\n<h4>\u0421\u0446\u0435\u043d\u0430\u0440\u0438\u0439: \u043c\u0430\u043b\u0435\u043d\u044c\u043a\u0438\u0439 CLI-\u0438\u043d\u0442\u0435\u0433\u0440\u0430\u0446\u0438\u043e\u043d\u043d\u044b\u0439 \u0441\u043a\u0440\u0438\u043f\u0442<\/h4>\n<p>\u0423\u0441\u043b\u043e\u0432\u0438\u0435: \u043d\u0443\u0436\u0435\u043d \u043e\u0434\u043d\u043e\u0444\u0430\u0439\u043b\u043e\u0432\u044b\u0439 \u0438\u043d\u0441\u0442\u0440\u0443\u043c\u0435\u043d\u0442 \u0434\u043b\u044f\u00a0\u0432\u044b\u0433\u0440\u0443\u0437\u043a\u0438 \u0434\u0430\u043d\u043d\u044b\u0445 \u0438\u0437\u00a0API, \u0441\u00a0\u0440\u0435\u0442\u0440\u0430\u044f\u043c\u0438 \u0438 \u043b\u043e\u0433\u0430\u043c\u0438, \u0431\u0435\u0437\u00a0\u0443\u0441\u0442\u0430\u043d\u043e\u0432\u043a\u0438 \u043f\u0440\u043e\u0435\u043a\u0442\u0430 \u0438 \u0441\u00a0\u0432\u043e\u0441\u043f\u0440\u043e\u0438\u0437\u0432\u043e\u0434\u0438\u043c\u044b\u043c\u0438 \u0432\u0435\u0440\u0441\u0438\u044f\u043c\u0438.<\/p>\n<p><code>fetch_users.py<\/code>:<\/p>\n<pre><code class=\"python\"># \/\/\/ script # requires-python = \"&gt;=3.11\" # dependencies = [ #   \"httpx==0.27.2\", #   \"tenacity==9.0.0\", #   \"structlog==24.4.0\", # ] # [tool.uv] # exclude-newer = \"2025-08-01T00:00:00Z\" # \/\/\/ from __future__ import annotations  import os import sys import json import time import httpx import structlog from tenacity import retry, stop_after_attempt, wait_exponential, retry_if_exception_type  log = structlog.get_logger()  API_URL = os.environ.get(\"API_URL\", \"https:\/\/httpbin.org\/json\") TIMEOUT = httpx.Timeout(10.0, connect=5.0)  class FetchError(RuntimeError):     pass  @retry(     reraise=True,     stop=stop_after_attempt(4),     wait=wait_exponential(multiplier=0.5, min=0.5, max=5),     retry=retry_if_exception_type((httpx.HTTPError, FetchError)), ) def fetch_json(client: httpx.Client, url: str) -&gt; dict:     r = client.get(url)     if r.status_code &gt;= 500:         # \u0441\u0435\u0440\u0432\u0435\u0440\u043d\u0430\u044f \u043e\u0448\u0438\u0431\u043a\u0430 \u2014 \u043f\u0440\u043e\u0431\u0443\u0435\u043c \u043f\u043e\u0432\u0442\u043e\u0440\u0438\u0442\u044c         raise FetchError(f\"server_error={r.status_code}\")     r.raise_for_status()     return r.json()  def main() -&gt; int:     structlog.configure(processors=[structlog.processors.add_log_level, structlog.processors.JSONRenderer()])     with httpx.Client(timeout=TIMEOUT, headers={\"User-Agent\": \"fetch-users\/1.0\"}) as client:         t0 = time.perf_counter()         data = fetch_json(client, API_URL)         dt = time.perf_counter() - t0         log.info(\"fetched\", bytes=len(json.dumps(data).encode(\"utf-8\")), seconds=round(dt, 3))         print(json.dumps(data, ensure_ascii=False))     return 0  if __name__ == \"__main__\":     try:         sys.exit(main())     except httpx.RequestError as e:         log.error(\"http_error\", error=str(e))         sys.exit(2)     except Exception as e:         log.error(\"unexpected\", error=str(e.__class__.__name__))         sys.exit(3)<\/code><\/pre>\n<p>\u0417\u0430\u043f\u0443\u0441\u043a:<\/p>\n<pre><code class=\"bash\">uv run fetch_users.py # \u0437\u0430\u043a\u0440\u0435\u043f\u043b\u044f\u0435\u043c \u0432\u0435\u0440\u0441\u0438\u0438 \u0440\u044f\u0434\u043e\u043c \u0441\u043e \u0441\u043a\u0440\u0438\u043f\u0442\u043e\u043c uv lock --script fetch_users.py # \u044d\u043a\u0441\u043f\u043e\u0440\u0442 \u0434\u043b\u044f CI uv export --script fetch_users.py -o requirements.txt uvx pip-audit -r requirements.txt<\/code><\/pre>\n<p><code>uv<\/code> \u0437\u0430\u0434\u043e\u043a\u0443\u043c\u0435\u043d\u0442\u0438\u0440\u043e\u0432\u0430\u043b \u0438 lock \u0434\u043b\u044f\u00a0\u0441\u043a\u0440\u0438\u043f\u0442\u043e\u0432, \u0438 \u044d\u043a\u0441\u043f\u043e\u0440\u0442. <\/p>\n<h3>\u0418\u043d\u0441\u043f\u0435\u043a\u0446\u0438\u044f \u0434\u0435\u0440\u0435\u0432\u0430 \u0437\u0430\u0432\u0438\u0441\u0438\u043c\u043e\u0441\u0442\u0435\u0439 \u0438 \u043f\u043e\u043b\u0438\u0442\u0438\u043a\u0430 \u0438\u0441\u0442\u043e\u0447\u043d\u0438\u043a\u043e\u0432<\/h3>\n<p>\u041f\u0435\u0440\u0435\u0434 \u0432\u044b\u043a\u043b\u0430\u0434\u043a\u043e\u0439 \u0432\u00a0\u0445\u0443\u043a pre\u2011commit \u043c\u043e\u0436\u043d\u043e \u043e\u0431\u043e\u0437\u0440\u0435\u0442\u044c \u0434\u0435\u0440\u0435\u0432\u043e:<\/p>\n<pre><code class=\"bash\">uv tree --script fetch_users.py<\/code><\/pre>\n<p>\u0414\u043b\u044f\u00a0\u0447\u0430\u0441\u0442\u043d\u044b\u0445 \u0438\u043d\u0434\u0435\u043a\u0441\u043e\u0432 \u0438 \u0437\u0435\u0440\u043a\u0430\u043b \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u0435\u043c \u043a\u043e\u043d\u0444\u0438\u0433\u0443\u0440\u0430\u0446\u0438\u043e\u043d\u043d\u044b\u0435 \u0444\u0430\u0439\u043b\u044b <code>uv.toml<\/code> \u0438\u043b\u0438 <code>pyproject.toml<\/code> \u0434\u043b\u044f <code>uv pip<\/code> \u0438 \u043d\u0435\u00a0\u0440\u0430\u0437\u0434\u0430\u0435\u043c <code>--allow-insecure-host<\/code>. \u041a\u043e\u043d\u0444\u0438\u0433\u0438 \u043f\u043e\u0434\u0434\u0435\u0440\u0436\u0438\u0432\u0430\u044e\u0442\u0441\u044f \u043d\u0430\u00a0\u0443\u0440\u043e\u0432\u043d\u0435 \u043f\u0440\u043e\u0435\u043a\u0442\u0430 \u0438 \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044f; \u0435\u0441\u0442\u044c \u0438 \u0441\u0438\u0441\u0442\u0435\u043c\u043d\u044b\u0439 \u0443\u0440\u043e\u0432\u0435\u043d\u044c. \u041f\u0443\u0442\u044c \u043a\u00a0\u043a\u0435\u0448\u0443 \u0438 \u043f\u0430\u0440\u0430\u043c\u0435\u0442\u0440\u044b \u043c\u043e\u0436\u043d\u043e \u043a\u043e\u043d\u0442\u0440\u043e\u043b\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u0447\u0435\u0440\u0435\u0437 \u043a\u043e\u043c\u0430\u043d\u0434\u044b \u0438 \u043f\u0435\u0440\u0435\u043c\u0435\u043d\u043d\u044b\u0435 \u043e\u043a\u0440\u0443\u0436\u0435\u043d\u0438\u044f. <\/p>\n<hr\/>\n<h3>\u0418\u0442\u043e\u0433<\/h3>\n<p>PEP 723\u00a0\u0440\u0435\u0448\u0430\u0435\u0442 \u0440\u0443\u0442\u0438\u043d\u0443: \u043e\u0434\u043d\u043e\u0444\u0430\u0439\u043b\u043e\u0432\u044b\u0435 \u0441\u043a\u0440\u0438\u043f\u0442\u044b \u043c\u043e\u0433\u0443\u0442\u00a0\u0431\u044b\u0442\u044c \u0441\u0430\u043c\u043e\u0434\u043e\u0441\u0442\u0430\u0442\u043e\u0447\u043d\u044b\u043c\u0438, \u0431\u0435\u0437\u00a0\u0432\u043e\u043f\u0440\u043e\u0441\u043e\u0432 \u043e\u00a0\u0442\u043e\u043c, \u043a\u0430\u043a\u00a0\u0443\u0441\u0442\u0430\u043d\u043e\u0432\u0438\u0442\u044c \u0437\u0430\u0432\u0438\u0441\u0438\u043c\u043e\u0441\u0442\u0438. <code>uv<\/code> \u0441\u0434\u0435\u043b\u0430\u043b \u044d\u0442\u043e\u00a0\u0431\u044b\u0441\u0442\u0440\u044b\u043c \u0438 \u043e\u043f\u0435\u0440\u0430\u0446\u0438\u043e\u043d\u043d\u043e \u0443\u0434\u043e\u0431\u043d\u044b\u043c: \u0440\u0435\u0434\u0430\u043a\u0442\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u0435 \u0437\u0430\u0432\u0438\u0441\u0438\u043c\u043e\u0441\u0442\u0435\u0439, shebang, lock \u0440\u044f\u0434\u043e\u043c \u0441\u043e \u0441\u043a\u0440\u0438\u043f\u0442\u043e\u043c, \u044d\u043a\u0441\u043f\u043e\u0440\u0442 \u0434\u043b\u044f\u00a0CI, \u043a\u043e\u043d\u0442\u0440\u043e\u043b\u044c \u0432\u0435\u0440\u0441\u0438\u0438 \u0438\u043d\u0442\u0435\u0440\u043f\u0440\u0435\u0442\u0430\u0442\u043e\u0440\u0430. <\/p>\n<blockquote>\n<p>\u041a\u0430\u043a \u0438 \u0432\u00a0\u0441\u043b\u0443\u0447\u0430\u0435 \u0441\u00a0\u043e\u0434\u043d\u043e\u0444\u0430\u0439\u043b\u043e\u0432\u044b\u043c\u0438 \u0441\u043a\u0440\u0438\u043f\u0442\u0430\u043c\u0438 \u043f\u043e\u00a0PEP 723\u00a0\u0438 uv, \u043a\u043e\u0442\u043e\u0440\u044b\u0435 \u043f\u043e\u0437\u0432\u043e\u043b\u044f\u044e\u0442\u00a0\u0431\u044b\u0441\u0442\u0440\u043e \u043f\u0440\u043e\u0432\u0435\u0440\u0438\u0442\u044c \u0438\u0434\u0435\u044e \u0431\u0435\u0437\u00a0\u043b\u0438\u0448\u043d\u0438\u0445 \u043f\u043e\u0434\u0433\u043e\u0442\u043e\u0432\u0438\u0442\u0435\u043b\u044c\u043d\u044b\u0445 \u0448\u0430\u0433\u043e\u0432, \u0443\u00a0\u0432\u0430\u0441 \u0435\u0441\u0442\u044c \u0432\u043e\u0437\u043c\u043e\u0436\u043d\u043e\u0441\u0442\u044c \u0442\u0430\u043a\u00a0\u0436\u0435 \u043f\u043e\u0437\u043d\u0430\u043a\u043e\u043c\u0438\u0442\u044c\u0441\u044f \u0441\u00a0\u043a\u0443\u0440\u0441\u043e\u043c Python Developer. Professional\u00a0\u2014 \u0447\u0435\u0440\u0435\u0437 <strong>\u0431\u0435\u0441\u043f\u043b\u0430\u0442\u043d\u044b\u0435<\/strong> <a href=\"https:\/\/otus.pw\/Chl4o\/\" rel=\"noopener noreferrer nofollow\">\u043e\u0442\u043a\u0440\u044b\u0442\u044b\u0435 \u0443\u0440\u043e\u043a\u0438<\/a>, \u0434\u043e\u0441\u0442\u0443\u043f\u043d\u044b\u0435 \u043f\u043e\u00a0\u0441\u0441\u044b\u043b\u043a\u0435.<\/p>\n<p>\u041a\u0440\u043e\u043c\u0435 \u0442\u043e\u0433\u043e, \u0432\u044b \u043c\u043e\u0436\u0435\u0442\u0435 \u043f\u0440\u043e\u0439\u0442\u0438 <strong>\u0431\u0435\u0441\u043f\u043b\u0430\u0442\u043d\u043e\u0435<\/strong> <a href=\"https:\/\/otus.pw\/5ofm\/\" rel=\"noopener noreferrer nofollow\">\u0432\u0441\u0442\u0443\u043f\u0438\u0442\u0435\u043b\u044c\u043d\u043e\u0435 \u0442\u0435\u0441\u0442\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u0435<\/a>, \u043a\u043e\u0442\u043e\u0440\u043e\u0435 \u043f\u043e\u0437\u0432\u043e\u043b\u0438\u0442 \u043e\u0446\u0435\u043d\u0438\u0442\u044c \u0432\u0430\u0448\u0438 \u0437\u043d\u0430\u043d\u0438\u044f \u0438 \u043d\u0430\u0432\u044b\u043a\u0438.<\/p>\n<p>\u0410\u00a0\u0435\u0441\u043b\u0438 \u0445\u043e\u0442\u0438\u0442\u0435 \u0443\u0437\u043d\u0430\u0442\u044c \u0431\u043e\u043b\u044c\u0448\u0435 \u043e\u00a0\u0441\u0430\u043c\u043e\u043c \u043a\u0443\u0440\u0441\u0435 \u0438 \u0432\u043f\u0435\u0447\u0430\u0442\u043b\u0435\u043d\u0438\u044f\u0445 \u0443\u0447\u0430\u0441\u0442\u043d\u0438\u043a\u043e\u0432, \u0437\u0430\u0433\u043b\u044f\u043d\u0438\u0442\u0435 \u0432 <a href=\"https:\/\/otus.pw\/2zTI\/\" rel=\"noopener noreferrer nofollow\">\u0441\u0435\u043a\u0446\u0438\u044e \u0441\u00a0\u043e\u0442\u0437\u044b\u0432\u0430\u043c\u0438<\/a>.<\/p>\n<\/blockquote>\n<\/div>\n<\/div>\n<\/div>\n<p><!----><!----><\/div>\n<p><!----><!----><br \/> \u0441\u0441\u044b\u043b\u043a\u0430 \u043d\u0430 \u043e\u0440\u0438\u0433\u0438\u043d\u0430\u043b \u0441\u0442\u0430\u0442\u044c\u0438 <a href=\"https:\/\/habr.com\/ru\/articles\/940674\/\"> https:\/\/habr.com\/ru\/articles\/940674\/<\/a><br \/><\/br><\/br><\/p>\n","protected":false},"author":1,"featured_media":0,"comment_status":"open","ping_status":"open","sticky":false,"template":"","format":"standard","meta":{"footnotes":""},"categories":[],"tags":[],"class_list":["post-473023","post","type-post","status-publish","format-standard","hentry"],"_links":{"self":[{"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/posts\/473023","targetHints":{"allow":["GET"]}}],"collection":[{"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/posts"}],"about":[{"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/types\/post"}],"author":[{"embeddable":true,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/users\/1"}],"replies":[{"embeddable":true,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Fcomments&post=473023"}],"version-history":[{"count":0,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/posts\/473023\/revisions"}],"wp:attachment":[{"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Fmedia&parent=473023"}],"wp:term":[{"taxonomy":"category","embeddable":true,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Fcategories&post=473023"},{"taxonomy":"post_tag","embeddable":true,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Ftags&post=473023"}],"curies":[{"name":"wp","href":"https:\/\/api.w.org\/{rel}","templated":true}]}}