{"id":23936,"date":"2025-11-16T15:51:21","date_gmt":"2025-11-16T15:51:21","guid":{"rendered":"https:\/\/pokecon.jp\/job\/?p=23936"},"modified":"2025-11-16T15:51:21","modified_gmt":"2025-11-16T15:51:21","slug":"farm-%e3%82%b9%e3%82%bf%e3%83%83%e3%82%af%e3%82%92%e4%bd%bf%e3%81%a3%e3%81%9f%e3%83%95%e3%83%ab%e3%82%b9%e3%82%bf%e3%83%83%e3%82%af%e3%82%a2%e3%83%97%e3%83%aa%e9%96%8b%e7%99%ba-iimon-tech-blog","status":"publish","type":"post","link":"https:\/\/pokecon.jp\/job\/23936\/","title":{"rendered":"FARM \u30b9\u30bf\u30c3\u30af\u3092\u4f7f\u3063\u305f\u30d5\u30eb\u30b9\u30bf\u30c3\u30af\u30a2\u30d7\u30ea\u958b\u767a &#8211; iimon TECH BLOG"},"content":{"rendered":"\n<br \/><img decoding=\"async\" src=\"https:\/\/ogimage.blog.st-hatena.com\/6801883189062156672\/17179246901320728239\/1763301926\" \/><\/p>\n<div>\n<p><strong>FARM \u30b9\u30bf\u30c3\u30af\u3068\u306f\uff1f<\/strong><\/p>\n<p><strong>FARM \u30b9\u30bf\u30c3\u30af<\/strong>\u3068\u306f\u3001\u4ee5\u4e0b\u306e3\u3064\u306e\u5f37\u529b\u306a\u6280\u8853\u3092\u7d44\u307f\u5408\u308f\u305b\u305f\u3001\u73fe\u4ee3\u7684\u306a Web \u30a2\u30d7\u30ea\u30b1\u30fc\u30b7\u30e7\u30f3\u958b\u767a\u30b9\u30bf\u30c3\u30af\u3067\u3059\u3002<\/p>\n<ul>\n<li><strong>F: FastAPI\uff08\u30d0\u30c3\u30af\u30a8\u30f3\u30c9\uff09<\/strong><\/li>\n<li><strong>R: React\uff08\u30d5\u30ed\u30f3\u30c8\u30a8\u30f3\u30c9\uff09<\/strong><\/li>\n<li><strong>M: MongoDB\uff08\u30c7\u30fc\u30bf\u30d9\u30fc\u30b9\uff09<\/strong><\/li>\n<\/ul>\n<p>\u3053\u308c\u3089\u306e\u6280\u8853\u3092\u7d44\u307f\u5408\u308f\u305b\u308b\u3053\u3068\u3067\u3001\u9ad8\u901f\u30fb\u30b9\u30b1\u30fc\u30e9\u30d6\u30eb\u30fb\u67d4\u8edf\u6027\u306e\u3042\u308b<a target=\"_blank\" class=\"keyword\" href=\"https:\/\/d.hatena.ne.jp\/keyword\/%A5%D5%A5%EB%A5%B9%A5%BF\">\u30d5\u30eb\u30b9\u30bf<\/a>\u30c3\u30af\u30a2\u30d7\u30ea\u30b1\u30fc\u30b7\u30e7\u30f3\u3092\u52b9\u7387\u3088\u304f\u69cb\u7bc9\u3067\u304d\u307e\u3059\u3002<\/p>\n<hr\/>\n<h2 id=\"FARM-\u30b9\u30bf\u30c3\u30af\u306e\u69cb\u6210\u8981\u7d20\"><strong>FARM \u30b9\u30bf\u30c3\u30af\u306e\u69cb\u6210\u8981\u7d20<\/strong><\/h2>\n<h3 id=\"1-FastAPI--\u30d0\u30c3\u30af\u30a8\u30f3\u30c9\"><strong>1. FastAPI \u2014 \u30d0\u30c3\u30af\u30a8\u30f3\u30c9<\/strong><\/h3>\n<p>FastAPI \u306f <a target=\"_blank\" class=\"keyword\" href=\"https:\/\/d.hatena.ne.jp\/keyword\/Python\">Python<\/a> \u88fd\u306e\u6700\u65b0 Web <a target=\"_blank\" class=\"keyword\" href=\"https:\/\/d.hatena.ne.jp\/keyword\/%A5%D5%A5%EC%A1%BC%A5%E0%A5%EF%A1%BC%A5%AF\">\u30d5\u30ec\u30fc\u30e0\u30ef\u30fc\u30af<\/a>\u3067\u3001\u4ee5\u4e0b\u306e\u7279\u5fb4\u304c\u3042\u308a\u307e\u3059\u3002<\/p>\n<ul>\n<li>\u975e\u5e38\u306b\u9ad8\u901f\uff08<a target=\"_blank\" class=\"keyword\" href=\"https:\/\/d.hatena.ne.jp\/keyword\/Python\">Python<\/a> <a target=\"_blank\" class=\"keyword\" href=\"https:\/\/d.hatena.ne.jp\/keyword\/%A5%D5%A5%EC%A1%BC%A5%E0%A5%EF%A1%BC%A5%AF\">\u30d5\u30ec\u30fc\u30e0\u30ef\u30fc\u30af<\/a>\u306e\u4e2d\u3067\u3082\u30c8\u30c3\u30d7\u30af\u30e9\u30b9\uff09<\/li>\n<li>\u578b\u30d2\u30f3\u30c8\u306b\u57fa\u3065\u304f\u81ea\u52d5\u691c\u8a3c\u304c\u53ef\u80fd<\/li>\n<li>Swagger UI \u306b\u3088\u308b <a target=\"_blank\" class=\"keyword\" href=\"https:\/\/d.hatena.ne.jp\/keyword\/API\">API<\/a> \u30c9\u30ad\u30e5\u30e1\u30f3\u30c8\u81ea\u52d5\u751f\u6210<\/li>\n<li>\u975e\u540c\u671f\u51e6\u7406\u306b\u5f37\u3044<\/li>\n<\/ul>\n<p>\u4fe1\u983c\u6027\u306e\u9ad8\u3044 <a target=\"_blank\" class=\"keyword\" href=\"https:\/\/d.hatena.ne.jp\/keyword\/API\">API<\/a> \u3092\u7d20\u65e9\u304f\u69cb\u7bc9\u3067\u304d\u308b\u70b9\u304c\u9b45\u529b\u3067\u3059\u3002<\/p>\n<hr\/>\n<h3 id=\"2-React--\u30d5\u30ed\u30f3\u30c8\u30a8\u30f3\u30c9\"><strong>2. React \u2014 \u30d5\u30ed\u30f3\u30c8\u30a8\u30f3\u30c9<\/strong><\/h3>\n<p>React \u306f UI \u3092\u69cb\u7bc9\u3059\u308b\u305f\u3081\u306e <a target=\"_blank\" class=\"keyword\" href=\"https:\/\/d.hatena.ne.jp\/keyword\/JavaScript\">JavaScript<\/a> \u30e9\u30a4\u30d6\u30e9\u30ea\u3067\u3001\u4ee5\u4e0b\u304c\u5f37\u307f\u3067\u3059\uff1a<\/p>\n<ul>\n<li><a target=\"_blank\" class=\"keyword\" href=\"https:\/\/d.hatena.ne.jp\/keyword\/%A5%B3%A5%F3%A5%DD%A1%BC%A5%CD%A5%F3%A5%C8\">\u30b3\u30f3\u30dd\u30fc\u30cd\u30f3\u30c8<\/a>\u30d9\u30fc\u30b9\u306e\u8a2d\u8a08<\/li>\n<li>\u4eee\u60f3DOM\u306b\u3088\u308b\u52b9\u7387\u7684\u306a\u63cf\u753b<\/li>\n<li>\u5927\u898f\u6a21\u30a2\u30d7\u30ea\u3067\u3082\u6271\u3044\u3084\u3059\u3044<\/li>\n<li>\u8c4a\u5bcc\u306a\u30a8\u30b3\u30b7\u30b9\u30c6\u30e0<\/li>\n<\/ul>\n<hr\/>\n<h3 id=\"3-MongoDB--\u30c7\u30fc\u30bf\u30d9\u30fc\u30b9\"><strong>3. MongoDB \u2014 \u30c7\u30fc\u30bf\u30d9\u30fc\u30b9<\/strong><\/h3>\n<p>MongoDB \u306f\u30c9\u30ad\u30e5\u30e1\u30f3\u30c8\u6307\u5411\u306e NoSQL \u30c7\u30fc\u30bf\u30d9\u30fc\u30b9\u3067\u3059\u3002<\/p>\n<ul>\n<li><a target=\"_blank\" class=\"keyword\" href=\"https:\/\/d.hatena.ne.jp\/keyword\/JSON\">JSON<\/a> \u30e9\u30a4\u30af\u306a\u67d4\u8edf\u306a\u30c7\u30fc\u30bf\u69cb\u9020<\/li>\n<li><a target=\"_blank\" class=\"keyword\" href=\"https:\/\/d.hatena.ne.jp\/keyword\/%A5%B9%A5%AD%A1%BC%A5%DE\">\u30b9\u30ad\u30fc\u30de<\/a>\u306e\u5909\u66f4\u306b\u5f37\u3044<\/li>\n<li>\u30b9\u30b1\u30fc\u30eb\u30a2\u30a6\u30c8\u304c\u5bb9\u6613<\/li>\n<\/ul>\n<p>Todo \u30a2\u30d7\u30ea\u306e\u3088\u3046\u306a\u983b\u7e41\u306b\u69cb\u9020\u304c\u5909\u308f\u308b\u30c7\u30fc\u30bf\u306b\u306f\u7279\u306b\u76f8\u6027\u304c\u826f\u3044\u3067\u3059\u3002<\/p>\n<h2 id=\"FARM-\u30b9\u30bf\u30c3\u30af\u3092\u4f7f\u3046\u30e1\u30ea\u30c3\u30c8\"><strong>FARM \u30b9\u30bf\u30c3\u30af\u3092\u4f7f\u3046\u30e1\u30ea\u30c3\u30c8<\/strong><\/h2>\n<ol>\n<li>\n<p><strong>\u9ad8\u3044\u30d1\u30d5\u30a9\u30fc\u30de\u30f3\u30b9<\/strong><\/p>\n<p> FastAPI + React + MongoDB \u306e\u7d44\u307f\u5408\u308f\u305b\u306f\u3001<a target=\"_blank\" class=\"keyword\" href=\"https:\/\/d.hatena.ne.jp\/keyword\/API\">API<\/a> \u5fdc\u7b54\u3068 UI \u66f4\u65b0\u304c\u975e\u5e38\u306b\u9ad8\u901f\u3002<\/p>\n<\/li>\n<li>\n<p><strong>\u30b9\u30b1\u30fc\u30e9\u30d3\u30ea\u30c6\u30a3\u304c\u9ad8\u3044<\/strong><\/p>\n<p> \u5404<a target=\"_blank\" class=\"keyword\" href=\"https:\/\/d.hatena.ne.jp\/keyword\/%A5%B3%A5%F3%A5%DD%A1%BC%A5%CD%A5%F3%A5%C8\">\u30b3\u30f3\u30dd\u30fc\u30cd\u30f3\u30c8<\/a>\u304c\u5206\u96e2\u3055\u308c\u3066\u3044\u308b\u305f\u3081\u3001\u500b\u5225\u306b\u30b9\u30b1\u30fc\u30eb\u53ef\u80fd\u3002<\/p>\n<\/li>\n<li>\n<p><strong>\u5927\u304d\u306a\u30b3\u30df\u30e5\u30cb\u30c6\u30a3\u3068\u8c4a\u5bcc\u306a\u30e9\u30a4\u30d6\u30e9\u30ea<\/strong><\/p>\n<p> \u554f\u984c\u89e3\u6c7a\u3084\u6a5f\u80fd\u62e1\u5f35\u306b\u56f0\u3089\u306a\u3044\u3002<\/p>\n<\/li>\n<li>\n<p><strong>\u67d4\u8edf\u6027<\/strong><\/p>\n<p> \u5c0f\u898f\u6a21\u306a<a target=\"_blank\" class=\"keyword\" href=\"https:\/\/d.hatena.ne.jp\/keyword\/CRUD\">CRUD<\/a>\u30a2\u30d7\u30ea\u304b\u3089\u5927\u898f\u6a21\u30b7\u30b9\u30c6\u30e0\u307e\u3067\u5bfe\u5fdc\u53ef\u80fd\u3002<\/p>\n<\/li>\n<\/ol>\n<p>\u3053\u308c\u3089\u306e\u6280\u8853\u3092\u7d44\u307f\u5408\u308f\u305b\u308b\u3053\u3068\u3067\u3001FARM\u30b9\u30bf\u30c3\u30af\u306f\u6700\u65b0\u306aWeb\u30a2\u30d7\u30ea\u30b1\u30fc\u30b7\u30e7\u30f3\u3092\u69cb\u7bc9\u3059\u308b\u305f\u3081\u306e\u5305\u62ec\u7684\u306a\u30bd\u30ea\u30e5\u30fc\u30b7\u30e7\u30f3\u3068\u306a\u308a\u307e\u3059\u3002FastAPI \u3092\u4f7f\u3063\u3066\u9ad8\u901f\u3067\u30b9\u30b1\u30fc\u30e9\u30d6\u30eb\u306a\u30d0\u30c3\u30af\u30a8\u30f3\u30c9\u3092\u69cb\u7bc9\u3057\u3001React \u3067\u76f4\u611f\u7684\u304b\u3064\u30ec\u30b9\u30dd\u30f3\u30b7\u30d6\u306a\u30d5\u30ed\u30f3\u30c8\u30a8\u30f3\u30c9\u3092\u5b9f\u73fe\u3057\u3001MongoDB \u3067\u67d4\u8edf\u304b\u3064\u52b9\u7387\u7684\u306a\u30c7\u30fc\u30bf\u4fdd\u5b58\u3092\u884c\u3046\u3053\u3068\u304c\u3067\u304d\u307e\u3059\u3002<br \/>\n\u3053\u306e\u30b9\u30bf\u30c3\u30af\u306f\u3001\u30ea\u30a2\u30eb\u30bf\u30a4\u30e0\u306a\u66f4\u65b0\u304c\u5fc5\u8981\u306a\u30a2\u30d7\u30ea\u30b1\u30fc\u30b7\u30e7\u30f3\u3084\u3001\u8907\u96d1\u306a\u30c7\u30fc\u30bf\u30e2\u30c7\u30eb\u3001\u9ad8\u3044\u30d1\u30d5\u30a9\u30fc\u30de\u30f3\u30b9\u304c\u6c42\u3081\u3089\u308c\u308b\u30a2\u30d7\u30ea\u30b1\u30fc\u30b7\u30e7\u30f3\u306b\u7279\u306b\u9069\u3057\u3066\u3044\u307e\u3059\u3002<\/p>\n<h2 id=\"\u4eca\u56de\u306e\u30d7\u30ed\u30b8\u30a7\u30af\u30c8Todo-\u30a2\u30d7\u30ea\u69cb\u7bc9\"><strong>\u4eca\u56de\u306e\u30d7\u30ed\u30b8\u30a7\u30af\u30c8\uff1aTodo \u30a2\u30d7\u30ea\u69cb\u7bc9<\/strong><\/h2>\n<p>\u3053\u306e FARM \u30b9\u30bf\u30c3\u30af\u3092\u5229\u7528\u3057\u3001\u30aa\u30f3\u30e9\u30a4\u30f3\u306e Todo \u30a2\u30d7\u30ea\u3092\u53c2\u8003\u306b\u3001\u6a5f\u80fd\u3092\u518d\u73fe\u3057\u305f\u00a0<strong><a target=\"_blank\" class=\"keyword\" href=\"https:\/\/d.hatena.ne.jp\/keyword\/%A5%D5%A5%EB%A5%B9%A5%BF\">\u30d5\u30eb\u30b9\u30bf<\/a>\u30c3\u30af Todo \u30a2\u30d7\u30ea<\/strong>\u3092\u69cb\u7bc9\u3057\u307e\u3057\u305f\u3002<\/p>\n<h2 id=\"Todo-\u30a2\u30d7\u30ea\u306e\u6a5f\u80fd\"><strong>Todo \u30a2\u30d7\u30ea\u306e\u6a5f\u80fd<\/strong><\/h2>\n<ol>\n<li><strong>\u8907\u6570\u306e Todo \u30ea\u30b9\u30c8<\/strong>\n<ul>\n<li>\u4f5c\u6210\u30fb\u8868\u793a\u30fb\u66f4\u65b0\u30fb\u524a\u9664\u304c\u53ef\u80fd<\/li>\n<li>\u5404\u30ea\u30b9\u30c8\u306b\u306f\u8907\u6570\u306e Todo \u30a2\u30a4\u30c6\u30e0\u3092\u4fdd\u6301<\/li>\n<\/ul>\n<\/li>\n<li><strong>Todo \u30a2\u30a4\u30c6\u30e0<\/strong>\n<ul>\n<li>\u8ffd\u52a0\u30fb\u66f4\u65b0\u30fb\u524a\u9664<\/li>\n<li>&#8220;\u30c1\u30a7\u30c3\u30af\u6e08\u307f \/ \u672a\u30c1\u30a7\u30c3\u30af&#8221; \u72b6\u614b\u306e\u5207\u308a\u66ff\u3048<\/li>\n<\/ul>\n<\/li>\n<li><strong>\u30ea\u30a2\u30eb\u30bf\u30a4\u30e0\u3067UI\u66f4\u65b0<\/strong><\/li>\n<li><strong>\u30ec\u30b9\u30dd\u30f3\u30b7\u30d6\u30c7\u30b6\u30a4\u30f3\u5bfe\u5fdc<\/strong><\/li>\n<\/ol>\n<p>\u30a2\u30d7\u30ea\u306f\u5178\u578b\u7684\u306a FARM \u69cb\u6210\u3092\u63a1\u7528\uff1a<\/p>\n<ol>\n<li><strong>\u30d5\u30ed\u30f3\u30c8\u30a8\u30f3\u30c9\uff08React\uff09<\/strong>\n<\/li>\n<li><strong>\u30d0\u30c3\u30af\u30a8\u30f3\u30c9\uff08FastAPI\uff09<\/strong>\n<\/li>\n<li><strong>\u30c7\u30fc\u30bf\u30d9\u30fc\u30b9\uff08MongoDB\uff09<\/strong><\/li>\n<li><strong>Docker \u306b\u3088\u308b\u30b3\u30f3\u30c6\u30ca\u5316<\/strong>\n<ul>\n<li>\u74b0\u5883\u5dee\u7570\u3092\u5438\u53ce\u3057\u3001\u30c7\u30d7\u30ed\u30a4\u3082\u5bb9\u6613\u306b<\/li>\n<\/ul>\n<\/li>\n<\/ol>\n<p>\u3053\u3053\u3067\u306f\u3001\u30d7\u30ed\u30b8\u30a7\u30af\u30c8\u69cb\u9020\u3001\u30d0\u30c3\u30af\u30a8\u30f3\u30c9\u3001\u30d5\u30ed\u30f3\u30c8\u30a8\u30f3\u30c9\u3001Docker \u306b\u3088\u308b\u904b\u7528\u307e\u3067\u3092\u6bb5\u968e\u7684\u306b\u30bb\u30c3\u30c8\u30a2\u30c3\u30d7\u3057\u3066\u3044\u304d\u307e\u3057\u305f\u3002<\/p>\n<p><strong>\u30d7\u30ed\u30b8\u30a7\u30af\u30c8\u7528\u306e\u65b0\u3057\u3044<a target=\"_blank\" class=\"keyword\" href=\"https:\/\/d.hatena.ne.jp\/keyword\/%A5%C7%A5%A3%A5%EC%A5%AF%A5%C8\">\u30c7\u30a3\u30ec\u30af\u30c8<\/a>\u30ea\u3092\u4f5c\u6210\u3057\u307e\u3059\uff1a<\/strong><\/p>\n<pre class=\"code csharp\" data-lang=\"csharp\" data-unlink=\"\">   mkdir farm-stack-todo\n   cd farm-stack-todo\n<\/pre>\n<p>\u30d0\u30c3\u30af\u30a8\u30f3\u30c9\u3068\u30d5\u30ed\u30f3\u30c8\u30a8\u30f3\u30c9\u7528\u306e\u30b5\u30d6<a target=\"_blank\" class=\"keyword\" href=\"https:\/\/d.hatena.ne.jp\/keyword\/%A5%C7%A5%A3%A5%EC%A5%AF%A5%C8\">\u30c7\u30a3\u30ec\u30af\u30c8<\/a>\u30ea\u3092\u4f5c\u6210\u3057\u307e\u3059\uff1a<\/p>\n<pre class=\"code lang-jsx\" data-lang=\"jsx\" data-unlink=\"\"> mkdir backend frontend\n<\/pre>\n<p><strong>\u30b9\u30c6\u30c3\u30d7 2\uff1a\u30d0\u30c3\u30af\u30a8\u30f3\u30c9\u74b0\u5883\u3092\u30bb\u30c3\u30c8\u30a2\u30c3\u30d7\u3059\u308b<\/strong><\/p>\n<p>\u30d0\u30c3\u30af\u30a8\u30f3\u30c9<a target=\"_blank\" class=\"keyword\" href=\"https:\/\/d.hatena.ne.jp\/keyword\/%A5%C7%A5%A3%A5%EC%A5%AF%A5%C8\">\u30c7\u30a3\u30ec\u30af\u30c8<\/a>\u30ea\u3078\u79fb\u52d5\u3057\u307e\u3059\uff1a<\/p>\n<pre class=\"code lang-jsx\" data-lang=\"jsx\" data-unlink=\"\">cd backend\n<\/pre>\n<p>\u4eee\u60f3\u74b0\u5883\u3092\u4f5c\u6210\u3057\u3001\u6709\u52b9\u5316\u3057\u307e\u3059\uff1a<\/p>\n<pre class=\"code csharp\" data-lang=\"csharp\" data-unlink=\"\">   python -m venv venv\n   source venv\/bin\/activate  # On Windows, use: venv\\Scripts\\activate<\/pre>\n<p><strong>\u30d0\u30c3\u30af\u30a8\u30f3\u30c9<a target=\"_blank\" class=\"keyword\" href=\"https:\/\/d.hatena.ne.jp\/keyword\/%A5%C7%A5%A3%A5%EC%A5%AF%A5%C8\">\u30c7\u30a3\u30ec\u30af\u30c8<\/a>\u30ea\u306b\u6b21\u306e\u30d5\u30a1\u30a4\u30eb\u3092\u4f5c\u6210\u3057\u307e\u3059\uff1a<\/strong><\/p>\n<ol>\n<li>\n<ul>\n<li><strong>Dockerfile<\/strong><\/li>\n<li><strong>pyproject.toml<\/strong><\/li>\n<\/ul>\n<\/li>\n<\/ol>\n<p>\u305d\u306e\u5f8c\u3001\u30bf\u30fc\u30df\u30ca\u30eb\u3067\u5fc5\u8981\u306a\u30d1\u30c3\u30b1\u30fc\u30b8\u3092\u30a4\u30f3\u30b9\u30c8\u30fc\u30eb\u3057\u307e\u3059\uff1a<\/p>\n<pre class=\"code lang-jsx\" data-lang=\"jsx\" data-unlink=\"\">pip install <span class=\"synConstant\">\"fastapi[all]\"<\/span> <span class=\"synConstant\">\"motor[srv]\"<\/span> beanie aiostream\n<\/pre>\n<p>requirements.txt \u30d5\u30a1\u30a4\u30eb\u3092\u751f\u6210\u3057\u307e\u3059\uff1a<\/p>\n<pre class=\"code lang-jsx\" data-lang=\"jsx\" data-unlink=\"\">pip freeze <span class=\"synStatement\">&gt;<\/span> requirements<span class=\"synStatement\">.<\/span>txt\n<\/pre>\n<p><strong>requirements.txt \u30d5\u30a1\u30a4\u30eb\u3092\u4f5c\u6210\u3057\u305f\u5f8c\uff08pip-compile \u3067\u751f\u6210\u3057\u3066\u3082\u624b\u52d5\u3067\u4f5c\u6210\u3057\u3066\u3082\uff09\u3001\u6b21\u306e\u30b3\u30de\u30f3\u30c9\u3067\u4f9d\u5b58\u95a2\u4fc2\u3092\u30a4\u30f3\u30b9\u30c8\u30fc\u30eb\u3067\u304d\u307e\u3059\uff1a<\/strong><\/p>\n<pre class=\"code lang-jsx\" data-lang=\"jsx\" data-unlink=\"\">pip install <span class=\"synStatement\">-<\/span>r requirements<span class=\"synStatement\">.<\/span>txt\n<\/pre>\n<p>Dockerfile \u306b\u6b21\u306e\u5185\u5bb9\u3092\u8ffd\u52a0\u3057\u307e\u3059\uff1a<\/p>\n<pre class=\"code csharp\" data-lang=\"csharp\" data-unlink=\"\"> FROM python:3\n\n   WORKDIR \/usr\/src\/app\n   COPY requirements.txt .\/\n\n   RUN pip install --no-cache-dir --upgrade -r .\/requirements.txt\n\n   EXPOSE 3001\n\n   CMD [ \"python\", \".\/src\/server.py\" ]<\/pre>\n<p><strong>pyproject.toml \u306b\u6b21\u306e\u5185\u5bb9\u3092\u8ffd\u52a0\u3057\u307e\u3059\uff1a<\/strong><\/p>\n<pre class=\"code csharp\" data-lang=\"csharp\" data-unlink=\"\">  [tool.pytest.ini_options]\n   pythonpath = \"src\"<\/pre>\n<p><strong>\u30d0\u30c3\u30af\u30a8\u30f3\u30c9\u69cb\u9020\u3092\u8a2d\u5b9a\u3059\u308b\u30b9\u30c6\u30c3\u30d7 4:<\/strong><\/p>\n<p>\u30d0\u30c3\u30af\u30a8\u30f3\u30c9<a target=\"_blank\" class=\"keyword\" href=\"https:\/\/d.hatena.ne.jp\/keyword\/%A5%C7%A5%A3%A5%EC%A5%AF%A5%C8\">\u30c7\u30a3\u30ec\u30af\u30c8<\/a>\u30ea\u306e\u4e2d\u306b\u00a0<code>src<\/code>\u00a0<a target=\"_blank\" class=\"keyword\" href=\"https:\/\/d.hatena.ne.jp\/keyword\/%A5%C7%A5%A3%A5%EC%A5%AF%A5%C8\">\u30c7\u30a3\u30ec\u30af\u30c8<\/a>\u30ea\u3092\u4f5c\u6210\u3057\u307e\u3059\uff1a<\/p>\n<pre class=\"code lang-jsx\" data-lang=\"jsx\" data-unlink=\"\"> mkdir src\n<\/pre>\n<p><code>src<\/code>\u00a0<a target=\"_blank\" class=\"keyword\" href=\"https:\/\/d.hatena.ne.jp\/keyword\/%A5%C7%A5%A3%A5%EC%A5%AF%A5%C8\">\u30c7\u30a3\u30ec\u30af\u30c8<\/a>\u30ea\u5185\u306b\u4ee5\u4e0b\u306e\u30d5\u30a1\u30a4\u30eb\u3092\u4f5c\u6210\u3057\u307e\u3059\uff1a<\/p>\n<p>\u30b9\u30c6\u30c3\u30d7 5: \u30c7\u30fc\u30bf\u30a2\u30af\u30bb\u30b9\u5c64 (DAL) \u3092\u5b9f\u88c5\u3059\u308b<\/p>\n<p><code>src\/dal.py<\/code>\u00a0\u3092\u958b\u304d\u3001\u4ee5\u4e0b\u306e\u5185\u5bb9\u3092\u8ffd\u52a0\u3057\u307e\u3059\uff1a<\/p>\n<pre class=\"code lang-python\" data-lang=\"python\" data-unlink=\"\"><span class=\"synPreProc\">from<\/span> bson <span class=\"synPreProc\">import<\/span> ObjectId\n<span class=\"synPreProc\">from<\/span> motor.motor_asyncio <span class=\"synPreProc\">import<\/span> AsyncIOMotorCollection\n<span class=\"synPreProc\">from<\/span> pymongo <span class=\"synPreProc\">import<\/span> ReturnDocument\n\n<span class=\"synPreProc\">from<\/span> pydantic <span class=\"synPreProc\">import<\/span> BaseModel\n\n<span class=\"synPreProc\">from<\/span> uuid <span class=\"synPreProc\">import<\/span> uuid4\n\n<span class=\"synStatement\">class<\/span> <span class=\"synIdentifier\">ListSummary<\/span>(BaseModel):\n  <span class=\"synIdentifier\">id<\/span>: <span class=\"synIdentifier\">str<\/span>\n  name: <span class=\"synIdentifier\">str<\/span>\n  item_count: <span class=\"synIdentifier\">int<\/span>\n\n  <span class=\"synPreProc\">@<\/span><span class=\"synIdentifier\">staticmethod<\/span>\n  <span class=\"synStatement\">def<\/span> <span class=\"synIdentifier\">from_doc<\/span>(doc) -&gt; <span class=\"synConstant\">\"ListSummary\"<\/span>:\n      <span class=\"synStatement\">return<\/span> ListSummary(\n          <span class=\"synIdentifier\">id<\/span>=<span class=\"synIdentifier\">str<\/span>(doc[<span class=\"synConstant\">\"_id\"<\/span>]),\n          name=doc[<span class=\"synConstant\">\"name\"<\/span>],\n          item_count=doc[<span class=\"synConstant\">\"item_count\"<\/span>],\n      )\n\n<span class=\"synStatement\">class<\/span> <span class=\"synIdentifier\">ToDoListItem<\/span>(BaseModel):\n  <span class=\"synIdentifier\">id<\/span>: <span class=\"synIdentifier\">str<\/span>\n  label: <span class=\"synIdentifier\">str<\/span>\n  checked: <span class=\"synIdentifier\">bool<\/span>\n\n  <span class=\"synPreProc\">@<\/span><span class=\"synIdentifier\">staticmethod<\/span>\n  <span class=\"synStatement\">def<\/span> <span class=\"synIdentifier\">from_doc<\/span>(item) -&gt; <span class=\"synConstant\">\"ToDoListItem\"<\/span>:\n      <span class=\"synStatement\">return<\/span> ToDoListItem(\n          <span class=\"synIdentifier\">id<\/span>=item[<span class=\"synConstant\">\"id\"<\/span>],\n          label=item[<span class=\"synConstant\">\"label\"<\/span>],\n          checked=item[<span class=\"synConstant\">\"checked\"<\/span>],\n      )\n\n<span class=\"synStatement\">class<\/span> <span class=\"synIdentifier\">ToDoList<\/span>(BaseModel):\n  <span class=\"synIdentifier\">id<\/span>: <span class=\"synIdentifier\">str<\/span>\n  name: <span class=\"synIdentifier\">str<\/span>\n  items: <span class=\"synIdentifier\">list<\/span>[ToDoListItem]\n\n  <span class=\"synPreProc\">@<\/span><span class=\"synIdentifier\">staticmethod<\/span>\n  <span class=\"synStatement\">def<\/span> <span class=\"synIdentifier\">from_doc<\/span>(doc) -&gt; <span class=\"synConstant\">\"ToDoList\"<\/span>:\n      <span class=\"synStatement\">return<\/span> ToDoList(\n          <span class=\"synIdentifier\">id<\/span>=<span class=\"synIdentifier\">str<\/span>(doc[<span class=\"synConstant\">\"_id\"<\/span>]),\n          name=doc[<span class=\"synConstant\">\"name\"<\/span>],\n          items=[ToDoListItem.from_doc(item) <span class=\"synStatement\">for<\/span> item <span class=\"synStatement\">in<\/span> doc[<span class=\"synConstant\">\"items\"<\/span>]],\n      )\n\n<span class=\"synStatement\">class<\/span> <span class=\"synIdentifier\">ToDoDAL<\/span>:\n  <span class=\"synStatement\">def<\/span> <span class=\"synIdentifier\">__init__<\/span>(self, todo_collection: AsyncIOMotorCollection):\n      self._todo_collection = todo_collection\n\n  <span class=\"synStatement\">async<\/span> <span class=\"synStatement\">def<\/span> <span class=\"synIdentifier\">list_todo_lists<\/span>(self, session=<span class=\"synIdentifier\">None<\/span>):\n      <span class=\"synStatement\">async<\/span> <span class=\"synStatement\">for<\/span> doc <span class=\"synStatement\">in<\/span> self._todo_collection.find(\n          {},\n          projection={\n              <span class=\"synConstant\">\"name\"<\/span>: <span class=\"synConstant\">1<\/span>,\n              <span class=\"synConstant\">\"item_count\"<\/span>: {<span class=\"synConstant\">\"$size\"<\/span>: <span class=\"synConstant\">\"$items\"<\/span>},\n          },\n          sort={<span class=\"synConstant\">\"name\"<\/span>: <span class=\"synConstant\">1<\/span>},\n          session=session,\n      ):\n          <span class=\"synStatement\">yield<\/span> ListSummary.from_doc(doc)\n\n  <span class=\"synStatement\">async<\/span> <span class=\"synStatement\">def<\/span> <span class=\"synIdentifier\">create_todo_list<\/span>(self, name: <span class=\"synIdentifier\">str<\/span>, session=<span class=\"synIdentifier\">None<\/span>) -&gt; <span class=\"synIdentifier\">str<\/span>:\n      response = <span class=\"synStatement\">await<\/span> self._todo_collection.insert_one(\n          {<span class=\"synConstant\">\"name\"<\/span>: name, <span class=\"synConstant\">\"items\"<\/span>: []},\n          session=session,\n      )\n      <span class=\"synStatement\">return<\/span> <span class=\"synIdentifier\">str<\/span>(response.inserted_id)\n\n  <span class=\"synStatement\">async<\/span> <span class=\"synStatement\">def<\/span> <span class=\"synIdentifier\">get_todo_list<\/span>(self, <span class=\"synIdentifier\">id<\/span>: <span class=\"synIdentifier\">str<\/span> | ObjectId, session=<span class=\"synIdentifier\">None<\/span>) -&gt; ToDoList:\n      doc = <span class=\"synStatement\">await<\/span> self._todo_collection.find_one(\n          {<span class=\"synConstant\">\"_id\"<\/span>: ObjectId(<span class=\"synIdentifier\">id<\/span>)},\n          session=session,\n      )\n      <span class=\"synStatement\">return<\/span> ToDoList.from_doc(doc)\n\n  <span class=\"synStatement\">async<\/span> <span class=\"synStatement\">def<\/span> <span class=\"synIdentifier\">delete_todo_list<\/span>(self, <span class=\"synIdentifier\">id<\/span>: <span class=\"synIdentifier\">str<\/span> | ObjectId, session=<span class=\"synIdentifier\">None<\/span>) -&gt; <span class=\"synIdentifier\">bool<\/span>:\n      response = <span class=\"synStatement\">await<\/span> self._todo_collection.delete_one(\n          {<span class=\"synConstant\">\"_id\"<\/span>: ObjectId(<span class=\"synIdentifier\">id<\/span>)},\n          session=session,\n      )\n      <span class=\"synStatement\">return<\/span> response.deleted_count == <span class=\"synConstant\">1<\/span>\n\n  <span class=\"synStatement\">async<\/span> <span class=\"synStatement\">def<\/span> <span class=\"synIdentifier\">create_item<\/span>(\n      self,\n      <span class=\"synIdentifier\">id<\/span>: <span class=\"synIdentifier\">str<\/span> | ObjectId,\n      label: <span class=\"synIdentifier\">str<\/span>,\n      session=<span class=\"synIdentifier\">None<\/span>,\n  ) -&gt; ToDoList | <span class=\"synIdentifier\">None<\/span>:\n      result = <span class=\"synStatement\">await<\/span> self._todo_collection.find_one_and_update(\n          {<span class=\"synConstant\">\"_id\"<\/span>: ObjectId(<span class=\"synIdentifier\">id<\/span>)},\n          {\n              <span class=\"synConstant\">\"$push\"<\/span>: {\n                  <span class=\"synConstant\">\"items\"<\/span>: {\n                      <span class=\"synConstant\">\"id\"<\/span>: uuid4().hex,\n                      <span class=\"synConstant\">\"label\"<\/span>: label,\n                      <span class=\"synConstant\">\"checked\"<\/span>: <span class=\"synIdentifier\">False<\/span>,\n                  }\n              }\n          },\n          session=session,\n          return_document=ReturnDocument.AFTER,\n      )\n      <span class=\"synStatement\">if<\/span> result:\n          <span class=\"synStatement\">return<\/span> ToDoList.from_doc(result)\n\n  <span class=\"synStatement\">async<\/span> <span class=\"synStatement\">def<\/span> <span class=\"synIdentifier\">set_checked_state<\/span>(\n      self,\n      doc_id: <span class=\"synIdentifier\">str<\/span> | ObjectId,\n      item_id: <span class=\"synIdentifier\">str<\/span>,\n      checked_state: <span class=\"synIdentifier\">bool<\/span>,\n      session=<span class=\"synIdentifier\">None<\/span>,\n  ) -&gt; ToDoList | <span class=\"synIdentifier\">None<\/span>:\n      result = <span class=\"synStatement\">await<\/span> self._todo_collection.find_one_and_update(\n          {<span class=\"synConstant\">\"_id\"<\/span>: ObjectId(doc_id), <span class=\"synConstant\">\"items.id\"<\/span>: item_id},\n          {<span class=\"synConstant\">\"$set\"<\/span>: {<span class=\"synConstant\">\"items.$.checked\"<\/span>: checked_state}},\n          session=session,\n          return_document=ReturnDocument.AFTER,\n      )\n      <span class=\"synStatement\">if<\/span> result:\n          <span class=\"synStatement\">return<\/span> ToDoList.from_doc(result)\n\n  <span class=\"synStatement\">async<\/span> <span class=\"synStatement\">def<\/span> <span class=\"synIdentifier\">delete_item<\/span>(\n      self,\n      doc_id: <span class=\"synIdentifier\">str<\/span> | ObjectId,\n      item_id: <span class=\"synIdentifier\">str<\/span>,\n      session=<span class=\"synIdentifier\">None<\/span>,\n  ) -&gt; ToDoList | <span class=\"synIdentifier\">None<\/span>:\n      result = <span class=\"synStatement\">await<\/span> self._todo_collection.find_one_and_update(\n          {<span class=\"synConstant\">\"_id\"<\/span>: ObjectId(doc_id)},\n          {<span class=\"synConstant\">\"$pull\"<\/span>: {<span class=\"synConstant\">\"items\"<\/span>: {<span class=\"synConstant\">\"id\"<\/span>: item_id}}},\n          session=session,\n          return_document=ReturnDocument.AFTER,\n      )\n      <span class=\"synStatement\">if<\/span> result:\n          <span class=\"synStatement\">return<\/span> ToDoList.from_doc(result)\n<\/pre>\n<p>\u3053\u308c\u3067\u3001\u30d7\u30ed\u30b8\u30a7\u30af\u30c8\u69cb\u9020\u3092\u8a2d\u5b9a\u3057\u3001FARM\u30b9\u30bf\u30c3\u30af\u306eTodo\u30a2\u30d7\u30ea\u30b1\u30fc\u30b7\u30e7\u30f3\u306e\u30c7\u30fc\u30bf\u30a2\u30af\u30bb\u30b9\u5c64 (DAL) \u3092\u5b9f\u88c5\u3057\u307e\u3057\u305f\u3002\u6b21\u306f\u3001FastAPI \u30b5\u30fc\u30d0\u30fc\u3092\u5b9f\u88c5\u3057\u3001<a target=\"_blank\" class=\"keyword\" href=\"https:\/\/d.hatena.ne.jp\/keyword\/API\">API<\/a> \u30a8\u30f3\u30c9\u30dd\u30a4\u30f3\u30c8\u3092\u4f5c\u6210\u3057\u307e\u3059\u3002<\/p>\n<p>\u30b9\u30c6\u30c3\u30d76: FastAPI \u30b5\u30fc\u30d0\u30fc\u3092\u5b9f\u88c5\u3059\u308b<\/p>\n<p>src\/<a target=\"_blank\" href=\"http:\/\/server.py\/\">server.py<\/a>\u00a0\u3092\u958b\u304d\u3001\u6b21\u306e\u5185\u5bb9\u3092\u8ffd\u52a0\u3057\u307e\u3059:<\/p>\n<pre class=\"code lang-python\" data-lang=\"python\" data-unlink=\"\"><span class=\"synPreProc\">from<\/span> contextlib <span class=\"synPreProc\">import<\/span> asynccontextmanager\n<span class=\"synPreProc\">from<\/span> datetime <span class=\"synPreProc\">import<\/span> datetime\n<span class=\"synPreProc\">import<\/span> os\n<span class=\"synPreProc\">import<\/span> sys\n\n<span class=\"synPreProc\">from<\/span> bson <span class=\"synPreProc\">import<\/span> ObjectId\n<span class=\"synPreProc\">from<\/span> fastapi <span class=\"synPreProc\">import<\/span> FastAPI, status\n<span class=\"synPreProc\">from<\/span> motor.motor_asyncio <span class=\"synPreProc\">import<\/span> AsyncIOMotorClient\n<span class=\"synPreProc\">from<\/span> pydantic <span class=\"synPreProc\">import<\/span> BaseModel\n<span class=\"synPreProc\">import<\/span> uvicorn\n\n<span class=\"synPreProc\">from<\/span> dal <span class=\"synPreProc\">import<\/span> ToDoDAL, ListSummary, ToDoList\n\nCOLLECTION_NAME = <span class=\"synConstant\">\"todo_lists\"<\/span>\nMONGODB_URI = os.environ[<span class=\"synConstant\">\"MONGODB_URI\"<\/span>]\nDEBUG = os.environ.get(<span class=\"synConstant\">\"DEBUG\"<\/span>, <span class=\"synConstant\">\"\"<\/span>).strip().lower() <span class=\"synStatement\">in<\/span> {<span class=\"synConstant\">\"1\"<\/span>, <span class=\"synConstant\">\"true\"<\/span>, <span class=\"synConstant\">\"on\"<\/span>, <span class=\"synConstant\">\"yes\"<\/span>}\n\n<span class=\"synPreProc\">@<\/span><span class=\"synIdentifier\">asynccontextmanager<\/span>\n<span class=\"synStatement\">async<\/span> <span class=\"synStatement\">def<\/span> <span class=\"synIdentifier\">lifespan<\/span>(app: FastAPI):\n    \n    client = AsyncIOMotorClient(MONGODB_URI)\n    database = client.get_default_database()\n\n    \n    pong = <span class=\"synStatement\">await<\/span> database.command(<span class=\"synConstant\">\"ping\"<\/span>)\n    <span class=\"synStatement\">if<\/span> <span class=\"synIdentifier\">int<\/span>(pong[<span class=\"synConstant\">\"ok\"<\/span>]) != <span class=\"synConstant\">1<\/span>:\n        <span class=\"synStatement\">raise<\/span> <span class=\"synType\">Exception<\/span>(<span class=\"synConstant\">\"Cluster connection is not okay!\"<\/span>)\n\n    todo_lists = database.get_collection(COLLECTION_NAME)\n    app.todo_dal = ToDoDAL(todo_lists)\n\n    \n    <span class=\"synStatement\">yield<\/span>\n\n    \n    client.close()\n\napp = FastAPI(lifespan=lifespan, debug=DEBUG)\n\n<span class=\"synPreProc\">@<\/span><span class=\"synIdentifier\">app.get<\/span>(<span class=\"synConstant\">\"\/api\/lists\"<\/span>)\n<span class=\"synStatement\">async<\/span> <span class=\"synStatement\">def<\/span> <span class=\"synIdentifier\">get_all_lists<\/span>() -&gt; <span class=\"synIdentifier\">list<\/span>[ListSummary]:\n    <span class=\"synStatement\">return<\/span> [i <span class=\"synStatement\">async<\/span> <span class=\"synStatement\">for<\/span> i <span class=\"synStatement\">in<\/span> app.todo_dal.list_todo_lists()]\n\n<span class=\"synStatement\">class<\/span> <span class=\"synIdentifier\">NewList<\/span>(BaseModel):\n    name: <span class=\"synIdentifier\">str<\/span>\n\n<span class=\"synStatement\">class<\/span> <span class=\"synIdentifier\">NewListResponse<\/span>(BaseModel):\n    <span class=\"synIdentifier\">id<\/span>: <span class=\"synIdentifier\">str<\/span>\n    name: <span class=\"synIdentifier\">str<\/span>\n\n<span class=\"synPreProc\">@<\/span><span class=\"synIdentifier\">app.post<\/span>(<span class=\"synConstant\">\"\/api\/lists\"<\/span>, status_code=status.HTTP_201_CREATED)\n<span class=\"synStatement\">async<\/span> <span class=\"synStatement\">def<\/span> <span class=\"synIdentifier\">create_todo_list<\/span>(new_list: NewList) -&gt; NewListResponse:\n    <span class=\"synStatement\">return<\/span> NewListResponse(\n        <span class=\"synIdentifier\">id<\/span>=<span class=\"synStatement\">await<\/span> app.todo_dal.create_todo_list(new_list.name),\n        name=new_list.name,\n    )\n\n<span class=\"synPreProc\">@<\/span><span class=\"synIdentifier\">app.get<\/span>(<span class=\"synConstant\">\"\/api\/lists\/{list_id}\"<\/span>)\n<span class=\"synStatement\">async<\/span> <span class=\"synStatement\">def<\/span> <span class=\"synIdentifier\">get_list<\/span>(list_id: <span class=\"synIdentifier\">str<\/span>) -&gt; ToDoList:\n    <span class=\"synConstant\">\"\"\"Get a single to-do list\"\"\"<\/span>\n    <span class=\"synStatement\">return<\/span> <span class=\"synStatement\">await<\/span> app.todo_dal.get_todo_list(list_id)\n\n<span class=\"synPreProc\">@<\/span><span class=\"synIdentifier\">app.delete<\/span>(<span class=\"synConstant\">\"\/api\/lists\/{list_id}\"<\/span>)\n<span class=\"synStatement\">async<\/span> <span class=\"synStatement\">def<\/span> <span class=\"synIdentifier\">delete_list<\/span>(list_id: <span class=\"synIdentifier\">str<\/span>) -&gt; <span class=\"synIdentifier\">bool<\/span>:\n    <span class=\"synStatement\">return<\/span> <span class=\"synStatement\">await<\/span> app.todo_dal.delete_todo_list(list_id)\n\n<span class=\"synStatement\">class<\/span> <span class=\"synIdentifier\">NewItem<\/span>(BaseModel):\n    label: <span class=\"synIdentifier\">str<\/span>\n\n<span class=\"synStatement\">class<\/span> <span class=\"synIdentifier\">NewItemResponse<\/span>(BaseModel):\n    <span class=\"synIdentifier\">id<\/span>: <span class=\"synIdentifier\">str<\/span>\n    label: <span class=\"synIdentifier\">str<\/span>\n\n<span class=\"synPreProc\">@<\/span><span class=\"synIdentifier\">app.post<\/span>(\n    <span class=\"synConstant\">\"\/api\/lists\/{list_id}\/items\/\"<\/span>,\n    status_code=status.HTTP_201_CREATED,\n)\n<span class=\"synStatement\">async<\/span> <span class=\"synStatement\">def<\/span> <span class=\"synIdentifier\">create_item<\/span>(list_id: <span class=\"synIdentifier\">str<\/span>, new_item: NewItem) -&gt; ToDoList:\n    <span class=\"synStatement\">return<\/span> <span class=\"synStatement\">await<\/span> app.todo_dal.create_item(list_id, new_item.label)\n\n<span class=\"synPreProc\">@<\/span><span class=\"synIdentifier\">app.delete<\/span>(<span class=\"synConstant\">\"\/api\/lists\/{list_id}\/items\/{item_id}\"<\/span>)\n<span class=\"synStatement\">async<\/span> <span class=\"synStatement\">def<\/span> <span class=\"synIdentifier\">delete_item<\/span>(list_id: <span class=\"synIdentifier\">str<\/span>, item_id: <span class=\"synIdentifier\">str<\/span>) -&gt; ToDoList:\n    <span class=\"synStatement\">return<\/span> <span class=\"synStatement\">await<\/span> app.todo_dal.delete_item(list_id, item_id)\n\n<span class=\"synStatement\">class<\/span> <span class=\"synIdentifier\">ToDoItemUpdate<\/span>(BaseModel):\n    item_id: <span class=\"synIdentifier\">str<\/span>\n    checked_state: <span class=\"synIdentifier\">bool<\/span>\n\n<span class=\"synPreProc\">@<\/span><span class=\"synIdentifier\">app.patch<\/span>(<span class=\"synConstant\">\"\/api\/lists\/{list_id}\/checked_state\"<\/span>)\n<span class=\"synStatement\">async<\/span> <span class=\"synStatement\">def<\/span> <span class=\"synIdentifier\">set_checked_state<\/span>(list_id: <span class=\"synIdentifier\">str<\/span>, update: ToDoItemUpdate) -&gt; ToDoList:\n    <span class=\"synStatement\">return<\/span> <span class=\"synStatement\">await<\/span> app.todo_dal.set_checked_state(\n        list_id, update.item_id, update.checked_state\n    )\n\n<span class=\"synStatement\">class<\/span> <span class=\"synIdentifier\">DummyResponse<\/span>(BaseModel):\n    <span class=\"synIdentifier\">id<\/span>: <span class=\"synIdentifier\">str<\/span>\n    when: datetime\n\n<span class=\"synPreProc\">@<\/span><span class=\"synIdentifier\">app.get<\/span>(<span class=\"synConstant\">\"\/api\/dummy\"<\/span>)\n<span class=\"synStatement\">async<\/span> <span class=\"synStatement\">def<\/span> <span class=\"synIdentifier\">get_dummy<\/span>() -&gt; DummyResponse:\n    <span class=\"synStatement\">return<\/span> DummyResponse(\n        <span class=\"synIdentifier\">id<\/span>=<span class=\"synIdentifier\">str<\/span>(ObjectId()),\n        when=datetime.now(),\n    )\n\n<span class=\"synStatement\">def<\/span> <span class=\"synIdentifier\">main<\/span>(argv=sys.argv[<span class=\"synConstant\">1<\/span>:]):\n    <span class=\"synStatement\">try<\/span>:\n        uvicorn.run(<span class=\"synConstant\">\"server:app\"<\/span>, host=<span class=\"synConstant\">\"0.0.0.0\"<\/span>, port=<span class=\"synConstant\">3001<\/span>, <span class=\"synIdentifier\">reload<\/span>=DEBUG)\n    <span class=\"synStatement\">except<\/span> <span class=\"synType\">KeyboardInterrupt<\/span>:\n        <span class=\"synStatement\">pass<\/span>\n\n<span class=\"synStatement\">if<\/span> __name__ == <span class=\"synConstant\">\"__main__\"<\/span>:\n    main()\n<\/pre>\n<p>\u3053\u306e\u5b9f\u88c5\u3067\u306f\u3001FastAPI \u30b5\u30fc\u30d0\u30fc\u3092 CORS <a target=\"_blank\" class=\"keyword\" href=\"https:\/\/d.hatena.ne.jp\/keyword\/%A5%DF%A5%C9%A5%EB%A5%A6%A5%A7%A5%A2\">\u30df\u30c9\u30eb\u30a6\u30a7\u30a2<\/a>\u4ed8\u304d\u3067\u8a2d\u5b9a\u3057\u3001MongoDB \u306b\u63a5\u7d9a\u3057\u3001Todo \u30a2\u30d7\u30ea\u30b1\u30fc\u30b7\u30e7\u30f3\u7528\u306e <a target=\"_blank\" class=\"keyword\" href=\"https:\/\/d.hatena.ne.jp\/keyword\/API\">API<\/a> \u30a8\u30f3\u30c9\u30dd\u30a4\u30f3\u30c8\u3092\u5b9a\u7fa9\u3057\u307e\u3059\u3002<\/p>\n<p>\u30b9\u30c6\u30c3\u30d77: <a target=\"_blank\" class=\"keyword\" href=\"https:\/\/d.hatena.ne.jp\/keyword\/%B4%C4%B6%AD%CA%D1%BF%F4\">\u74b0\u5883\u5909\u6570<\/a>\u3092\u8a2d\u5b9a\u3059\u308b<\/p>\n<p>\u30eb\u30fc\u30c8<a target=\"_blank\" class=\"keyword\" href=\"https:\/\/d.hatena.ne.jp\/keyword\/%A5%C7%A5%A3%A5%EC%A5%AF%A5%C8\">\u30c7\u30a3\u30ec\u30af\u30c8<\/a>\u30ea\u306b\u00a0<code>.env<\/code>\u00a0\u30d5\u30a1\u30a4\u30eb\u3092\u4f5c\u6210\u3057\u3001\u4ee5\u4e0b\u306e\u5185\u5bb9\u3092\u8ffd\u52a0\u3057\u307e\u3059\u3002<code>.mongodb.net\/<\/code>\u00a0\u306e\u672b\u5c3e\u306b\u30c7\u30fc\u30bf\u30d9\u30fc\u30b9\u540d\uff08&#8221;todo&#8221;\uff09\u3092\u8ffd\u52a0\u3059\u308b\u3053\u3068\u3092\u5fd8\u308c\u306a\u3044\u3067\u304f\u3060\u3055\u3044\u3002<\/p>\n<p><span style=\"color: #ff0000\">MONGODB_<a target=\"_blank\" class=\"keyword\" href=\"https:\/\/d.hatena.ne.jp\/keyword\/URI\">URI<\/a>=&#8217;mongodb+<a target=\"_blank\" class=\"keyword\" href=\"https:\/\/d.hatena.ne.jp\/keyword\/srv\">srv<\/a>:\/\/fon2christian:<db_password>@cluster0.4fz7ql7.mongodb.net\/?appName=Cluster0&#8242;<\/db_password><\/span><\/p>\n<p>\u30b9\u30c6\u30c3\u30d78: docker-compose \u30d5\u30a1\u30a4\u30eb\u3092\u4f5c\u6210\u3059\u308b<\/p>\n<p>\u30d7\u30ed\u30b8\u30a7\u30af\u30c8\u306e\u30eb\u30fc\u30c8<a target=\"_blank\" class=\"keyword\" href=\"https:\/\/d.hatena.ne.jp\/keyword\/%A5%C7%A5%A3%A5%EC%A5%AF%A5%C8\">\u30c7\u30a3\u30ec\u30af\u30c8<\/a>\u30ea\uff08farm-stack-todo\uff09\u306b\u3001<code>compose.yml<\/code>\u00a0\u3068\u3044\u3046\u540d\u524d\u306e\u30d5\u30a1\u30a4\u30eb\u3092\u4f5c\u6210\u3057\u3001\u4ee5\u4e0b\u306e\u5185\u5bb9\u3092\u8ffd\u52a0\u3057\u307e\u3059\u3002<\/p>\n<pre class=\"code csharp\" data-lang=\"csharp\" data-unlink=\"\">name: todo-app\nservices:\n  nginx:\n    image: nginx:1.17\n    volumes:\n      - .\/nginx\/nginx.conf:\/etc\/nginx\/conf.d\/default.conf\n    ports:\n      - 8000:80\n    depends_on:\n      - backend\n      - frontend\n  frontend:\n    image: \"node:22\"\n    user: \"node\"\n    working_dir: \/home\/node\/app\n    environment:\n      - NODE_ENV=development\n      - WDS_SOCKET_PORT=0\n    volumes:\n      - .\/frontend\/:\/home\/node\/app\n    expose:\n      - \"3000\"\n    ports:\n      - \"3000:3000\"\n    command: \"npm start\"\n  backend:\n    image: todo-app\/backend\n    build: .\/backend\n    volumes:\n      - .\/backend\/:\/usr\/src\/app\n    expose:\n      - \"3001\"\n    ports:\n      - \"8001:3001\"\n    command: \"python src\/server.py\"\n    environment:\n      - DEBUG=true\n    env_file:\n      - path: .\/.env\n        required: true<\/pre>\n<p>\u30b9\u30c6\u30c3\u30d79: Nginx \u306e\u8a2d\u5b9a\u3092\u884c\u3046<\/p>\n<p>\u30d7\u30ed\u30b8\u30a7\u30af\u30c8\u306e\u30eb\u30fc\u30c8\u306b\u00a0<code>nginx<\/code>\u00a0\u3068\u3044\u3046\u540d\u524d\u306e<a target=\"_blank\" class=\"keyword\" href=\"https:\/\/d.hatena.ne.jp\/keyword\/%A5%C7%A5%A3%A5%EC%A5%AF%A5%C8\">\u30c7\u30a3\u30ec\u30af\u30c8<\/a>\u30ea\u3092\u4f5c\u6210\u3057\u307e\u3059\u3002<\/p>\n<pre class=\"code lang-jsx\" data-lang=\"jsx\" data-unlink=\"\">mkdir nginx\n<\/pre>\n<p><code>nginx<\/code>\u00a0<a target=\"_blank\" class=\"keyword\" href=\"https:\/\/d.hatena.ne.jp\/keyword\/%A5%C7%A5%A3%A5%EC%A5%AF%A5%C8\">\u30c7\u30a3\u30ec\u30af\u30c8<\/a>\u30ea\u5185\u306b\u00a0<code>nginx.conf<\/code>\u00a0\u3068\u3044\u3046\u540d\u524d\u306e\u30d5\u30a1\u30a4\u30eb\u3092\u4f5c\u6210\u3057\u3001\u4ee5\u4e0b\u306e\u5185\u5bb9\u3092\u8ffd\u52a0\u3057\u307e\u3059\u3002<\/p>\n<pre class=\"code lang-python\" data-lang=\"python\" data-unlink=\"\">server {\n    listen <span class=\"synConstant\">80<\/span>;\n    server_name farm_intro;\n\n    location \/ {\n        proxy_pass http:\/\/frontend:<span class=\"synConstant\">3000<\/span>;\n        proxy_set_header Upgrade $http_upgrade;\n        proxy_set_header Connection <span class=\"synConstant\">\"upgrade\"<\/span>;\n    }\n\n    location \/api {\n        proxy_pass http:\/\/backend:<span class=\"synConstant\">3001<\/span>\/api;\n    }\n}\n<\/pre>\n<p>\u3053\u306e\u6642\u70b9\u3067\u3001FastAPI\u30b5\u30fc\u30d0\u30fc\u306e\u5b9f\u88c5\u3001<a target=\"_blank\" class=\"keyword\" href=\"https:\/\/d.hatena.ne.jp\/keyword\/%B4%C4%B6%AD%CA%D1%BF%F4\">\u74b0\u5883\u5909\u6570<\/a>\u306e\u8a2d\u5b9a\u3001docker-compose\u30d5\u30a1\u30a4\u30eb\u306e\u4f5c\u6210\u3001Nginx\u306e\u8a2d\u5b9a\u3092\u5b8c\u4e86\u3057\u307e\u3057\u305f\u3002\u6b21\u306f\u3001FARM\u30b9\u30bf\u30c3\u30af\u306eTodo\u30a2\u30d7\u30ea\u30b1\u30fc\u30b7\u30e7\u30f3\u7528\u306bReact\u30d5\u30ed\u30f3\u30c8\u30a8\u30f3\u30c9\u306e\u30bb\u30c3\u30c8\u30a2\u30c3\u30d7\u306b\u53d6\u308a\u7d44\u307f\u307e\u3059\u3002<\/p>\n<p>\u30b9\u30c6\u30c3\u30d710: React\u30a2\u30d7\u30ea\u30b1\u30fc\u30b7\u30e7\u30f3\u3092\u4f5c\u6210\u3059\u308b<\/p>\n<p>frontend<a target=\"_blank\" class=\"keyword\" href=\"https:\/\/d.hatena.ne.jp\/keyword\/%A5%C7%A5%A3%A5%EC%A5%AF%A5%C8\">\u30c7\u30a3\u30ec\u30af\u30c8<\/a>\u30ea\u306b\u79fb\u52d5\u3057\u307e\u3059:<\/p>\n<pre class=\"code lang-jsx\" data-lang=\"jsx\" data-unlink=\"\">cd <span class=\"synStatement\">.<\/span>.<span class=\"synSpecial\">\/<\/span><span class=\"synConstant\">frontend<\/span>\n<\/pre>\n<p>reate React App\u3092\u4f7f\u3063\u3066\u65b0\u3057\u3044React\u30a2\u30d7\u30ea\u30b1\u30fc\u30b7\u30e7\u30f3\u3092\u4f5c\u6210\u3057\u307e\u3059:<\/p>\n<pre class=\"code lang-jsx\" data-lang=\"jsx\" data-unlink=\"\">npx create<span class=\"synStatement\">-<\/span>react<span class=\"synStatement\">-<\/span>app <span class=\"synStatement\">.<\/span>\n<\/pre>\n<p>\u8ffd\u52a0dependencies\u3092\u30a4\u30f3\u30b9\u30c8\u30fc\u30eb\u3057\u307e\u3059:<\/p>\n<pre class=\"code lang-jsx\" data-lang=\"jsx\" data-unlink=\"\">npm install axios react<span class=\"synStatement\">-<\/span>icons\n<\/pre>\n<p>\u30b9\u30c6\u30c3\u30d711: \u30e1\u30a4\u30f3\u306e App <a target=\"_blank\" class=\"keyword\" href=\"https:\/\/d.hatena.ne.jp\/keyword\/%A5%B3%A5%F3%A5%DD%A1%BC%A5%CD%A5%F3%A5%C8\">\u30b3\u30f3\u30dd\u30fc\u30cd\u30f3\u30c8<\/a>\u3092\u8a2d\u5b9a\u3059\u308b<\/p>\n<p><code>src\/App.js<\/code>\u00a0\u306e\u5185\u5bb9\u3092\u4ee5\u4e0b\u306e\u3082\u306e\u306b\u7f6e\u304d\u63db\u3048\u307e\u3059:<\/p>\n<pre class=\"code lang-jsx\" data-lang=\"jsx\" data-unlink=\"\"><span class=\"synPreProc\">import<\/span> <span class=\"synSpecial\">{<\/span> useEffect<span class=\"synStatement\">,<\/span> useState <span class=\"synSpecial\">}<\/span> <span class=\"synPreProc\">from<\/span> <span class=\"synConstant\">\"react\"<\/span><span class=\"synStatement\">;<\/span>\n<span class=\"synPreProc\">import<\/span> axios <span class=\"synPreProc\">from<\/span> <span class=\"synConstant\">\"axios\"<\/span><span class=\"synStatement\">;<\/span>\n<span class=\"synPreProc\">import<\/span> <span class=\"synConstant\">\".\/App.css\"<\/span><span class=\"synStatement\">;<\/span>\n<span class=\"synPreProc\">import<\/span> ListToDoLists <span class=\"synPreProc\">from<\/span> <span class=\"synConstant\">\".\/ListTodoLists\"<\/span><span class=\"synStatement\">;<\/span>\n<span class=\"synPreProc\">import<\/span> ToDoList <span class=\"synPreProc\">from<\/span> <span class=\"synConstant\">\".\/ToDoList\"<\/span><span class=\"synStatement\">;<\/span>\n\n<span class=\"synStatement\">function<\/span> <span class=\"synIdentifier\">App<\/span><span class=\"synSpecial\">()<\/span> <span class=\"synSpecial\">{<\/span>\n  <span class=\"synType\">const<\/span> <span class=\"synSpecial\">[<\/span>listSummaries<span class=\"synStatement\">,<\/span> setListSummaries<span class=\"synSpecial\">]<\/span> <span class=\"synStatement\">=<\/span> <span class=\"synIdentifier\">useState<\/span><span class=\"synSpecial\">(<\/span><span class=\"synConstant\">null<\/span><span class=\"synSpecial\">)<\/span><span class=\"synStatement\">;<\/span>\n  <span class=\"synType\">const<\/span> <span class=\"synSpecial\">[<\/span>selectedItem<span class=\"synStatement\">,<\/span> setSelectedItem<span class=\"synSpecial\">]<\/span> <span class=\"synStatement\">=<\/span> <span class=\"synIdentifier\">useState<\/span><span class=\"synSpecial\">(<\/span><span class=\"synConstant\">null<\/span><span class=\"synSpecial\">)<\/span><span class=\"synStatement\">;<\/span>\n\n  <span class=\"synIdentifier\">useEffect<\/span><span class=\"synSpecial\">(()<\/span> <span class=\"synStatement\">=&gt;<\/span> <span class=\"synSpecial\">{<\/span>\n    <span class=\"synIdentifier\">reloadData<\/span><span class=\"synSpecial\">()<\/span><span class=\"synStatement\">.<\/span><span class=\"synIdentifier\">catch<\/span><span class=\"synSpecial\">(<\/span><span class=\"synConstant\">console<\/span><span class=\"synStatement\">.<\/span>error<span class=\"synSpecial\">)<\/span><span class=\"synStatement\">;<\/span>\n  <span class=\"synSpecial\">}<\/span><span class=\"synStatement\">,<\/span> <span class=\"synSpecial\">[])<\/span><span class=\"synStatement\">;<\/span>\n\n  <span class=\"synStatement\">async<\/span> <span class=\"synStatement\">function<\/span> <span class=\"synIdentifier\">reloadData<\/span><span class=\"synSpecial\">()<\/span> <span class=\"synSpecial\">{<\/span>\n    <span class=\"synType\">const<\/span> response <span class=\"synStatement\">=<\/span> <span class=\"synStatement\">await<\/span> axios<span class=\"synStatement\">.<\/span><span class=\"synIdentifier\">get<\/span><span class=\"synSpecial\">(<\/span><span class=\"synConstant\">\"\/api\/lists\"<\/span><span class=\"synSpecial\">)<\/span><span class=\"synStatement\">;<\/span>\n    <span class=\"synType\">const<\/span> data <span class=\"synStatement\">=<\/span> <span class=\"synStatement\">await<\/span> response<span class=\"synStatement\">.<\/span>data<span class=\"synStatement\">;<\/span>\n    <span class=\"synIdentifier\">setListSummaries<\/span><span class=\"synSpecial\">(<\/span>data<span class=\"synSpecial\">)<\/span><span class=\"synStatement\">;<\/span>\n  <span class=\"synSpecial\">}<\/span>\n\n  <span class=\"synStatement\">function<\/span> <span class=\"synIdentifier\">handleNewToDoList<\/span><span class=\"synSpecial\">(<\/span>newName<span class=\"synSpecial\">)<\/span> <span class=\"synSpecial\">{<\/span>\n    <span class=\"synType\">const<\/span> updateData <span class=\"synStatement\">=<\/span> <span class=\"synStatement\">async<\/span> <span class=\"synSpecial\">()<\/span> <span class=\"synStatement\">=&gt;<\/span> <span class=\"synSpecial\">{<\/span>\n      <span class=\"synType\">const<\/span> newListData <span class=\"synStatement\">=<\/span> <span class=\"synSpecial\">{<\/span>\n        <span class=\"synIdentifier\">name<\/span><span class=\"synStatement\">:<\/span> newName<span class=\"synStatement\">,<\/span>\n      <span class=\"synSpecial\">}<\/span><span class=\"synStatement\">;<\/span>\n\n      <span class=\"synStatement\">await<\/span> axios<span class=\"synStatement\">.<\/span><span class=\"synIdentifier\">post<\/span><span class=\"synSpecial\">(<\/span><span class=\"synConstant\">`\/api\/lists`<\/span><span class=\"synStatement\">,<\/span> newListData<span class=\"synSpecial\">)<\/span><span class=\"synStatement\">;<\/span>\n      <span class=\"synIdentifier\">reloadData<\/span><span class=\"synSpecial\">()<\/span><span class=\"synStatement\">.<\/span><span class=\"synIdentifier\">catch<\/span><span class=\"synSpecial\">(<\/span><span class=\"synConstant\">console<\/span><span class=\"synStatement\">.<\/span>error<span class=\"synSpecial\">)<\/span><span class=\"synStatement\">;<\/span>\n    <span class=\"synSpecial\">}<\/span><span class=\"synStatement\">;<\/span>\n    <span class=\"synIdentifier\">updateData<\/span><span class=\"synSpecial\">()<\/span><span class=\"synStatement\">;<\/span>\n  <span class=\"synSpecial\">}<\/span>\n\n  <span class=\"synStatement\">function<\/span> <span class=\"synIdentifier\">handleDeleteToDoList<\/span><span class=\"synSpecial\">(<\/span>id<span class=\"synSpecial\">)<\/span> <span class=\"synSpecial\">{<\/span>\n    <span class=\"synType\">const<\/span> updateData <span class=\"synStatement\">=<\/span> <span class=\"synStatement\">async<\/span> <span class=\"synSpecial\">()<\/span> <span class=\"synStatement\">=&gt;<\/span> <span class=\"synSpecial\">{<\/span>\n      <span class=\"synStatement\">await<\/span> axios<span class=\"synStatement\">.<\/span><span class=\"synIdentifier\">delete<\/span><span class=\"synSpecial\">(<\/span><span class=\"synConstant\">`\/api\/lists\/<\/span><span class=\"synSpecial\">${<\/span>id<span class=\"synSpecial\">}<\/span><span class=\"synConstant\">`<\/span><span class=\"synSpecial\">)<\/span><span class=\"synStatement\">;<\/span>\n      <span class=\"synIdentifier\">reloadData<\/span><span class=\"synSpecial\">()<\/span><span class=\"synStatement\">.<\/span><span class=\"synIdentifier\">catch<\/span><span class=\"synSpecial\">(<\/span><span class=\"synConstant\">console<\/span><span class=\"synStatement\">.<\/span>error<span class=\"synSpecial\">)<\/span><span class=\"synStatement\">;<\/span>\n    <span class=\"synSpecial\">}<\/span><span class=\"synStatement\">;<\/span>\n    <span class=\"synIdentifier\">updateData<\/span><span class=\"synSpecial\">()<\/span><span class=\"synStatement\">;<\/span>\n  <span class=\"synSpecial\">}<\/span>\n\n  <span class=\"synStatement\">function<\/span> <span class=\"synIdentifier\">handleSelectList<\/span><span class=\"synSpecial\">(<\/span>id<span class=\"synSpecial\">)<\/span> <span class=\"synSpecial\">{<\/span>\n    <span class=\"synConstant\">console<\/span><span class=\"synStatement\">.<\/span><span class=\"synIdentifier\">log<\/span><span class=\"synSpecial\">(<\/span><span class=\"synConstant\">\"Selecting item\"<\/span><span class=\"synStatement\">,<\/span> id<span class=\"synSpecial\">)<\/span><span class=\"synStatement\">;<\/span>\n    <span class=\"synIdentifier\">setSelectedItem<\/span><span class=\"synSpecial\">(<\/span>id<span class=\"synSpecial\">)<\/span><span class=\"synStatement\">;<\/span>\n  <span class=\"synSpecial\">}<\/span>\n\n  <span class=\"synStatement\">function<\/span> <span class=\"synIdentifier\">backToList<\/span><span class=\"synSpecial\">()<\/span> <span class=\"synSpecial\">{<\/span>\n    <span class=\"synIdentifier\">setSelectedItem<\/span><span class=\"synSpecial\">(<\/span><span class=\"synConstant\">null<\/span><span class=\"synSpecial\">)<\/span><span class=\"synStatement\">;<\/span>\n    <span class=\"synIdentifier\">reloadData<\/span><span class=\"synSpecial\">()<\/span><span class=\"synStatement\">.<\/span><span class=\"synIdentifier\">catch<\/span><span class=\"synSpecial\">(<\/span><span class=\"synConstant\">console<\/span><span class=\"synStatement\">.<\/span>error<span class=\"synSpecial\">)<\/span><span class=\"synStatement\">;<\/span>\n  <span class=\"synSpecial\">}<\/span>\n\n  <span class=\"synStatement\">if<\/span> <span class=\"synSpecial\">(<\/span>selectedItem <span class=\"synStatement\">===<\/span> <span class=\"synConstant\">null<\/span><span class=\"synSpecial\">)<\/span> <span class=\"synSpecial\">{<\/span>\n    <span class=\"synStatement\">return<\/span> <span class=\"synSpecial\">(<\/span>\n      <\/pre>\n<p>\u30b9\u30c6\u30c3\u30d712: ListTodoLists <a target=\"_blank\" class=\"keyword\" href=\"https:\/\/d.hatena.ne.jp\/keyword\/%A5%B3%A5%F3%A5%DD%A1%BC%A5%CD%A5%F3%A5%C8\">\u30b3\u30f3\u30dd\u30fc\u30cd\u30f3\u30c8<\/a>\u3092\u4f5c\u6210\u3059\u308b<\/p>\n<p><code>src\/ListTodoLists.js<\/code>\u00a0\u3068\u3044\u3046\u65b0\u3057\u3044\u30d5\u30a1\u30a4\u30eb\u3092\u4f5c\u6210\u3057\u3001\u4ee5\u4e0b\u306e\u5185\u5bb9\u3092\u8ffd\u52a0\u3057\u307e\u3059:<\/p>\n<pre class=\"code lang-jsx\" data-lang=\"jsx\" data-unlink=\"\"><span class=\"synPreProc\">import<\/span> <span class=\"synConstant\">\".\/ListTodoLists.css\"<\/span><span class=\"synStatement\">;<\/span>\n<span class=\"synPreProc\">import<\/span> <span class=\"synSpecial\">{<\/span> useRef <span class=\"synSpecial\">}<\/span> <span class=\"synPreProc\">from<\/span> <span class=\"synConstant\">\"react\"<\/span><span class=\"synStatement\">;<\/span>\n<span class=\"synPreProc\">import<\/span> <span class=\"synSpecial\">{<\/span> BiSolidTrash <span class=\"synSpecial\">}<\/span> <span class=\"synPreProc\">from<\/span> <span class=\"synConstant\">\"react-icons\/bi\"<\/span><span class=\"synStatement\">;<\/span>\n\n<span class=\"synStatement\">function<\/span> <span class=\"synIdentifier\">ListToDoLists<\/span><span class=\"synSpecial\">({<\/span>\n  listSummaries<span class=\"synStatement\">,<\/span>\n  handleSelectList<span class=\"synStatement\">,<\/span>\n  handleNewToDoList<span class=\"synStatement\">,<\/span>\n  handleDeleteToDoList<span class=\"synStatement\">,<\/span>\n<span class=\"synSpecial\">})<\/span> <span class=\"synSpecial\">{<\/span>\n  <span class=\"synType\">const<\/span> labelRef <span class=\"synStatement\">=<\/span> <span class=\"synIdentifier\">useRef<\/span><span class=\"synSpecial\">()<\/span><span class=\"synStatement\">;<\/span>\n\n  <span class=\"synStatement\">if<\/span> <span class=\"synSpecial\">(<\/span>listSummaries <span class=\"synStatement\">===<\/span> <span class=\"synConstant\">null<\/span><span class=\"synSpecial\">)<\/span> <span class=\"synSpecial\">{<\/span>\n    <span class=\"synStatement\">return<\/span> <\/pre>\n<p><code>src\/ListTodoLists.css<\/code>\u00a0\u3068\u3044\u3046\u65b0\u3057\u3044\u30d5\u30a1\u30a4\u30eb\u3092\u4f5c\u6210\u3057\u3001\u4ee5\u4e0b\u306e\u5185\u5bb9\u3092\u8ffd\u52a0\u3057\u307e\u3059:<\/p>\n<pre class=\"code lang-css\" data-lang=\"css\" data-unlink=\"\"><span class=\"synIdentifier\">.ListToDoLists<\/span> <span class=\"synIdentifier\">.summary<\/span> <span class=\"synIdentifier\">{<\/span>\n    <span class=\"synType\">border<\/span>: <span class=\"synConstant\">1px<\/span> <span class=\"synConstant\">solid<\/span> <span class=\"synConstant\">lightgray<\/span>;\n    <span class=\"synType\">padding<\/span>: <span class=\"synConstant\">1em<\/span>;\n    <span class=\"synType\">margin<\/span>: <span class=\"synConstant\">1em<\/span>;\n    <span class=\"synType\">cursor<\/span>: <span class=\"synConstant\">pointer<\/span>;\n    <span class=\"synType\">display<\/span>: <span class=\"synConstant\">flex<\/span>;\n<span class=\"synIdentifier\">}<\/span>\n\n<span class=\"synIdentifier\">.ListToDoLists<\/span> <span class=\"synIdentifier\">.count<\/span> <span class=\"synIdentifier\">{<\/span>\n    <span class=\"synType\">padding-left<\/span>: <span class=\"synConstant\">1ex<\/span>;\n    <span class=\"synType\">color<\/span>: <span class=\"synConstant\">blueviolet<\/span>;\n    <span class=\"synType\">font-size<\/span>: <span class=\"synConstant\">92%<\/span>;\n<span class=\"synIdentifier\">}<\/span>\n<\/pre>\n<p><code>src\/ToDoList.js<\/code>\u00a0\u3068\u3044\u3046\u65b0\u3057\u3044\u30d5\u30a1\u30a4\u30eb\u3092\u4f5c\u6210\u3057\u3001\u4ee5\u4e0b\u306e\u5185\u5bb9\u3092\u8ffd\u52a0\u3057\u307e\u3059:<\/p>\n<pre class=\"code lang-jsx\" data-lang=\"jsx\" data-unlink=\"\"><span class=\"synPreProc\">import<\/span> <span class=\"synConstant\">\".\/ToDoList.css\"<\/span><span class=\"synStatement\">;<\/span>\n<span class=\"synPreProc\">import<\/span> <span class=\"synSpecial\">{<\/span> useEffect<span class=\"synStatement\">,<\/span> useState<span class=\"synStatement\">,<\/span> useRef <span class=\"synSpecial\">}<\/span> <span class=\"synPreProc\">from<\/span> <span class=\"synConstant\">\"react\"<\/span><span class=\"synStatement\">;<\/span>\n<span class=\"synPreProc\">import<\/span> axios <span class=\"synPreProc\">from<\/span> <span class=\"synConstant\">\"axios\"<\/span><span class=\"synStatement\">;<\/span>\n<span class=\"synPreProc\">import<\/span> <span class=\"synSpecial\">{<\/span> BiSolidTrash <span class=\"synSpecial\">}<\/span> <span class=\"synPreProc\">from<\/span> <span class=\"synConstant\">\"react-icons\/bi\"<\/span><span class=\"synStatement\">;<\/span>\n\n<span class=\"synStatement\">function<\/span> <span class=\"synIdentifier\">ToDoList<\/span><span class=\"synSpecial\">({<\/span> listId<span class=\"synStatement\">,<\/span> handleBackButton <span class=\"synSpecial\">})<\/span> <span class=\"synSpecial\">{<\/span>\n  <span class=\"synType\">let<\/span> labelRef <span class=\"synStatement\">=<\/span> <span class=\"synIdentifier\">useRef<\/span><span class=\"synSpecial\">()<\/span><span class=\"synStatement\">;<\/span>\n  <span class=\"synType\">const<\/span> <span class=\"synSpecial\">[<\/span>listData<span class=\"synStatement\">,<\/span> setListData<span class=\"synSpecial\">]<\/span> <span class=\"synStatement\">=<\/span> <span class=\"synIdentifier\">useState<\/span><span class=\"synSpecial\">(<\/span><span class=\"synConstant\">null<\/span><span class=\"synSpecial\">)<\/span><span class=\"synStatement\">;<\/span>\n\n  <span class=\"synIdentifier\">useEffect<\/span><span class=\"synSpecial\">(()<\/span> <span class=\"synStatement\">=&gt;<\/span> <span class=\"synSpecial\">{<\/span>\n    <span class=\"synType\">const<\/span> fetchData <span class=\"synStatement\">=<\/span> <span class=\"synStatement\">async<\/span> <span class=\"synSpecial\">()<\/span> <span class=\"synStatement\">=&gt;<\/span> <span class=\"synSpecial\">{<\/span>\n      <span class=\"synType\">const<\/span> response <span class=\"synStatement\">=<\/span> <span class=\"synStatement\">await<\/span> axios<span class=\"synStatement\">.<\/span><span class=\"synIdentifier\">get<\/span><span class=\"synSpecial\">(<\/span><span class=\"synConstant\">`\/api\/lists\/<\/span><span class=\"synSpecial\">${<\/span>listId<span class=\"synSpecial\">}<\/span><span class=\"synConstant\">`<\/span><span class=\"synSpecial\">)<\/span><span class=\"synStatement\">;<\/span>\n      <span class=\"synType\">const<\/span> newData <span class=\"synStatement\">=<\/span> <span class=\"synStatement\">await<\/span> response<span class=\"synStatement\">.<\/span>data<span class=\"synStatement\">;<\/span>\n      <span class=\"synIdentifier\">setListData<\/span><span class=\"synSpecial\">(<\/span>newData<span class=\"synSpecial\">)<\/span><span class=\"synStatement\">;<\/span>\n    <span class=\"synSpecial\">}<\/span><span class=\"synStatement\">;<\/span>\n    <span class=\"synIdentifier\">fetchData<\/span><span class=\"synSpecial\">()<\/span><span class=\"synStatement\">;<\/span>\n  <span class=\"synSpecial\">}<\/span><span class=\"synStatement\">,<\/span> <span class=\"synSpecial\">[<\/span>listId<span class=\"synSpecial\">])<\/span><span class=\"synStatement\">;<\/span>\n\n  <span class=\"synStatement\">function<\/span> <span class=\"synIdentifier\">handleCreateItem<\/span><span class=\"synSpecial\">(<\/span>label<span class=\"synSpecial\">)<\/span> <span class=\"synSpecial\">{<\/span>\n    <span class=\"synType\">const<\/span> updateData <span class=\"synStatement\">=<\/span> <span class=\"synStatement\">async<\/span> <span class=\"synSpecial\">()<\/span> <span class=\"synStatement\">=&gt;<\/span> <span class=\"synSpecial\">{<\/span>\n      <span class=\"synType\">const<\/span> response <span class=\"synStatement\">=<\/span> <span class=\"synStatement\">await<\/span> axios<span class=\"synStatement\">.<\/span><span class=\"synIdentifier\">post<\/span><span class=\"synSpecial\">(<\/span><span class=\"synConstant\">`\/api\/lists\/<\/span><span class=\"synSpecial\">${<\/span>listData<span class=\"synStatement\">.<\/span>id<span class=\"synSpecial\">}<\/span><span class=\"synConstant\">\/items\/`<\/span><span class=\"synStatement\">,<\/span> <span class=\"synSpecial\">{<\/span>\n        <span class=\"synIdentifier\">label<\/span><span class=\"synStatement\">:<\/span> label<span class=\"synStatement\">,<\/span>\n      <span class=\"synSpecial\">})<\/span><span class=\"synStatement\">;<\/span>\n      <span class=\"synIdentifier\">setListData<\/span><span class=\"synSpecial\">(<\/span><span class=\"synStatement\">await<\/span> response<span class=\"synStatement\">.<\/span>data<span class=\"synSpecial\">)<\/span><span class=\"synStatement\">;<\/span>\n    <span class=\"synSpecial\">}<\/span><span class=\"synStatement\">;<\/span>\n    <span class=\"synIdentifier\">updateData<\/span><span class=\"synSpecial\">()<\/span><span class=\"synStatement\">;<\/span>\n  <span class=\"synSpecial\">}<\/span>\n\n  <span class=\"synStatement\">function<\/span> <span class=\"synIdentifier\">handleDeleteItem<\/span><span class=\"synSpecial\">(<\/span>id<span class=\"synSpecial\">)<\/span> <span class=\"synSpecial\">{<\/span>\n    <span class=\"synType\">const<\/span> updateData <span class=\"synStatement\">=<\/span> <span class=\"synStatement\">async<\/span> <span class=\"synSpecial\">()<\/span> <span class=\"synStatement\">=&gt;<\/span> <span class=\"synSpecial\">{<\/span>\n      <span class=\"synType\">const<\/span> response <span class=\"synStatement\">=<\/span> <span class=\"synStatement\">await<\/span> axios<span class=\"synStatement\">.<\/span><span class=\"synIdentifier\">delete<\/span><span class=\"synSpecial\">(<\/span>\n        <span class=\"synConstant\">`\/api\/lists\/<\/span><span class=\"synSpecial\">${<\/span>listData<span class=\"synStatement\">.<\/span>id<span class=\"synSpecial\">}<\/span><span class=\"synConstant\">\/items\/<\/span><span class=\"synSpecial\">${<\/span>id<span class=\"synSpecial\">}<\/span><span class=\"synConstant\">`<\/span><span class=\"synSpecial\">)<\/span><span class=\"synStatement\">;<\/span>\n      <span class=\"synIdentifier\">setListData<\/span><span class=\"synSpecial\">(<\/span><span class=\"synStatement\">await<\/span> response<span class=\"synStatement\">.<\/span>data<span class=\"synSpecial\">)<\/span><span class=\"synStatement\">;<\/span>\n    <span class=\"synSpecial\">}<\/span><span class=\"synStatement\">;<\/span>\n    <span class=\"synIdentifier\">updateData<\/span><span class=\"synSpecial\">()<\/span><span class=\"synStatement\">;<\/span>\n  <span class=\"synSpecial\">}<\/span>\n\n  <span class=\"synStatement\">function<\/span> <span class=\"synIdentifier\">handleCheckToggle<\/span><span class=\"synSpecial\">(<\/span>itemId<span class=\"synStatement\">,<\/span> newState<span class=\"synSpecial\">)<\/span> <span class=\"synSpecial\">{<\/span>\n    <span class=\"synType\">const<\/span> updateData <span class=\"synStatement\">=<\/span> <span class=\"synStatement\">async<\/span> <span class=\"synSpecial\">()<\/span> <span class=\"synStatement\">=&gt;<\/span> <span class=\"synSpecial\">{<\/span>\n      <span class=\"synType\">const<\/span> response <span class=\"synStatement\">=<\/span> <span class=\"synStatement\">await<\/span> axios<span class=\"synStatement\">.<\/span><span class=\"synIdentifier\">patch<\/span><span class=\"synSpecial\">(<\/span>\n        <span class=\"synConstant\">`\/api\/lists\/<\/span><span class=\"synSpecial\">${<\/span>listData<span class=\"synStatement\">.<\/span>id<span class=\"synSpecial\">}<\/span><span class=\"synConstant\">\/checked_state`<\/span><span class=\"synStatement\">,<\/span>\n        <span class=\"synSpecial\">{<\/span>\n          <span class=\"synIdentifier\">item_id<\/span><span class=\"synStatement\">:<\/span> itemId<span class=\"synStatement\">,<\/span>\n          <span class=\"synIdentifier\">checked_state<\/span><span class=\"synStatement\">:<\/span> newState<span class=\"synStatement\">,<\/span>\n        <span class=\"synSpecial\">}<\/span>\n      <span class=\"synSpecial\">)<\/span><span class=\"synStatement\">;<\/span>\n      <span class=\"synIdentifier\">setListData<\/span><span class=\"synSpecial\">(<\/span><span class=\"synStatement\">await<\/span> response<span class=\"synStatement\">.<\/span>data<span class=\"synSpecial\">)<\/span><span class=\"synStatement\">;<\/span>\n    <span class=\"synSpecial\">}<\/span><span class=\"synStatement\">;<\/span>\n    <span class=\"synIdentifier\">updateData<\/span><span class=\"synSpecial\">()<\/span><span class=\"synStatement\">;<\/span>\n  <span class=\"synSpecial\">}<\/span>\n\n  <span class=\"synStatement\">if<\/span> <span class=\"synSpecial\">(<\/span>listData <span class=\"synStatement\">===<\/span> <span class=\"synConstant\">null<\/span><span class=\"synSpecial\">)<\/span> <span class=\"synSpecial\">{<\/span>\n    <span class=\"synStatement\">return<\/span> <span class=\"synSpecial\">(<\/span>\n      <\/pre>\n<p><code>src\/ToDoList.css<\/code>\u00a0\u3068\u3044\u3046\u65b0\u3057\u3044\u30d5\u30a1\u30a4\u30eb\u3092\u4f5c\u6210\u3057\u3001\u4ee5\u4e0b\u306e\u5185\u5bb9\u3092\u8ffd\u52a0\u3057\u307e\u3059:<\/p>\n<pre class=\"code lang-css\" data-lang=\"css\" data-unlink=\"\"><span class=\"synIdentifier\">.ToDoList<\/span> <span class=\"synIdentifier\">.back<\/span> <span class=\"synIdentifier\">{<\/span>\n    <span class=\"synType\">margin<\/span>: <span class=\"synConstant\">0<\/span> <span class=\"synConstant\">1em<\/span>;\n    <span class=\"synType\">padding<\/span>: <span class=\"synConstant\">1em<\/span>;\n    <span class=\"synType\">float<\/span>: <span class=\"synPreProc\">left<\/span>;\n<span class=\"synIdentifier\">}<\/span>\n\n<span class=\"synIdentifier\">.ToDoList<\/span> <span class=\"synIdentifier\">.item<\/span> <span class=\"synIdentifier\">{<\/span>\n    <span class=\"synType\">border<\/span>: <span class=\"synConstant\">1px<\/span> <span class=\"synConstant\">solid<\/span> <span class=\"synConstant\">lightgray<\/span>;\n    <span class=\"synType\">padding<\/span>: <span class=\"synConstant\">1em<\/span>;\n    <span class=\"synType\">margin<\/span>: <span class=\"synConstant\">1em<\/span>;\n    <span class=\"synType\">cursor<\/span>: <span class=\"synConstant\">pointer<\/span>;\n    <span class=\"synType\">display<\/span>: <span class=\"synConstant\">flex<\/span>;\n<span class=\"synIdentifier\">}<\/span>\n\n<span class=\"synIdentifier\">.ToDoList<\/span> <span class=\"synIdentifier\">.label<\/span> <span class=\"synIdentifier\">{<\/span>\n    <span class=\"synType\">margin-left<\/span>: <span class=\"synConstant\">1ex<\/span>;\n<span class=\"synIdentifier\">}<\/span>\n\n<span class=\"synIdentifier\">.ToDoList<\/span> <span class=\"synIdentifier\">.checked<\/span> <span class=\"synIdentifier\">.label<\/span> <span class=\"synIdentifier\">{<\/span>\n    <span class=\"synType\">text-decoration<\/span>: <span class=\"synConstant\">line-through<\/span>;\n    <span class=\"synType\">color<\/span>: <span class=\"synConstant\">lightgray<\/span>;\n<span class=\"synIdentifier\">}<\/span>\n<\/pre>\n<p><code>src\/index.css<\/code>\u00a0\u306e\u5185\u5bb9\u3092\u4ee5\u4e0b\u306e\u3088\u3046\u306b\u7f6e\u304d\u63db\u3048\u307e\u3059:<\/p>\n<pre class=\"code lang-css\" data-lang=\"css\" data-unlink=\"\"><span class=\"synStatement\">html<\/span><span class=\"synSpecial\">,<\/span> <span class=\"synStatement\">body<\/span> <span class=\"synIdentifier\">{<\/span>\n  <span class=\"synType\">margin<\/span>: <span class=\"synConstant\">0<\/span>;\n  <span class=\"synType\">font-family<\/span>: -apple-system<span class=\"synSpecial\">,<\/span> BlinkMacSystemFont<span class=\"synSpecial\">,<\/span> <span class=\"synConstant\">'Segoe UI'<\/span><span class=\"synSpecial\">,<\/span> <span class=\"synConstant\">'Roboto'<\/span><span class=\"synSpecial\">,<\/span> <span class=\"synConstant\">'Oxygen'<\/span><span class=\"synSpecial\">,<\/span>\n    <span class=\"synConstant\">'Ubuntu'<\/span><span class=\"synSpecial\">,<\/span> <span class=\"synConstant\">'Cantarell'<\/span><span class=\"synSpecial\">,<\/span> <span class=\"synConstant\">'Fira Sans'<\/span><span class=\"synSpecial\">,<\/span> <span class=\"synConstant\">'Droid Sans'<\/span><span class=\"synSpecial\">,<\/span> <span class=\"synConstant\">'Helvetica Neue'<\/span><span class=\"synSpecial\">,<\/span>\n    <span class=\"synConstant\">sans-serif<\/span>;\n  <span class=\"synType\">font<\/span>-smoothing: <span class=\"synConstant\">antialiased<\/span>;\n  osx-<span class=\"synType\">font<\/span>-smoothing: grayscale;\n  <span class=\"synType\">font-size<\/span>: <span class=\"synConstant\">12pt<\/span>;\n<span class=\"synIdentifier\">}<\/span>\n\n<span class=\"synStatement\">input<\/span><span class=\"synSpecial\">,<\/span> <span class=\"synStatement\">button<\/span> <span class=\"synIdentifier\">{<\/span>\n  <span class=\"synType\">font-size<\/span>: <span class=\"synConstant\">1em<\/span>;\n<span class=\"synIdentifier\">}<\/span>\n\n<span class=\"synStatement\">code<\/span> <span class=\"synIdentifier\">{<\/span>\n  <span class=\"synType\">font-family<\/span>: source-<span class=\"synConstant\">code<\/span>-pro<span class=\"synSpecial\">,<\/span> Menlo<span class=\"synSpecial\">,<\/span> <span class=\"synConstant\">Monaco<\/span><span class=\"synSpecial\">,<\/span> Consolas<span class=\"synSpecial\">,<\/span> <span class=\"synConstant\">'Courier New'<\/span><span class=\"synSpecial\">,<\/span>\n    <span class=\"synConstant\">monospace<\/span>;\n<span class=\"synIdentifier\">}<\/span>\n\n<span class=\"synIdentifier\">.box<\/span> <span class=\"synIdentifier\">{<\/span>\n    <span class=\"synType\">border<\/span>: <span class=\"synConstant\">1px<\/span> <span class=\"synConstant\">solid<\/span> <span class=\"synConstant\">lightgray<\/span>;\n    <span class=\"synType\">padding<\/span>: <span class=\"synConstant\">1em<\/span>;\n    <span class=\"synType\">margin<\/span>: <span class=\"synConstant\">1em<\/span>;\n<span class=\"synIdentifier\">}<\/span>\n\n<span class=\"synIdentifier\">.flex<\/span> <span class=\"synIdentifier\">{<\/span>\n  <span class=\"synType\">flex<\/span>: <span class=\"synConstant\">1<\/span>;\n<span class=\"synIdentifier\">}<\/span>\n<\/pre>\n<p>\u3053\u308c\u3067\u3001FARM\u30b9\u30bf\u30c3\u30af\u306eTodo\u30a2\u30d7\u30ea\u30b1\u30fc\u30b7\u30e7\u30f3\u7528\u306eReact\u30d5\u30ed\u30f3\u30c8\u30a8\u30f3\u30c9\u306e\u30bb\u30c3\u30c8\u30a2\u30c3\u30d7\u306f\u5b8c\u4e86\u3067\u3059\u3002\u30e1\u30a4\u30f3\u306e\u00a0<code>App<\/code>\u00a0<a target=\"_blank\" class=\"keyword\" href=\"https:\/\/d.hatena.ne.jp\/keyword\/%A5%B3%A5%F3%A5%DD%A1%BC%A5%CD%A5%F3%A5%C8\">\u30b3\u30f3\u30dd\u30fc\u30cd\u30f3\u30c8<\/a>\u3001\u5168\u3066\u306eTodo\u30ea\u30b9\u30c8\u3092\u8868\u793a\u3059\u308b\u00a0<code>ListTodoLists<\/code>\u00a0<a target=\"_blank\" class=\"keyword\" href=\"https:\/\/d.hatena.ne.jp\/keyword\/%A5%B3%A5%F3%A5%DD%A1%BC%A5%CD%A5%F3%A5%C8\">\u30b3\u30f3\u30dd\u30fc\u30cd\u30f3\u30c8<\/a>\u3001\u500b\u5225\u306eTodo\u30ea\u30b9\u30c8\u7528\u306e\u00a0<code>ToDoList<\/code>\u00a0<a target=\"_blank\" class=\"keyword\" href=\"https:\/\/d.hatena.ne.jp\/keyword\/%A5%B3%A5%F3%A5%DD%A1%BC%A5%CD%A5%F3%A5%C8\">\u30b3\u30f3\u30dd\u30fc\u30cd\u30f3\u30c8<\/a>\u3092\u4f5c\u6210\u3057\u307e\u3057\u305f\u3002\u6b21\u306b\u3001\u30a2\u30d7\u30ea\u30b1\u30fc\u30b7\u30e7\u30f3\u3092\u5b9f\u884c\u3057\u3066\u307f\u307e\u3057\u3087\u3046\u3002<\/p>\n<p>\u30b9\u30c6\u30c3\u30d718\uff1aDocker Compose\u3092\u4f7f\u3063\u3066\u30a2\u30d7\u30ea\u30b1\u30fc\u30b7\u30e7\u30f3\u3092\u5b9f\u884c\u3059\u308b<\/p>\n<ol>\n<li>\u30b7\u30b9\u30c6\u30e0\u306bDocker\u3068Docker Compose\u304c\u30a4\u30f3\u30b9\u30c8\u30fc\u30eb\u3055\u308c\u3066\u3044\u308b\u3053\u3068\u3092\u78ba\u8a8d\u3057\u307e\u3059<\/li>\n<li>\u30d7\u30ed\u30b8\u30a7\u30af\u30c8\u306e\u30eb\u30fc\u30c8<a target=\"_blank\" class=\"keyword\" href=\"https:\/\/d.hatena.ne.jp\/keyword\/%A5%C7%A5%A3%A5%EC%A5%AF%A5%C8\">\u30c7\u30a3\u30ec\u30af\u30c8<\/a>\u30ea\uff08<code>farm-stack-todo<\/code>\uff09\u3067\u30bf\u30fc\u30df\u30ca\u30eb\u3092\u958b\u304d\u307e\u3059<\/li>\n<li>\u30b3\u30f3\u30c6\u30ca\u3092\u30d3\u30eb\u30c9\u3057\u3066\u8d77\u52d5\u3057\u307e\u3059:<\/li>\n<\/ol>\n<pre class=\"code lang-jsx\" data-lang=\"jsx\" data-unlink=\"\">docker<span class=\"synStatement\">-<\/span>compose up <span class=\"synStatement\">--<\/span>build\n<\/pre>\n<ol>\n<li>\u30b3\u30f3\u30c6\u30ca\u304c\u8d77\u52d5\u3057\u305f\u3089\u3001\u30a6\u30a7\u30d6\u30d6\u30e9\u30a6\u30b6\u3092\u958b\u304d\u3001<a target=\"_blank\" href=\"http:\/\/localhost:8000\/\">http:\/\/localhost:8000<\/a>\u00a0\u306b\u30a2\u30af\u30bb\u30b9\u3057\u307e\u3059<\/li>\n<li>\u30b3\u30f3\u30c6\u30ca\u3092\u505c\u6b62\u3057\u3066\u524a\u9664\u3059\u308b\u306b\u306f\u3001\u4ee5\u4e0b\u306e\u30b3\u30de\u30f3\u30c9\u3092\u5b9f\u884c\u3057\u307e\u3059:<\/li>\n<\/ol>\n<pre class=\"code lang-jsx\" data-lang=\"jsx\" data-unlink=\"\">docker<span class=\"synStatement\">-<\/span>compose down\n<\/pre>\n<p>\u7d42\u308f\u308a<\/p>\n<p>\u3053\u306eTodo\u30a2\u30d7\u30ea\u30b1\u30fc\u30b7\u30e7\u30f3\u3092\u69cb\u7bc9\u3059\u308b\u3053\u3068\u3067\u3001\u73fe\u4ee3\u306e\u30a6\u30a7\u30d6\u958b\u767a\u3067\u6700\u3082\u5f37\u529b\u304b\u3064\u4eba\u6c17\u306e\u3042\u308b\u6280\u8853\u3092\u5b9f\u8df5\u7684\u306b\u5b66\u3076\u3053\u3068\u304c\u3067\u304d\u307e\u3057\u305f\u3002FastAPI\u3092\u4f7f\u3063\u3066\u5805\u7262\u306a\u30d0\u30c3\u30af\u30a8\u30f3\u30c9<a target=\"_blank\" class=\"keyword\" href=\"https:\/\/d.hatena.ne.jp\/keyword\/API\">API<\/a>\u3092\u4f5c\u6210\u3057\u3001React\u3067\u52d5\u7684\u3067\u30ec\u30b9\u30dd\u30f3\u30b7\u30d6\u306a\u30d5\u30ed\u30f3\u30c8\u30a8\u30f3\u30c9\u3092\u69cb\u7bc9\u3057\u3001MongoDB\u3067\u30c7\u30fc\u30bf\u3092\u6c38\u7d9a\u5316\u3057\u3001Docker\u3067\u30a2\u30d7\u30ea\u30b1\u30fc\u30b7\u30e7\u30f3\u5168\u4f53\u3092\u30b3\u30f3\u30c6\u30ca\u5316\u3059\u308b\u65b9\u6cd5\u3092\u5b66\u3073\u307e\u3057\u305f\u3002\u3053\u306e\u30d7\u30ed\u30b8\u30a7\u30af\u30c8\u3092\u901a\u3058\u3066\u3001\u3053\u308c\u3089\u306e\u6280\u8853\u304c\u30b7\u30fc\u30e0\u30ec\u30b9\u306b\u9023\u643a\u3057\u3001\u30d5\u30eb\u6a5f\u80fd\u3067\u30b9\u30b1\u30fc\u30e9\u30d6\u30eb\u306a<a target=\"_blank\" class=\"keyword\" href=\"https:\/\/d.hatena.ne.jp\/keyword\/%A5%A6%A5%A7%A5%D6%A5%A2%A5%D7%A5%EA%A5%B1%A1%BC%A5%B7%A5%E7%A5%F3\">\u30a6\u30a7\u30d6\u30a2\u30d7\u30ea\u30b1\u30fc\u30b7\u30e7\u30f3<\/a>\u3092\u4f5c\u6210\u3067\u304d\u308b\u3053\u3068\u3092\u5b9f\u611f\u3067\u304d\u307e\u3057\u305f\u3002<\/p>\n<\/div>\n<p><script>(function(d, s, id) {\n  var js, fjs = d.getElementsByTagName(s)[0];\n  if (d.getElementById(id)) return;\n  js = d.createElement(s); js.id = id;\n  js.src = \"\/\/connect.facebook.net\/ja_JP\/sdk.js#xfbml=1&appId=719729204785177&version=v17.0\";\n  fjs.parentNode.insertBefore(js, fjs);\n}(document, 'script', 'facebook-jssdk'));<\/script><br \/>\n<br \/>\n<br \/><a href=\"https:\/\/tech.iimon.co.jp\/entry\/2025\/11\/16\/225643\">\u5143\u306e\u8a18\u4e8b\u3092\u78ba\u8a8d\u3059\u308b <\/a><\/p>\n","protected":false},"excerpt":{"rendered":"FARM \u30b9\u30bf\u30c3\u30af\u3068\u306f\uff1f FARM \u30b9\u30bf\u30c3\u30af\u3068\u306f\u3001\u4ee5\u4e0b\u306e3\u3064\u306e\u5f37\u529b\u306a\u6280\u8853\u3092\u7d44\u307f\u5408\u308f\u305b\u305f\u3001\u73fe\u4ee3\u7684\u306a Web \u30a2\u30d7\u30ea\u30b1\u30fc\u30b7\u30e7\u30f3\u958b\u767a\u30b9\u30bf\u30c3\u30af\u3067\u3059\u3002 F: FastAPI\uff08\u30d0\u30c3\u30af\u30a8\u30f3\u30c9\uff09 R: React\uff08\u30d5\u30ed\u30f3\u30c8\u30a8\u30f3\u30c9\uff09 M: [&hellip;]","protected":false},"author":1,"featured_media":23937,"comment_status":"closed","ping_status":"closed","sticky":false,"template":"","format":"standard","meta":{"_acf_changed":false,"footnotes":""},"categories":[4],"tags":[],"class_list":["post-23936","post","type-post","status-publish","format-standard","has-post-thumbnail","hentry","category-company-tec"],"acf":[],"yoast_head":"<!-- This site is optimized with the Yoast SEO plugin v27.6 - https:\/\/yoast.com\/product\/yoast-seo-wordpress\/ -->\n<title>FARM \u30b9\u30bf\u30c3\u30af\u3092\u4f7f\u3063\u305f\u30d5\u30eb\u30b9\u30bf\u30c3\u30af\u30a2\u30d7\u30ea\u958b\u767a - iimon TECH BLOG - \u30dd\u30b1\u30b3\u30f3<\/title>\n<meta name=\"robots\" content=\"index, follow, max-snippet:-1, max-image-preview:large, max-video-preview:-1\" \/>\n<link rel=\"canonical\" href=\"https:\/\/tech.iimon.co.jp\/entry\/2025\/11\/16\/225643\" \/>\n<meta property=\"og:locale\" content=\"ja_JP\" \/>\n<meta property=\"og:type\" content=\"article\" \/>\n<meta property=\"og:title\" content=\"FARM \u30b9\u30bf\u30c3\u30af\u3092\u4f7f\u3063\u305f\u30d5\u30eb\u30b9\u30bf\u30c3\u30af\u30a2\u30d7\u30ea\u958b\u767a - iimon TECH BLOG - \u30dd\u30b1\u30b3\u30f3\" \/>\n<meta property=\"og:description\" content=\"FARM \u30b9\u30bf\u30c3\u30af\u3068\u306f\uff1f FARM \u30b9\u30bf\u30c3\u30af\u3068\u306f\u3001\u4ee5\u4e0b\u306e3\u3064\u306e\u5f37\u529b\u306a\u6280\u8853\u3092\u7d44\u307f\u5408\u308f\u305b\u305f\u3001\u73fe\u4ee3\u7684\u306a Web \u30a2\u30d7\u30ea\u30b1\u30fc\u30b7\u30e7\u30f3\u958b\u767a\u30b9\u30bf\u30c3\u30af\u3067\u3059\u3002 F: FastAPI\uff08\u30d0\u30c3\u30af\u30a8\u30f3\u30c9\uff09 R: React\uff08\u30d5\u30ed\u30f3\u30c8\u30a8\u30f3\u30c9\uff09 M: [&hellip;]\" \/>\n<meta property=\"og:url\" content=\"https:\/\/tech.iimon.co.jp\/entry\/2025\/11\/16\/225643\" \/>\n<meta property=\"og:site_name\" content=\"\u30dd\u30b1\u30b3\u30f3\" \/>\n<meta property=\"article:published_time\" content=\"2025-11-16T15:51:21+00:00\" \/>\n<meta property=\"og:image\" content=\"https:\/\/pokecon.jp\/job\/wp-content\/uploads\/2025\/11\/1763301926.png\" \/>\n\t<meta property=\"og:image:width\" content=\"1200\" \/>\n\t<meta property=\"og:image:height\" content=\"630\" \/>\n\t<meta property=\"og:image:type\" content=\"image\/png\" \/>\n<meta name=\"author\" content=\"info@pokecon.jp\" \/>\n<meta name=\"twitter:card\" content=\"summary_large_image\" \/>\n<meta name=\"twitter:label1\" content=\"\u57f7\u7b46\u8005\" \/>\n\t<meta name=\"twitter:data1\" content=\"info@pokecon.jp\" \/>\n\t<meta name=\"twitter:label2\" content=\"\u63a8\u5b9a\u8aad\u307f\u53d6\u308a\u6642\u9593\" \/>\n\t<meta name=\"twitter:data2\" content=\"8\u5206\" \/>\n<script type=\"application\/ld+json\" class=\"yoast-schema-graph\">{\"@context\":\"https:\\\/\\\/schema.org\",\"@graph\":[{\"@type\":\"Article\",\"@id\":\"https:\\\/\\\/tech.iimon.co.jp\\\/entry\\\/2025\\\/11\\\/16\\\/225643#article\",\"isPartOf\":{\"@id\":\"https:\\\/\\\/pokecon.jp\\\/job\\\/23936\\\/\"},\"author\":{\"name\":\"info@pokecon.jp\",\"@id\":\"https:\\\/\\\/pokecon.jp\\\/job\\\/#\\\/schema\\\/person\\\/16c9f07b1ba984d165d9aee259bda997\"},\"headline\":\"FARM \u30b9\u30bf\u30c3\u30af\u3092\u4f7f\u3063\u305f\u30d5\u30eb\u30b9\u30bf\u30c3\u30af\u30a2\u30d7\u30ea\u958b\u767a &#8211; iimon TECH BLOG\",\"datePublished\":\"2025-11-16T15:51:21+00:00\",\"mainEntityOfPage\":{\"@id\":\"https:\\\/\\\/pokecon.jp\\\/job\\\/23936\\\/\"},\"wordCount\":139,\"image\":{\"@id\":\"https:\\\/\\\/tech.iimon.co.jp\\\/entry\\\/2025\\\/11\\\/16\\\/225643#primaryimage\"},\"thumbnailUrl\":\"https:\\\/\\\/pokecon.jp\\\/job\\\/wp-content\\\/uploads\\\/2025\\\/11\\\/1763301926.png\",\"articleSection\":[\"\u4f01\u696d\u30c6\u30c3\u30af\"],\"inLanguage\":\"ja\"},{\"@type\":\"WebPage\",\"@id\":\"https:\\\/\\\/pokecon.jp\\\/job\\\/23936\\\/\",\"url\":\"https:\\\/\\\/tech.iimon.co.jp\\\/entry\\\/2025\\\/11\\\/16\\\/225643\",\"name\":\"FARM \u30b9\u30bf\u30c3\u30af\u3092\u4f7f\u3063\u305f\u30d5\u30eb\u30b9\u30bf\u30c3\u30af\u30a2\u30d7\u30ea\u958b\u767a - iimon TECH BLOG - \u30dd\u30b1\u30b3\u30f3\",\"isPartOf\":{\"@id\":\"https:\\\/\\\/pokecon.jp\\\/job\\\/#website\"},\"primaryImageOfPage\":{\"@id\":\"https:\\\/\\\/tech.iimon.co.jp\\\/entry\\\/2025\\\/11\\\/16\\\/225643#primaryimage\"},\"image\":{\"@id\":\"https:\\\/\\\/tech.iimon.co.jp\\\/entry\\\/2025\\\/11\\\/16\\\/225643#primaryimage\"},\"thumbnailUrl\":\"https:\\\/\\\/pokecon.jp\\\/job\\\/wp-content\\\/uploads\\\/2025\\\/11\\\/1763301926.png\",\"datePublished\":\"2025-11-16T15:51:21+00:00\",\"author\":{\"@id\":\"https:\\\/\\\/pokecon.jp\\\/job\\\/#\\\/schema\\\/person\\\/16c9f07b1ba984d165d9aee259bda997\"},\"breadcrumb\":{\"@id\":\"https:\\\/\\\/tech.iimon.co.jp\\\/entry\\\/2025\\\/11\\\/16\\\/225643#breadcrumb\"},\"inLanguage\":\"ja\",\"potentialAction\":[{\"@type\":\"ReadAction\",\"target\":[\"https:\\\/\\\/tech.iimon.co.jp\\\/entry\\\/2025\\\/11\\\/16\\\/225643\"]}]},{\"@type\":\"ImageObject\",\"inLanguage\":\"ja\",\"@id\":\"https:\\\/\\\/tech.iimon.co.jp\\\/entry\\\/2025\\\/11\\\/16\\\/225643#primaryimage\",\"url\":\"https:\\\/\\\/pokecon.jp\\\/job\\\/wp-content\\\/uploads\\\/2025\\\/11\\\/1763301926.png\",\"contentUrl\":\"https:\\\/\\\/pokecon.jp\\\/job\\\/wp-content\\\/uploads\\\/2025\\\/11\\\/1763301926.png\",\"width\":1200,\"height\":630},{\"@type\":\"BreadcrumbList\",\"@id\":\"https:\\\/\\\/tech.iimon.co.jp\\\/entry\\\/2025\\\/11\\\/16\\\/225643#breadcrumb\",\"itemListElement\":[{\"@type\":\"ListItem\",\"position\":1,\"name\":\"\u30db\u30fc\u30e0\",\"item\":\"https:\\\/\\\/pokecon.jp\\\/job\\\/\"},{\"@type\":\"ListItem\",\"position\":2,\"name\":\"FARM \u30b9\u30bf\u30c3\u30af\u3092\u4f7f\u3063\u305f\u30d5\u30eb\u30b9\u30bf\u30c3\u30af\u30a2\u30d7\u30ea\u958b\u767a &#8211; iimon TECH BLOG\"}]},{\"@type\":\"WebSite\",\"@id\":\"https:\\\/\\\/pokecon.jp\\\/job\\\/#website\",\"url\":\"https:\\\/\\\/pokecon.jp\\\/job\\\/\",\"name\":\"\u30dd\u30b1\u30b3\u30f3\",\"description\":\"\",\"potentialAction\":[{\"@type\":\"SearchAction\",\"target\":{\"@type\":\"EntryPoint\",\"urlTemplate\":\"https:\\\/\\\/pokecon.jp\\\/job\\\/?s={search_term_string}\"},\"query-input\":{\"@type\":\"PropertyValueSpecification\",\"valueRequired\":true,\"valueName\":\"search_term_string\"}}],\"inLanguage\":\"ja\"},{\"@type\":\"Person\",\"@id\":\"https:\\\/\\\/pokecon.jp\\\/job\\\/#\\\/schema\\\/person\\\/16c9f07b1ba984d165d9aee259bda997\",\"name\":\"info@pokecon.jp\",\"image\":{\"@type\":\"ImageObject\",\"inLanguage\":\"ja\",\"@id\":\"https:\\\/\\\/secure.gravatar.com\\\/avatar\\\/2b0549cd9f7907c092ca5fbb283baf72337f235726e4b46fa39ec0b701ac2fe2?s=96&d=wavatar&r=g\",\"url\":\"https:\\\/\\\/secure.gravatar.com\\\/avatar\\\/2b0549cd9f7907c092ca5fbb283baf72337f235726e4b46fa39ec0b701ac2fe2?s=96&d=wavatar&r=g\",\"contentUrl\":\"https:\\\/\\\/secure.gravatar.com\\\/avatar\\\/2b0549cd9f7907c092ca5fbb283baf72337f235726e4b46fa39ec0b701ac2fe2?s=96&d=wavatar&r=g\",\"caption\":\"info@pokecon.jp\"},\"url\":\"https:\\\/\\\/pokecon.jp\\\/job\\\/author\\\/infopokecon-jp\\\/\"}]}<\/script>\n<!-- \/ Yoast SEO plugin. -->","yoast_head_json":{"title":"FARM \u30b9\u30bf\u30c3\u30af\u3092\u4f7f\u3063\u305f\u30d5\u30eb\u30b9\u30bf\u30c3\u30af\u30a2\u30d7\u30ea\u958b\u767a - iimon TECH BLOG - \u30dd\u30b1\u30b3\u30f3","robots":{"index":"index","follow":"follow","max-snippet":"max-snippet:-1","max-image-preview":"max-image-preview:large","max-video-preview":"max-video-preview:-1"},"canonical":"https:\/\/tech.iimon.co.jp\/entry\/2025\/11\/16\/225643","og_locale":"ja_JP","og_type":"article","og_title":"FARM \u30b9\u30bf\u30c3\u30af\u3092\u4f7f\u3063\u305f\u30d5\u30eb\u30b9\u30bf\u30c3\u30af\u30a2\u30d7\u30ea\u958b\u767a - iimon TECH BLOG - \u30dd\u30b1\u30b3\u30f3","og_description":"FARM \u30b9\u30bf\u30c3\u30af\u3068\u306f\uff1f FARM \u30b9\u30bf\u30c3\u30af\u3068\u306f\u3001\u4ee5\u4e0b\u306e3\u3064\u306e\u5f37\u529b\u306a\u6280\u8853\u3092\u7d44\u307f\u5408\u308f\u305b\u305f\u3001\u73fe\u4ee3\u7684\u306a Web \u30a2\u30d7\u30ea\u30b1\u30fc\u30b7\u30e7\u30f3\u958b\u767a\u30b9\u30bf\u30c3\u30af\u3067\u3059\u3002 F: FastAPI\uff08\u30d0\u30c3\u30af\u30a8\u30f3\u30c9\uff09 R: React\uff08\u30d5\u30ed\u30f3\u30c8\u30a8\u30f3\u30c9\uff09 M: [&hellip;]","og_url":"https:\/\/tech.iimon.co.jp\/entry\/2025\/11\/16\/225643","og_site_name":"\u30dd\u30b1\u30b3\u30f3","article_published_time":"2025-11-16T15:51:21+00:00","og_image":[{"width":1200,"height":630,"url":"https:\/\/pokecon.jp\/job\/wp-content\/uploads\/2025\/11\/1763301926.png","type":"image\/png"}],"author":"info@pokecon.jp","twitter_card":"summary_large_image","twitter_misc":{"\u57f7\u7b46\u8005":"info@pokecon.jp","\u63a8\u5b9a\u8aad\u307f\u53d6\u308a\u6642\u9593":"8\u5206"},"schema":{"@context":"https:\/\/schema.org","@graph":[{"@type":"Article","@id":"https:\/\/tech.iimon.co.jp\/entry\/2025\/11\/16\/225643#article","isPartOf":{"@id":"https:\/\/pokecon.jp\/job\/23936\/"},"author":{"name":"info@pokecon.jp","@id":"https:\/\/pokecon.jp\/job\/#\/schema\/person\/16c9f07b1ba984d165d9aee259bda997"},"headline":"FARM \u30b9\u30bf\u30c3\u30af\u3092\u4f7f\u3063\u305f\u30d5\u30eb\u30b9\u30bf\u30c3\u30af\u30a2\u30d7\u30ea\u958b\u767a &#8211; iimon TECH BLOG","datePublished":"2025-11-16T15:51:21+00:00","mainEntityOfPage":{"@id":"https:\/\/pokecon.jp\/job\/23936\/"},"wordCount":139,"image":{"@id":"https:\/\/tech.iimon.co.jp\/entry\/2025\/11\/16\/225643#primaryimage"},"thumbnailUrl":"https:\/\/pokecon.jp\/job\/wp-content\/uploads\/2025\/11\/1763301926.png","articleSection":["\u4f01\u696d\u30c6\u30c3\u30af"],"inLanguage":"ja"},{"@type":"WebPage","@id":"https:\/\/pokecon.jp\/job\/23936\/","url":"https:\/\/tech.iimon.co.jp\/entry\/2025\/11\/16\/225643","name":"FARM \u30b9\u30bf\u30c3\u30af\u3092\u4f7f\u3063\u305f\u30d5\u30eb\u30b9\u30bf\u30c3\u30af\u30a2\u30d7\u30ea\u958b\u767a - iimon TECH BLOG - \u30dd\u30b1\u30b3\u30f3","isPartOf":{"@id":"https:\/\/pokecon.jp\/job\/#website"},"primaryImageOfPage":{"@id":"https:\/\/tech.iimon.co.jp\/entry\/2025\/11\/16\/225643#primaryimage"},"image":{"@id":"https:\/\/tech.iimon.co.jp\/entry\/2025\/11\/16\/225643#primaryimage"},"thumbnailUrl":"https:\/\/pokecon.jp\/job\/wp-content\/uploads\/2025\/11\/1763301926.png","datePublished":"2025-11-16T15:51:21+00:00","author":{"@id":"https:\/\/pokecon.jp\/job\/#\/schema\/person\/16c9f07b1ba984d165d9aee259bda997"},"breadcrumb":{"@id":"https:\/\/tech.iimon.co.jp\/entry\/2025\/11\/16\/225643#breadcrumb"},"inLanguage":"ja","potentialAction":[{"@type":"ReadAction","target":["https:\/\/tech.iimon.co.jp\/entry\/2025\/11\/16\/225643"]}]},{"@type":"ImageObject","inLanguage":"ja","@id":"https:\/\/tech.iimon.co.jp\/entry\/2025\/11\/16\/225643#primaryimage","url":"https:\/\/pokecon.jp\/job\/wp-content\/uploads\/2025\/11\/1763301926.png","contentUrl":"https:\/\/pokecon.jp\/job\/wp-content\/uploads\/2025\/11\/1763301926.png","width":1200,"height":630},{"@type":"BreadcrumbList","@id":"https:\/\/tech.iimon.co.jp\/entry\/2025\/11\/16\/225643#breadcrumb","itemListElement":[{"@type":"ListItem","position":1,"name":"\u30db\u30fc\u30e0","item":"https:\/\/pokecon.jp\/job\/"},{"@type":"ListItem","position":2,"name":"FARM \u30b9\u30bf\u30c3\u30af\u3092\u4f7f\u3063\u305f\u30d5\u30eb\u30b9\u30bf\u30c3\u30af\u30a2\u30d7\u30ea\u958b\u767a &#8211; iimon TECH BLOG"}]},{"@type":"WebSite","@id":"https:\/\/pokecon.jp\/job\/#website","url":"https:\/\/pokecon.jp\/job\/","name":"\u30dd\u30b1\u30b3\u30f3","description":"","potentialAction":[{"@type":"SearchAction","target":{"@type":"EntryPoint","urlTemplate":"https:\/\/pokecon.jp\/job\/?s={search_term_string}"},"query-input":{"@type":"PropertyValueSpecification","valueRequired":true,"valueName":"search_term_string"}}],"inLanguage":"ja"},{"@type":"Person","@id":"https:\/\/pokecon.jp\/job\/#\/schema\/person\/16c9f07b1ba984d165d9aee259bda997","name":"info@pokecon.jp","image":{"@type":"ImageObject","inLanguage":"ja","@id":"https:\/\/secure.gravatar.com\/avatar\/2b0549cd9f7907c092ca5fbb283baf72337f235726e4b46fa39ec0b701ac2fe2?s=96&d=wavatar&r=g","url":"https:\/\/secure.gravatar.com\/avatar\/2b0549cd9f7907c092ca5fbb283baf72337f235726e4b46fa39ec0b701ac2fe2?s=96&d=wavatar&r=g","contentUrl":"https:\/\/secure.gravatar.com\/avatar\/2b0549cd9f7907c092ca5fbb283baf72337f235726e4b46fa39ec0b701ac2fe2?s=96&d=wavatar&r=g","caption":"info@pokecon.jp"},"url":"https:\/\/pokecon.jp\/job\/author\/infopokecon-jp\/"}]}},"_links":{"self":[{"href":"https:\/\/pokecon.jp\/job\/wp-json\/wp\/v2\/posts\/23936","targetHints":{"allow":["GET"]}}],"collection":[{"href":"https:\/\/pokecon.jp\/job\/wp-json\/wp\/v2\/posts"}],"about":[{"href":"https:\/\/pokecon.jp\/job\/wp-json\/wp\/v2\/types\/post"}],"author":[{"embeddable":true,"href":"https:\/\/pokecon.jp\/job\/wp-json\/wp\/v2\/users\/1"}],"replies":[{"embeddable":true,"href":"https:\/\/pokecon.jp\/job\/wp-json\/wp\/v2\/comments?post=23936"}],"version-history":[{"count":1,"href":"https:\/\/pokecon.jp\/job\/wp-json\/wp\/v2\/posts\/23936\/revisions"}],"predecessor-version":[{"id":23938,"href":"https:\/\/pokecon.jp\/job\/wp-json\/wp\/v2\/posts\/23936\/revisions\/23938"}],"wp:featuredmedia":[{"embeddable":true,"href":"https:\/\/pokecon.jp\/job\/wp-json\/wp\/v2\/media\/23937"}],"wp:attachment":[{"href":"https:\/\/pokecon.jp\/job\/wp-json\/wp\/v2\/media?parent=23936"}],"wp:term":[{"taxonomy":"category","embeddable":true,"href":"https:\/\/pokecon.jp\/job\/wp-json\/wp\/v2\/categories?post=23936"},{"taxonomy":"post_tag","embeddable":true,"href":"https:\/\/pokecon.jp\/job\/wp-json\/wp\/v2\/tags?post=23936"}],"curies":[{"name":"wp","href":"https:\/\/api.w.org\/{rel}","templated":true}]}}