{"id":3291,"date":"2025-03-24T08:46:31","date_gmt":"2025-03-23T23:46:31","guid":{"rendered":"https:\/\/dexall.co.jp\/articles\/?p=3291"},"modified":"2025-03-24T08:47:01","modified_gmt":"2025-03-23T23:47:01","slug":"%e3%80%90%e4%bf%9d%e5%ad%98%e7%89%88%e3%80%91laravel%e7%ae%a1%e7%90%86%e7%94%bb%e9%9d%a2%e3%81%ae%e4%bd%9c%e3%82%8a%e6%96%b9%e5%ae%8c%e5%85%a8%e3%82%ac%e3%82%a4%e3%83%892024-%e5%ae%9f%e8%a3%85","status":"publish","type":"post","link":"https:\/\/dexall.co.jp\/articles\/?p=3291","title":{"rendered":"\u3010\u4fdd\u5b58\u7248\u3011Laravel\u7ba1\u7406\u753b\u9762\u306e\u4f5c\u308a\u65b9\u5b8c\u5168\u30ac\u30a4\u30c92024 &#8211; \u5b9f\u88c5\u304b\u3089\u30bb\u30ad\u30e5\u30ea\u30c6\u30a3\u307e\u3067"},"content":{"rendered":"\n<div class=\"toc\"><br \/>\n<b>Warning<\/b>:  Undefined array key \"is_admin\" in <b>\/home\/xs392991\/dexall.co.jp\/public_html\/articles\/wp-content\/themes\/sango-theme\/library\/gutenberg\/dist\/classes\/Toc.php<\/b> on line <b>116<\/b><br \/>\n<br \/>\n<b>Warning<\/b>:  Undefined array key \"is_category_top\" in <b>\/home\/xs392991\/dexall.co.jp\/public_html\/articles\/wp-content\/themes\/sango-theme\/library\/gutenberg\/dist\/classes\/Toc.php<\/b> on line <b>121<\/b><br \/>\n<br \/>\n<b>Warning<\/b>:  Undefined array key \"is_top\" in <b>\/home\/xs392991\/dexall.co.jp\/public_html\/articles\/wp-content\/themes\/sango-theme\/library\/gutenberg\/dist\/classes\/Toc.php<\/b> on line <b>128<\/b><br \/>\n    <div id=\"toc_container\" class=\"sgb-toc--bullets js-smooth-scroll\" data-dialog-title=\"\u76ee\u6b21\">\n      <p class=\"toc_title\">\u76ee\u6b21 <\/p>\n      <ul class=\"toc_list\">  <li class=\"first\">    <a href=\"#i-0\">Laravel\u7ba1\u7406\u753b\u9762\u306e\u57fa\u790e\u77e5\u8b58<\/a>    <ul class=\"menu_level_1\">      <li class=\"first\">        <a href=\"#i-1\">\u7ba1\u7406\u753b\u9762\u304c\u5fc5\u8981\u306a\u7406\u7531\u3068\u4e3b\u8981\u306a\u6a5f\u80fd<\/a>      <\/li>      <li>        <a href=\"#i-2\">Laravel\u3067\u7ba1\u7406\u753b\u9762\u3092\u4f5c\u308b\u30e1\u30ea\u30c3\u30c8<\/a>      <\/li>      <li class=\"last\">        <a href=\"#i-3\">\u7ba1\u7406\u753b\u9762\u5b9f\u88c5\u306e\u5168\u4f53\u306e\u6d41\u308c<\/a>      <\/li>    <\/ul>  <\/li>  <li>    <a href=\"#i-4\">\u7ba1\u7406\u753b\u9762\u306e\u74b0\u5883\u69cb\u7bc9\u3068\u521d\u671f\u8a2d\u5b9a<\/a>    <ul class=\"menu_level_1\">      <li class=\"first\">        <a href=\"#i-5\">\u5fc5\u8981\u306a\u30d1\u30c3\u30b1\u30fc\u30b8\u3068\u30a4\u30f3\u30b9\u30c8\u30fc\u30eb\u624b\u9806<\/a>      <\/li>      <li>        <a href=\"#i-6\">\u30c7\u30fc\u30bf\u30d9\u30fc\u30b9\u8a2d\u8a08\u306e\u30d9\u30b9\u30c8\u30d7\u30e9\u30af\u30c6\u30a3\u30b9<\/a>      <\/li>      <li class=\"last\">        <a href=\"#i-7\">\u8a8d\u8a3c\u30b7\u30b9\u30c6\u30e0\u306e\u5c0e\u5165\u65b9\u6cd5<\/a>      <\/li>    <\/ul>  <\/li>  <li>    <a href=\"#i-8\">\u57fa\u672c\u7684\u306aCRUD\u6a5f\u80fd\u306e\u5b9f\u88c5\u624b\u9806<\/a>    <ul class=\"menu_level_1\">      <li class=\"first\">        <a href=\"#i-9\">\u30e2\u30c7\u30eb\u3068\u30de\u30a4\u30b0\u30ec\u30fc\u30b7\u30e7\u30f3\u306e\u4f5c\u6210\u65b9\u6cd5<\/a>      <\/li>      <li>        <a href=\"#i-10\">\u30b3\u30f3\u30c8\u30ed\u30fc\u30e9\u30fc\u306e\u5b9f\u88c5\u3068\u30eb\u30fc\u30c6\u30a3\u30f3\u30b0\u8a2d\u5b9a<\/a>      <\/li>      <li class=\"last\">        <a href=\"#i-11\">\u30d3\u30e5\u30fc\u30c6\u30f3\u30d7\u30ec\u30fc\u30c8\u306e\u4f5c\u6210\u3068\u30ab\u30b9\u30bf\u30de\u30a4\u30ba<\/a>      <\/li>    <\/ul>  <\/li>  <li>    <a href=\"#i-12\">\u30bb\u30ad\u30e5\u30ea\u30c6\u30a3\u5bfe\u7b56\u3068\u6a29\u9650\u7ba1\u7406<\/a>    <ul class=\"menu_level_1\">      <li class=\"first\">        <a href=\"#i-13\">\u30ed\u30fc\u30eb\u30d9\u30fc\u30b9\u306e\u30a2\u30af\u30bb\u30b9\u5236\u5fa1\u306e\u5b9f\u88c5<\/a>      <\/li>      <li>        <a href=\"#i-14\">\u30af\u30ed\u30b9\u30b5\u30a4\u30c8\u30b9\u30af\u30ea\u30d7\u30c6\u30a3\u30f3\u30b0\u5bfe\u7b56<\/a>      <\/li>      <li class=\"last\">        <a href=\"#i-15\">\u30bb\u30c3\u30b7\u30e7\u30f3\u7ba1\u7406\u3068CSRF\u5bfe\u7b56<\/a>      <\/li>    <\/ul>  <\/li>  <li>    <a href=\"#i-16\">\u7ba1\u7406\u753b\u9762\u306e\u30ab\u30b9\u30bf\u30de\u30a4\u30ba\u3068\u62e1\u5f35<\/a>    <ul class=\"menu_level_1\">      <li class=\"first\">        <a href=\"#i-17\">\u30ec\u30a4\u30a2\u30a6\u30c8\u306e\u30ab\u30b9\u30bf\u30de\u30a4\u30ba\u65b9\u6cd5<\/a>      <\/li>      <li>        <a href=\"#i-18\">\u30c0\u30c3\u30b7\u30e5\u30dc\u30fc\u30c9\u30a6\u30a3\u30b8\u30a7\u30c3\u30c8\u306e\u8ffd\u52a0<\/a>      <\/li>      <li class=\"last\">        <a href=\"#i-19\">\u30d5\u30a1\u30a4\u30eb\u30a2\u30c3\u30d7\u30ed\u30fc\u30c9\u6a5f\u80fd\u306e\u5b9f\u88c5<\/a>      <\/li>    <\/ul>  <\/li>  <li>    <a href=\"#i-20\">\u30d1\u30d5\u30a9\u30fc\u30de\u30f3\u30b9\u6700\u9069\u5316\u3068\u30e1\u30f3\u30c6\u30ca\u30f3\u30b9<\/a>    <ul class=\"menu_level_1\">      <li class=\"first\">        <a href=\"#i-21\">\u30ad\u30e3\u30c3\u30b7\u30e5\u8a2d\u5b9a\u3068\u30c7\u30fc\u30bf\u30d9\u30fc\u30b9\u6700\u9069\u5316<\/a>      <\/li>      <li>        <a href=\"#i-22\">\u5b9a\u671f\u7684\u306a\u30e1\u30f3\u30c6\u30ca\u30f3\u30b9\u4f5c\u696d\u306e\u81ea\u52d5\u5316<\/a>      <\/li>      <li class=\"last\">        <a href=\"#i-23\">\u30a8\u30e9\u30fc\u30ed\u30b0\u306e\u76e3\u8996\u3068\u5bfe\u5fdc\u65b9\u6cd5<\/a>      <\/li>    <\/ul>  <\/li>  <li class=\"last\">    <a href=\"#i-24\">\u5b9f\u88c5\u4f8b\u3068\u5fdc\u7528\u4e8b\u4f8b<\/a>    <ul class=\"menu_level_1\">      <li class=\"first\">        <a href=\"#i-25\">EC\u30b5\u30a4\u30c8\u7ba1\u7406\u753b\u9762\u306e\u5b9f\u88c5\u4f8b<\/a>      <\/li>      <li>        <a href=\"#i-26\">\u30e6\u30fc\u30b6\u30fc\u7ba1\u7406\u30b7\u30b9\u30c6\u30e0\u306e\u5b9f\u88c5\u4f8b<\/a>      <\/li>      <li class=\"last\">        <a href=\"#i-27\">\u30b3\u30f3\u30c6\u30f3\u30c4\u7ba1\u7406\u30b7\u30b9\u30c6\u30e0\u306e\u5b9f\u88c5\u4f8b<\/a>      <\/li>    <\/ul>  <\/li><\/ul>\n      <a href=\"#\" class=\"sgb-toc-button js-toc-button\" rel=\"nofollow\" data-open-dialog=\"true\"><i class=\"fa fa-list\"><\/i><span class=\"sgb-toc-button__text\">\u76ee\u6b21\u3078<\/span><\/a>\n    <\/div><\/div><h2 class=\"wp-block-heading\" id=\"i-0\">Laravel\u7ba1\u7406\u753b\u9762\u306e\u57fa\u790e\u77e5\u8b58<\/h2>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"i-1\">\u7ba1\u7406\u753b\u9762\u304c\u5fc5\u8981\u306a\u7406\u7531\u3068\u4e3b\u8981\u306a\u6a5f\u80fd<\/h3>\n\n\n\n<p>Web\u30a2\u30d7\u30ea\u30b1\u30fc\u30b7\u30e7\u30f3\u306b\u304a\u3044\u3066\u7ba1\u7406\u753b\u9762\uff08Admin Panel\uff09\u306f\u3001\u30b7\u30b9\u30c6\u30e0\u306e\u904b\u7528\u3068\u4fdd\u5b88\u306b\u4e0d\u53ef\u6b20\u306a\u8981\u7d20\u3067\u3059\u3002\u4ee5\u4e0b\u306b\u3001\u7ba1\u7406\u753b\u9762\u304c\u5fc5\u8981\u3068\u3055\u308c\u308b\u4e3b\u306a\u7406\u7531\u3068\u3001\u5b9f\u88c5\u3059\u3079\u304d\u4e3b\u8981\u306a\u6a5f\u80fd\u306b\u3064\u3044\u3066\u89e3\u8aac\u3057\u307e\u3059\u3002<\/p>\n\n\n\n<p><strong>\u7ba1\u7406\u753b\u9762\u304c\u5fc5\u8981\u306a\u7406\u7531\uff1a<\/strong><\/p>\n\n\n\n<ol class=\"wp-block-list\">\n<li>\u30c7\u30fc\u30bf\u7ba1\u7406\u306e\u52b9\u7387\u5316<\/li>\n<\/ol>\n\n\n\n<ul class=\"wp-block-list\">\n<li>\u30e6\u30fc\u30b6\u30fc\u30c7\u30fc\u30bf\u306e\u95b2\u89a7\u30fb\u7de8\u96c6<\/li>\n\n\n\n<li>\u30b3\u30f3\u30c6\u30f3\u30c4\u306e\u4f5c\u6210\u30fb\u66f4\u65b0\u30fb\u524a\u9664<\/li>\n\n\n\n<li>\u30b7\u30b9\u30c6\u30e0\u8a2d\u5b9a\u306e\u4e00\u5143\u7ba1\u7406<\/li>\n<\/ul>\n\n\n\n<ol class=\"wp-block-list\">\n<li>\u904b\u7528\u696d\u52d9\u306e\u52b9\u7387\u5316<\/li>\n<\/ol>\n\n\n\n<ul class=\"wp-block-list\">\n<li>\u5404\u7a2e\u7d71\u8a08\u60c5\u5831\u306e\u78ba\u8a8d<\/li>\n\n\n\n<li>\u30e6\u30fc\u30b6\u30fc\u304b\u3089\u306e\u554f\u3044\u5408\u308f\u305b\u5bfe\u5fdc<\/li>\n\n\n\n<li>\u30b7\u30b9\u30c6\u30e0\u30ed\u30b0\u306e\u76e3\u8996<\/li>\n<\/ul>\n\n\n\n<ol class=\"wp-block-list\">\n<li>\u30bb\u30ad\u30e5\u30ea\u30c6\u30a3\u7ba1\u7406<\/li>\n<\/ol>\n\n\n\n<ul class=\"wp-block-list\">\n<li>\u30a2\u30af\u30bb\u30b9\u6a29\u9650\u306e\u7ba1\u7406<\/li>\n\n\n\n<li>\u30bb\u30ad\u30e5\u30ea\u30c6\u30a3\u30ed\u30b0\u306e\u78ba\u8a8d<\/li>\n\n\n\n<li>\u4e0d\u6b63\u30a2\u30af\u30bb\u30b9\u306e\u76e3\u8996\u3068\u5bfe\u7b56<\/li>\n<\/ul>\n\n\n\n<p><strong>\u5b9f\u88c5\u3059\u3079\u304d\u4e3b\u8981\u6a5f\u80fd\uff1a<\/strong><\/p>\n\n\n<div id=\"id-50c629f2-85d4-4fa6-ab53-2e780c610b94\">\n<figure class=\"wp-block-table\"><table class=\"has-fixed-layout\"><thead><tr><th>\u6a5f\u80fd\u30ab\u30c6\u30b4\u30ea<\/th><th>\u5177\u4f53\u7684\u306a\u6a5f\u80fd<\/th><th>\u91cd\u8981\u5ea6<\/th><\/tr><\/thead><tbody><tr><td>\u8a8d\u8a3c\u30fb\u8a8d\u53ef<\/td><td>\u30ed\u30b0\u30a4\u30f3\/\u30ed\u30b0\u30a2\u30a6\u30c8<br>\u30d1\u30b9\u30ef\u30fc\u30c9\u7ba1\u7406<br>\u6a29\u9650\u7ba1\u7406<\/td><td>\u6700\u91cd\u8981<\/td><\/tr><tr><td>\u30c7\u30fc\u30bf\u7ba1\u7406<\/td><td>CRUD\u64cd\u4f5c<br>\u4e00\u62ec\u51e6\u7406<br>\u30c7\u30fc\u30bf\u30a8\u30af\u30b9\u30dd\u30fc\u30c8<\/td><td>\u91cd\u8981<\/td><\/tr><tr><td>\u76e3\u8996\u30fb\u5206\u6790<\/td><td>\u30a2\u30af\u30bb\u30b9\u30ed\u30b0<br>\u30a8\u30e9\u30fc\u30ed\u30b0<br>\u7d71\u8a08\u60c5\u5831<\/td><td>\u91cd\u8981<\/td><\/tr><tr><td>UI\/UX<\/td><td>\u30ec\u30b9\u30dd\u30f3\u30b7\u30d6\u30c7\u30b6\u30a4\u30f3<br>\u30c0\u30c3\u30b7\u30e5\u30dc\u30fc\u30c9<br>\u691c\u7d22\u30fb\u30d5\u30a3\u30eb\u30bf\u30ea\u30f3\u30b0<\/td><td>\u4e2d\u8981<\/td><\/tr><\/tbody><\/table><\/figure>\n<\/div>\n\n\n<h3 class=\"wp-block-heading\" id=\"i-2\">Laravel\u3067\u7ba1\u7406\u753b\u9762\u3092\u4f5c\u308b\u30e1\u30ea\u30c3\u30c8<\/h3>\n\n\n\n<p>Laravel\u3092\u4f7f\u7528\u3057\u3066\u7ba1\u7406\u753b\u9762\u3092\u958b\u767a\u3059\u308b\u3053\u3068\u306b\u306f\u3001\u4ee5\u4e0b\u306e\u3088\u3046\u306a\u5927\u304d\u306a\u30e1\u30ea\u30c3\u30c8\u304c\u3042\u308a\u307e\u3059\uff1a<\/p>\n\n\n\n<ol class=\"wp-block-list\">\n<li><strong>\u5145\u5b9f\u3057\u305f\u8a8d\u8a3c\u6a5f\u80fd<\/strong><\/li>\n<\/ol>\n\n\n\n<ul class=\"wp-block-list\">\n<li>Laravel Breeze\u306b\u3088\u308b\u8a8d\u8a3c\u6a5f\u80fd\u306e\u7c21\u5358\u306a\u5b9f\u88c5<\/li>\n\n\n\n<li>Laravel Sanctum\u306b\u3088\u308bAPI\u30c8\u30fc\u30af\u30f3\u8a8d\u8a3c<\/li>\n\n\n\n<li>\u67d4\u8edf\u306a\u30dd\u30ea\u30b7\u30fc\u3068\u30b2\u30fc\u30c8\u6a5f\u80fd<\/li>\n<\/ul>\n\n\n\n<ol class=\"wp-block-list\">\n<li><strong>\u5805\u7262\u306a\u30bb\u30ad\u30e5\u30ea\u30c6\u30a3\u6a5f\u80fd<\/strong><\/li>\n<\/ol>\n\n\n\n<ul class=\"wp-block-list\">\n<li>XSS\u5bfe\u7b56<\/li>\n\n\n\n<li>CSRF\u5bfe\u7b56<\/li>\n\n\n\n<li>SQL\u30a4\u30f3\u30b8\u30a7\u30af\u30b7\u30e7\u30f3\u5bfe\u7b56<\/li>\n\n\n\n<li>\u30bb\u30c3\u30b7\u30e7\u30f3\u7ba1\u7406<\/li>\n<\/ul>\n\n\n\n<ol class=\"wp-block-list\">\n<li><strong>\u958b\u767a\u52b9\u7387\u306e\u5411\u4e0a<\/strong><\/li>\n<\/ol>\n\n\n\n<ul class=\"wp-block-list\">\n<li>Artisan\u30b3\u30de\u30f3\u30c9\u306b\u3088\u308b\u81ea\u52d5\u751f\u6210<\/li>\n\n\n\n<li>\u30a8\u30ec\u30ac\u30f3\u30c8\u306aORM\uff08Eloquent\uff09<\/li>\n\n\n\n<li>\u8c4a\u5bcc\u306a\u30df\u30c9\u30eb\u30a6\u30a7\u30a2\u6a5f\u80fd<\/li>\n\n\n\n<li>\u5145\u5b9f\u3057\u305f\u30c6\u30b9\u30c8\u74b0\u5883<\/li>\n<\/ul>\n\n\n\n<ol class=\"wp-block-list\">\n<li><strong>\u4fdd\u5b88\u6027\u306e\u9ad8\u3055<\/strong><\/li>\n<\/ol>\n\n\n\n<ul class=\"wp-block-list\">\n<li>MVC \u30a2\u30fc\u30ad\u30c6\u30af\u30c1\u30e3\u306e\u63a1\u7528<\/li>\n\n\n\n<li>\u30e2\u30b8\u30e5\u30fc\u30eb\u5316\u3055\u308c\u305f\u69cb\u9020<\/li>\n\n\n\n<li>\u660e\u78ba\u306a\u547d\u540d\u898f\u5247<\/li>\n\n\n\n<li>\u8c4a\u5bcc\u306a\u30c9\u30ad\u30e5\u30e1\u30f3\u30c8<\/li>\n<\/ul>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"i-3\">\u7ba1\u7406\u753b\u9762\u5b9f\u88c5\u306e\u5168\u4f53\u306e\u6d41\u308c<\/h3>\n\n\n\n<p>Laravel\u3067\u306e\u7ba1\u7406\u753b\u9762\u5b9f\u88c5\u306f\u3001\u4ee5\u4e0b\u306e\u624b\u9806\u3067\u9032\u3081\u3066\u3044\u304d\u307e\u3059\uff1a<\/p>\n\n\n\n<ol class=\"wp-block-list\">\n<li><strong>\u30d7\u30ed\u30b8\u30a7\u30af\u30c8\u306e\u521d\u671f\u8a2d\u5b9a<\/strong><\/li>\n<\/ol>\n\n\n\n<pre class=\"EnlighterJSRAW\" data-enlighter-language=\"generic\" data-enlighter-theme=\"\" data-enlighter-highlight=\"\" data-enlighter-linenumbers=\"\" data-enlighter-lineoffset=\"\" data-enlighter-title=\"\" data-enlighter-group=\"\">   composer create-project laravel\/laravel admin-panel\n   cd admin-panel\n   composer require laravel\/breeze\n   php artisan breeze:install<\/pre>\n\n\n\n<ol start=\"2\" class=\"wp-block-list\">\n<li><strong>\u30c7\u30fc\u30bf\u30d9\u30fc\u30b9\u8a2d\u8a08\u3068\u30e2\u30c7\u30eb\u4f5c\u6210<\/strong><\/li>\n<\/ol>\n\n\n\n<pre class=\"EnlighterJSRAW\" data-enlighter-language=\"generic\" data-enlighter-theme=\"\" data-enlighter-highlight=\"\" data-enlighter-linenumbers=\"\" data-enlighter-lineoffset=\"\" data-enlighter-title=\"\" data-enlighter-group=\"\">   php artisan make:model Admin -m\n   php artisan make:model Role -m\n   php artisan make:model Permission -m<\/pre>\n\n\n\n<ol start=\"3\" class=\"wp-block-list\">\n<li><strong>\u8a8d\u8a3c\u30b7\u30b9\u30c6\u30e0\u306e\u5b9f\u88c5<\/strong><\/li>\n<\/ol>\n\n\n\n<ul class=\"wp-block-list\">\n<li>\u7ba1\u7406\u8005\u7528\u306e\u30ac\u30fc\u30c9\u8a2d\u5b9a<\/li>\n\n\n\n<li>\u30ed\u30b0\u30a4\u30f3\u30fb\u30ed\u30b0\u30a2\u30a6\u30c8\u6a5f\u80fd\u306e\u5b9f\u88c5<\/li>\n\n\n\n<li>\u30d1\u30b9\u30ef\u30fc\u30c9\u30ea\u30bb\u30c3\u30c8\u6a5f\u80fd\u306e\u8ffd\u52a0<\/li>\n<\/ul>\n\n\n\n<ol start=\"3\" class=\"wp-block-list\">\n<li><strong>\u57fa\u672cCRUD\u6a5f\u80fd\u306e\u5b9f\u88c5<\/strong><\/li>\n<\/ol>\n\n\n\n<ul class=\"wp-block-list\">\n<li>\u30b3\u30f3\u30c8\u30ed\u30fc\u30e9\u30fc\u306e\u4f5c\u6210<\/li>\n\n\n\n<li>\u30eb\u30fc\u30c6\u30a3\u30f3\u30b0\u306e\u8a2d\u5b9a<\/li>\n\n\n\n<li>\u30d3\u30e5\u30fc\u306e\u4f5c\u6210<\/li>\n<\/ul>\n\n\n\n<ol start=\"3\" class=\"wp-block-list\">\n<li><strong>\u6a29\u9650\u7ba1\u7406\u30b7\u30b9\u30c6\u30e0\u306e\u5b9f\u88c5<\/strong><\/li>\n<\/ol>\n\n\n\n<ul class=\"wp-block-list\">\n<li>\u30ed\u30fc\u30eb\u306e\u5b9a\u7fa9<\/li>\n\n\n\n<li>\u30d1\u30fc\u30df\u30c3\u30b7\u30e7\u30f3\u306e\u8a2d\u5b9a<\/li>\n\n\n\n<li>\u30a2\u30af\u30bb\u30b9\u5236\u5fa1\u306e\u5b9f\u88c5<\/li>\n<\/ul>\n\n\n\n<ol start=\"3\" class=\"wp-block-list\">\n<li><strong>UI\/UX\u306e\u5b9f\u88c5<\/strong><\/li>\n<\/ol>\n\n\n\n<ul class=\"wp-block-list\">\n<li>\u7ba1\u7406\u753b\u9762\u30c6\u30f3\u30d7\u30ec\u30fc\u30c8\u306e\u9069\u7528<\/li>\n\n\n\n<li>\u30ec\u30b9\u30dd\u30f3\u30b7\u30d6\u30c7\u30b6\u30a4\u30f3\u306e\u5b9f\u88c5<\/li>\n\n\n\n<li>\u30c7\u30fc\u30bf\u30c6\u30fc\u30d6\u30eb\u306e\u5b9f\u88c5<\/li>\n<\/ul>\n\n\n\n<ol start=\"3\" class=\"wp-block-list\">\n<li><strong>\u30c6\u30b9\u30c8\u3068\u30c7\u30d0\u30c3\u30b0<\/strong><\/li>\n<\/ol>\n\n\n\n<ul class=\"wp-block-list\">\n<li>\u30e6\u30cb\u30c3\u30c8\u30c6\u30b9\u30c8\u306e\u4f5c\u6210<\/li>\n\n\n\n<li>\u6a5f\u80fd\u30c6\u30b9\u30c8\u306e\u5b9f\u65bd<\/li>\n\n\n\n<li>\u30bb\u30ad\u30e5\u30ea\u30c6\u30a3\u30c6\u30b9\u30c8\u306e\u5b9f\u884c<\/li>\n<\/ul>\n\n\n\n<p>\u3053\u306e\u5b9f\u88c5\u30d5\u30ed\u30fc\u306b\u5f93\u3046\u3053\u3068\u3067\u3001\u30bb\u30ad\u30e5\u30a2\u3067\u4fdd\u5b88\u6027\u306e\u9ad8\u3044\u7ba1\u7406\u753b\u9762\u3092\u69cb\u7bc9\u3059\u308b\u3053\u3068\u304c\u3067\u304d\u307e\u3059\u3002\u4ee5\u964d\u306e\u30bb\u30af\u30b7\u30e7\u30f3\u3067\u306f\u3001\u5404\u30b9\u30c6\u30c3\u30d7\u306e\u8a73\u7d30\u306a\u5b9f\u88c5\u65b9\u6cd5\u306b\u3064\u3044\u3066\u89e3\u8aac\u3057\u3066\u3044\u304d\u307e\u3059\u3002<\/p>\n\n\n\n<h2 class=\"wp-block-heading\" id=\"i-4\">\u7ba1\u7406\u753b\u9762\u306e\u74b0\u5883\u69cb\u7bc9\u3068\u521d\u671f\u8a2d\u5b9a<\/h2>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"i-5\">\u5fc5\u8981\u306a\u30d1\u30c3\u30b1\u30fc\u30b8\u3068\u30a4\u30f3\u30b9\u30c8\u30fc\u30eb\u624b\u9806<\/h3>\n\n\n\n<p>Laravel\u7ba1\u7406\u753b\u9762\u306e\u69cb\u7bc9\u306b\u5fc5\u8981\u306a\u74b0\u5883\u3068\u30d1\u30c3\u30b1\u30fc\u30b8\u306e\u30bb\u30c3\u30c8\u30a2\u30c3\u30d7\u624b\u9806\u3092\u89e3\u8aac\u3057\u307e\u3059\u3002<\/p>\n\n\n\n<p><strong>1. \u57fa\u672c\u74b0\u5883\u306e\u8981\u4ef6<\/strong><\/p>\n\n\n<div id=\"id-f245307e-f958-42ac-8ac9-2466113fceba\">\n<figure class=\"wp-block-table\"><table class=\"has-fixed-layout\"><thead><tr><th>\u8981\u4ef6<\/th><th>\u63a8\u5968\u30d0\u30fc\u30b8\u30e7\u30f3<\/th><th>\u5099\u8003<\/th><\/tr><\/thead><tbody><tr><td>PHP<\/td><td>&gt;= 8.1<\/td><td>\u6700\u65b0\u306e\u5b89\u5b9a\u7248\u3092\u63a8\u5968<\/td><\/tr><tr><td>Composer<\/td><td>2.x<\/td><td>\u4f9d\u5b58\u95a2\u4fc2\u7ba1\u7406\u7528<\/td><\/tr><tr><td>MySQL\/PostgreSQL<\/td><td>\u6700\u65b0\u306e\u5b89\u5b9a\u7248<\/td><td>\u3069\u3061\u3089\u3067\u3082\u53ef<\/td><\/tr><tr><td>Node.js &amp; npm<\/td><td>\u6700\u65b0\u306eLTS\u7248<\/td><td>\u30d5\u30ed\u30f3\u30c8\u30a8\u30f3\u30c9\u958b\u767a\u7528<\/td><\/tr><\/tbody><\/table><\/figure>\n<\/div>\n\n\n<p><strong>2. \u30d7\u30ed\u30b8\u30a7\u30af\u30c8\u306e\u4f5c\u6210\u3068\u521d\u671f\u8a2d\u5b9a<\/strong><\/p>\n\n\n\n<pre class=\"EnlighterJSRAW\" data-enlighter-language=\"generic\" data-enlighter-theme=\"\" data-enlighter-highlight=\"\" data-enlighter-linenumbers=\"\" data-enlighter-lineoffset=\"\" data-enlighter-title=\"\" data-enlighter-group=\"\"># \u65b0\u898f\u30d7\u30ed\u30b8\u30a7\u30af\u30c8\u306e\u4f5c\u6210\ncomposer create-project laravel\/laravel admin-panel\ncd admin-panel\n\n# \u8a8d\u8a3c\u30d1\u30c3\u30b1\u30fc\u30b8\u306e\u30a4\u30f3\u30b9\u30c8\u30fc\u30eb\uff08Breeze\u3092\u4f7f\u7528\u3059\u308b\u5834\u5408\uff09\ncomposer require laravel\/breeze\nphp artisan breeze:install\nnpm install\nnpm run dev\n\n# \u7ba1\u7406\u753b\u9762\u7528\u306e\u8ffd\u52a0\u30d1\u30c3\u30b1\u30fc\u30b8\ncomposer require spatie\/laravel-permission  # \u6a29\u9650\u7ba1\u7406\ncomposer require yajra\/laravel-datatables  # \u30c7\u30fc\u30bf\u30c6\u30fc\u30d6\u30eb\ncomposer require intervention\/image         # \u753b\u50cf\u51e6\u7406<\/pre>\n\n\n\n<p><strong>3. \u8a2d\u5b9a\u30d5\u30a1\u30a4\u30eb\u306e\u8abf\u6574<\/strong><\/p>\n\n\n\n<pre class=\"EnlighterJSRAW\" data-enlighter-language=\"generic\" data-enlighter-theme=\"\" data-enlighter-highlight=\"\" data-enlighter-linenumbers=\"\" data-enlighter-lineoffset=\"\" data-enlighter-title=\"\" data-enlighter-group=\"\">\/\/ config\/auth.php\n'guards' =&gt; [\n    'web' =&gt; [\n        'driver' =&gt; 'session',\n        'provider' =&gt; 'users',\n    ],\n    'admin' =&gt; [                      \/\/ \u7ba1\u7406\u8005\u7528\u30ac\u30fc\u30c9\u306e\u8ffd\u52a0\n        'driver' =&gt; 'session',\n        'provider' =&gt; 'admins',\n    ],\n],\n\n'providers' =&gt; [\n    'users' =&gt; [\n        'driver' =&gt; 'eloquent',\n        'model' =&gt; App\\Models\\User::class,\n    ],\n    'admins' =&gt; [                     \/\/ \u7ba1\u7406\u8005\u7528\u30d7\u30ed\u30d0\u30a4\u30c0\u30fc\u306e\u8ffd\u52a0\n        'driver' =&gt; 'eloquent',\n        'model' =&gt; App\\Models\\Admin::class,\n    ],\n],<\/pre>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"i-6\">\u30c7\u30fc\u30bf\u30d9\u30fc\u30b9\u8a2d\u8a08\u306e\u30d9\u30b9\u30c8\u30d7\u30e9\u30af\u30c6\u30a3\u30b9<\/h3>\n\n\n\n<p>\u7ba1\u7406\u753b\u9762\u306e\u30c7\u30fc\u30bf\u30d9\u30fc\u30b9\u8a2d\u8a08\u306b\u304a\u3044\u3066\u3001\u4ee5\u4e0b\u306e\u30d9\u30b9\u30c8\u30d7\u30e9\u30af\u30c6\u30a3\u30b9\u3092\u8003\u616e\u3057\u307e\u3059\u3002<\/p>\n\n\n\n<p><strong>1. \u57fa\u672c\u30c6\u30fc\u30d6\u30eb\u69cb\u9020<\/strong><\/p>\n\n\n\n<pre class=\"EnlighterJSRAW\" data-enlighter-language=\"generic\" data-enlighter-theme=\"\" data-enlighter-highlight=\"\" data-enlighter-linenumbers=\"\" data-enlighter-lineoffset=\"\" data-enlighter-title=\"\" data-enlighter-group=\"\">\/\/ database\/migrations\/2024_02_19_000001_create_admins_table.php\npublic function up()\n{\n    Schema::create('admins', function (Blueprint $table) {\n        $table-&gt;id();\n        $table-&gt;string('name');\n        $table-&gt;string('email')-&gt;unique();\n        $table-&gt;string('password');\n        $table-&gt;boolean('is_super_admin')-&gt;default(false);\n        $table-&gt;timestamp('last_login_at')-&gt;nullable();\n        $table-&gt;rememberToken();\n        $table-&gt;timestamps();\n        $table-&gt;softDeletes();  \/\/ \u8ad6\u7406\u524a\u9664\u7528\n    });\n}\n\n\/\/ database\/migrations\/2024_02_19_000002_create_roles_table.php\npublic function up()\n{\n    Schema::create('roles', function (Blueprint $table) {\n        $table-&gt;id();\n        $table-&gt;string('name');\n        $table-&gt;string('description')-&gt;nullable();\n        $table-&gt;timestamps();\n    });\n}\n\n\/\/ database\/migrations\/2024_02_19_000003_create_permissions_table.php\npublic function up()\n{\n    Schema::create('permissions', function (Blueprint $table) {\n        $table-&gt;id();\n        $table-&gt;string('name');\n        $table-&gt;string('description')-&gt;nullable();\n        $table-&gt;timestamps();\n    });\n}<\/pre>\n\n\n\n<p><strong>\u30c6\u30fc\u30d6\u30eb\u8a2d\u8a08\u306e\u30dd\u30a4\u30f3\u30c8\uff1a<\/strong><\/p>\n\n\n\n<ol class=\"wp-block-list\">\n<li>\u9069\u5207\u306a\u30a4\u30f3\u30c7\u30c3\u30af\u30b9\u8a2d\u5b9a<\/li>\n<\/ol>\n\n\n\n<ul class=\"wp-block-list\">\n<li>\u691c\u7d22\u30fb\u7d50\u5408\u3067\u4f7f\u7528\u3055\u308c\u308b\u30ab\u30e9\u30e0\u306b\u306f\u30a4\u30f3\u30c7\u30c3\u30af\u30b9\u3092\u4ed8\u4e0e<\/li>\n\n\n\n<li>\u8907\u5408\u30a4\u30f3\u30c7\u30c3\u30af\u30b9\u306f\u4f7f\u7528\u983b\u5ea6\u3092\u8003\u616e\u3057\u3066\u8a2d\u8a08<\/li>\n<\/ul>\n\n\n\n<ol class=\"wp-block-list\">\n<li>\u30c7\u30fc\u30bf\u6574\u5408\u6027\u306e\u78ba\u4fdd<\/li>\n<\/ol>\n\n\n\n<ul class=\"wp-block-list\">\n<li>\u5916\u90e8\u30ad\u30fc\u5236\u7d04\u306e\u9069\u5207\u306a\u8a2d\u5b9a<\/li>\n\n\n\n<li>\u5fc5\u8981\u306b\u5fdc\u3058\u3066\u30e6\u30cb\u30fc\u30af\u5236\u7d04\u3092\u8a2d\u5b9a<\/li>\n\n\n\n<li>NOT NULL\u5236\u7d04\u306e\u9069\u5207\u306a\u4f7f\u7528<\/li>\n<\/ul>\n\n\n\n<ol class=\"wp-block-list\">\n<li>\u30d1\u30d5\u30a9\u30fc\u30de\u30f3\u30b9\u3092\u8003\u616e\u3057\u305f\u8a2d\u8a08<\/li>\n<\/ol>\n\n\n\n<ul class=\"wp-block-list\">\n<li>\u6b63\u898f\u5316\u3068\u975e\u6b63\u898f\u5316\u306e\u30d0\u30e9\u30f3\u30b9<\/li>\n\n\n\n<li>\u9069\u5207\u306a\u30c7\u30fc\u30bf\u578b\u306e\u9078\u629e<\/li>\n\n\n\n<li>\u5fc5\u8981\u306b\u5fdc\u3058\u3066\u30d1\u30fc\u30c6\u30a3\u30b7\u30e7\u30cb\u30f3\u30b0\u3092\u691c\u8a0e<\/li>\n<\/ul>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"i-7\">\u8a8d\u8a3c\u30b7\u30b9\u30c6\u30e0\u306e\u5c0e\u5165\u65b9\u6cd5<\/h3>\n\n\n\n<p>Laravel\u306e\u7ba1\u7406\u753b\u9762\u306b\u304a\u3051\u308b\u5805\u7262\u306a\u8a8d\u8a3c\u30b7\u30b9\u30c6\u30e0\u306e\u5b9f\u88c5\u65b9\u6cd5\u3092\u89e3\u8aac\u3057\u307e\u3059\u3002<\/p>\n\n\n\n<p><strong>1. \u8a8d\u8a3c\u30b3\u30f3\u30c8\u30ed\u30fc\u30e9\u30fc\u306e\u4f5c\u6210<\/strong><\/p>\n\n\n\n<pre class=\"EnlighterJSRAW\" data-enlighter-language=\"generic\" data-enlighter-theme=\"\" data-enlighter-highlight=\"\" data-enlighter-linenumbers=\"\" data-enlighter-lineoffset=\"\" data-enlighter-title=\"\" data-enlighter-group=\"\">\/\/ app\/Http\/Controllers\/Admin\/Auth\/LoginController.php\nnamespace App\\Http\\Controllers\\Admin\\Auth;\n\nuse App\\Http\\Controllers\\Controller;\nuse Illuminate\\Http\\Request;\nuse Illuminate\\Support\\Facades\\Auth;\n\nclass LoginController extends Controller\n{\n    public function showLoginForm()\n    {\n        return view('admin.auth.login');\n    }\n\n    public function login(Request $request)\n    {\n        $credentials = $request-&gt;validate([\n            'email' =&gt; ['required', 'email'],\n            'password' =&gt; ['required'],\n        ]);\n\n        if (Auth::guard('admin')-&gt;attempt($credentials)) {\n            $request-&gt;session()-&gt;regenerate();\n\n            \/\/ \u30ed\u30b0\u30a4\u30f3\u6642\u523b\u306e\u66f4\u65b0\n            Auth::guard('admin')-&gt;user()-&gt;update([\n                'last_login_at' =&gt; now()\n            ]);\n\n            return redirect()-&gt;intended(route('admin.dashboard'));\n        }\n\n        return back()-&gt;withErrors([\n            'email' =&gt; 'The provided credentials do not match our records.',\n        ]);\n    }\n}<\/pre>\n\n\n\n<p><strong>2. \u30df\u30c9\u30eb\u30a6\u30a7\u30a2\u306e\u8a2d\u5b9a<\/strong><\/p>\n\n\n\n<pre class=\"EnlighterJSRAW\" data-enlighter-language=\"generic\" data-enlighter-theme=\"\" data-enlighter-highlight=\"\" data-enlighter-linenumbers=\"\" data-enlighter-lineoffset=\"\" data-enlighter-title=\"\" data-enlighter-group=\"\">\/\/ app\/Http\/Middleware\/AdminAuthenticate.php\nnamespace App\\Http\\Middleware;\n\nuse Closure;\nuse Illuminate\\Auth\\Middleware\\Authenticate as Middleware;\n\nclass AdminAuthenticate extends Middleware\n{\n    protected function redirectTo($request)\n    {\n        if (! $request-&gt;expectsJson()) {\n            return route('admin.login');\n        }\n    }\n}\n\n\/\/ app\/Http\/Kernel.php\nprotected $routeMiddleware = [\n    \/\/ ...\n    'admin.auth' =&gt; \\App\\Http\\Middleware\\AdminAuthenticate::class,\n];<\/pre>\n\n\n\n<p><strong>3. \u30eb\u30fc\u30c6\u30a3\u30f3\u30b0\u306e\u8a2d\u5b9a<\/strong><\/p>\n\n\n\n<pre class=\"EnlighterJSRAW\" data-enlighter-language=\"generic\" data-enlighter-theme=\"\" data-enlighter-highlight=\"\" data-enlighter-linenumbers=\"\" data-enlighter-lineoffset=\"\" data-enlighter-title=\"\" data-enlighter-group=\"\">\/\/ routes\/admin.php\nRoute::middleware(['web'])-&gt;group(function () {\n    Route::get('login', [LoginController::class, 'showLoginForm'])\n        -&gt;name('admin.login');\n    Route::post('login', [LoginController::class, 'login']);\n\n    Route::middleware(['admin.auth'])-&gt;group(function () {\n        Route::get('dashboard', [DashboardController::class, 'index'])\n            -&gt;name('admin.dashboard');\n        \/\/ \u305d\u306e\u4ed6\u306e\u7ba1\u7406\u753b\u9762\u30eb\u30fc\u30c8\n    });\n});\n\n\/\/ app\/Providers\/RouteServiceProvider.php\npublic function boot()\n{\n    $this-&gt;routes(function () {\n        Route::prefix('admin')\n             -&gt;group(base_path('routes\/admin.php'));\n    });\n}<\/pre>\n\n\n\n<p>\u3053\u306e\u74b0\u5883\u69cb\u7bc9\u3068\u521d\u671f\u8a2d\u5b9a\u306b\u3088\u308a\u3001\u4ee5\u4e0b\u306e\u5229\u70b9\u304c\u5f97\u3089\u308c\u307e\u3059\uff1a<\/p>\n\n\n\n<ol class=\"wp-block-list\">\n<li>\u30bb\u30ad\u30e5\u30a2\u306a\u8a8d\u8a3c\u30b7\u30b9\u30c6\u30e0<\/li>\n\n\n\n<li>\u62e1\u5f35\u6027\u306e\u9ad8\u3044\u30c7\u30fc\u30bf\u30d9\u30fc\u30b9\u69cb\u9020<\/li>\n\n\n\n<li>\u52b9\u7387\u7684\u306a\u958b\u767a\u74b0\u5883<\/li>\n\n\n\n<li>\u4fdd\u5b88\u6027\u306e\u9ad8\u3044\u30b3\u30fc\u30c9\u69cb\u6210<\/li>\n<\/ol>\n\n\n\n<p>\u6b21\u306e\u30b9\u30c6\u30c3\u30d7\u3067\u306f\u3001\u3053\u306e\u57fa\u76e4\u306e\u4e0a\u306bCRUD\u6a5f\u80fd\u3092\u5b9f\u88c5\u3057\u3066\u3044\u304d\u307e\u3059\u3002<\/p>\n\n\n\n<h2 class=\"wp-block-heading\" id=\"i-8\">\u57fa\u672c\u7684\u306aCRUD\u6a5f\u80fd\u306e\u5b9f\u88c5\u624b\u9806<\/h2>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"i-9\">\u30e2\u30c7\u30eb\u3068\u30de\u30a4\u30b0\u30ec\u30fc\u30b7\u30e7\u30f3\u306e\u4f5c\u6210\u65b9\u6cd5<\/h3>\n\n\n\n<p>\u7ba1\u7406\u753b\u9762\u3067\u3088\u304f\u4f7f\u7528\u3055\u308c\u308b\u5546\u54c1\u7ba1\u7406\u3092\u4f8b\u306b\u3001CRUD\u6a5f\u80fd\u306e\u5b9f\u88c5\u624b\u9806\u3092\u89e3\u8aac\u3057\u307e\u3059\u3002<\/p>\n\n\n\n<p><strong>1. \u30e2\u30c7\u30eb\u3068\u30de\u30a4\u30b0\u30ec\u30fc\u30b7\u30e7\u30f3\u306e\u751f\u6210<\/strong><\/p>\n\n\n\n<pre class=\"EnlighterJSRAW\" data-enlighter-language=\"generic\" data-enlighter-theme=\"\" data-enlighter-highlight=\"\" data-enlighter-linenumbers=\"\" data-enlighter-lineoffset=\"\" data-enlighter-title=\"\" data-enlighter-group=\"\"># \u30e2\u30c7\u30eb\u3001\u30de\u30a4\u30b0\u30ec\u30fc\u30b7\u30e7\u30f3\u3001\u30d5\u30a1\u30af\u30c8\u30ea\u30fc\u3001\u30b7\u30fc\u30c0\u30fc\u3092\u4e00\u62ec\u751f\u6210\nphp artisan make:model Product -mfsr<\/pre>\n\n\n\n<p><strong>2. \u30de\u30a4\u30b0\u30ec\u30fc\u30b7\u30e7\u30f3\u30d5\u30a1\u30a4\u30eb\u306e\u5b9f\u88c5<\/strong><\/p>\n\n\n\n<pre class=\"EnlighterJSRAW\" data-enlighter-language=\"generic\" data-enlighter-theme=\"\" data-enlighter-highlight=\"\" data-enlighter-linenumbers=\"\" data-enlighter-lineoffset=\"\" data-enlighter-title=\"\" data-enlighter-group=\"\">\/\/ database\/migrations\/xxxx_xx_xx_create_products_table.php\npublic function up()\n{\n    Schema::create('products', function (Blueprint $table) {\n        $table-&gt;id();\n        $table-&gt;string('name');\n        $table-&gt;text('description')-&gt;nullable();\n        $table-&gt;decimal('price', 10, 2);\n        $table-&gt;integer('stock')-&gt;default(0);\n        $table-&gt;string('sku')-&gt;unique();\n        $table-&gt;boolean('is_active')-&gt;default(true);\n        $table-&gt;string('image_path')-&gt;nullable();\n        $table-&gt;timestamps();\n        $table-&gt;softDeletes();  \/\/ \u8ad6\u7406\u524a\u9664\u7528\n\n        \/\/ \u30a4\u30f3\u30c7\u30c3\u30af\u30b9\u306e\u8ffd\u52a0\n        $table-&gt;index(['name', 'sku']);\n        $table-&gt;index('is_active');\n    });\n}<\/pre>\n\n\n\n<p><strong>3. \u30e2\u30c7\u30eb\u306e\u5b9f\u88c5<\/strong><\/p>\n\n\n\n<pre class=\"EnlighterJSRAW\" data-enlighter-language=\"generic\" data-enlighter-theme=\"\" data-enlighter-highlight=\"\" data-enlighter-linenumbers=\"\" data-enlighter-lineoffset=\"\" data-enlighter-title=\"\" data-enlighter-group=\"\">\/\/ app\/Models\/Product.php\nnamespace App\\Models;\n\nuse Illuminate\\Database\\Eloquent\\Factories\\HasFactory;\nuse Illuminate\\Database\\Eloquent\\Model;\nuse Illuminate\\Database\\Eloquent\\SoftDeletes;\n\nclass Product extends Model\n{\n    use HasFactory, SoftDeletes;\n\n    protected $fillable = [\n        'name',\n        'description',\n        'price',\n        'stock',\n        'sku',\n        'is_active',\n        'image_path'\n    ];\n\n    protected $casts = [\n        'price' =&gt; 'decimal:2',\n        'stock' =&gt; 'integer',\n        'is_active' =&gt; 'boolean'\n    ];\n\n    \/\/ \u30d0\u30ea\u30c7\u30fc\u30b7\u30e7\u30f3\u30eb\u30fc\u30eb\n    public static $rules = [\n        'name' =&gt; 'required|max:255',\n        'description' =&gt; 'nullable',\n        'price' =&gt; 'required|numeric|min:0',\n        'stock' =&gt; 'required|integer|min:0',\n        'sku' =&gt; 'required|unique:products,sku',\n        'is_active' =&gt; 'boolean',\n        'image' =&gt; 'nullable|image|max:2048'  \/\/ 2MB\u5236\u9650\n    ];\n\n    \/\/ \u30b9\u30b3\u30fc\u30d7\u306e\u5b9a\u7fa9\n    public function scopeActive($query)\n    {\n        return $query-&gt;where('is_active', true);\n    }\n\n    public function scopeLowStock($query, $threshold = 5)\n    {\n        return $query-&gt;where('stock', '&lt;=', $threshold);\n    }\n}<\/pre>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"i-10\">\u30b3\u30f3\u30c8\u30ed\u30fc\u30e9\u30fc\u306e\u5b9f\u88c5\u3068\u30eb\u30fc\u30c6\u30a3\u30f3\u30b0\u8a2d\u5b9a<\/h3>\n\n\n\n<p><strong>1. \u30ea\u30bd\u30fc\u30b9\u30b3\u30f3\u30c8\u30ed\u30fc\u30e9\u30fc\u306e\u751f\u6210<\/strong><\/p>\n\n\n\n<pre class=\"EnlighterJSRAW\" data-enlighter-language=\"generic\" data-enlighter-theme=\"\" data-enlighter-highlight=\"\" data-enlighter-linenumbers=\"\" data-enlighter-lineoffset=\"\" data-enlighter-title=\"\" data-enlighter-group=\"\">php artisan make:controller Admin\/ProductController --resource<\/pre>\n\n\n\n<p><strong>2. \u30b3\u30f3\u30c8\u30ed\u30fc\u30e9\u30fc\u306e\u5b9f\u88c5<\/strong><\/p>\n\n\n\n<pre class=\"EnlighterJSRAW\" data-enlighter-language=\"generic\" data-enlighter-theme=\"\" data-enlighter-highlight=\"\" data-enlighter-linenumbers=\"\" data-enlighter-lineoffset=\"\" data-enlighter-title=\"\" data-enlighter-group=\"\">\/\/ app\/Http\/Controllers\/Admin\/ProductController.php\nnamespace App\\Http\\Controllers\\Admin;\n\nuse App\\Http\\Controllers\\Controller;\nuse App\\Models\\Product;\nuse Illuminate\\Http\\Request;\nuse Illuminate\\Support\\Facades\\Storage;\n\nclass ProductController extends Controller\n{\n    public function index()\n    {\n        $products = Product::latest()-&gt;paginate(20);\n        return view('admin.products.index', compact('products'));\n    }\n\n    public function create()\n    {\n        return view('admin.products.create');\n    }\n\n    public function store(Request $request)\n    {\n        \/\/ \u30d0\u30ea\u30c7\u30fc\u30b7\u30e7\u30f3\n        $validated = $request-&gt;validate(Product::$rules);\n\n        \/\/ \u753b\u50cf\u306e\u30a2\u30c3\u30d7\u30ed\u30fc\u30c9\u51e6\u7406\n        if ($request-&gt;hasFile('image')) {\n            $path = $request-&gt;file('image')-&gt;store('products', 'public');\n            $validated['image_path'] = $path;\n        }\n\n        \/\/ \u30c7\u30fc\u30bf\u306e\u4fdd\u5b58\n        Product::create($validated);\n\n        return redirect()\n            -&gt;route('admin.products.index')\n            -&gt;with('success', '\u5546\u54c1\u3092\u767b\u9332\u3057\u307e\u3057\u305f\u3002');\n    }\n\n    public function edit(Product $product)\n    {\n        return view('admin.products.edit', compact('product'));\n    }\n\n    public function update(Request $request, Product $product)\n    {\n        \/\/ SKU\u306e\u30e6\u30cb\u30fc\u30af\u5236\u7d04\u3092\u73fe\u5728\u306e\u5546\u54c1\u3092\u9664\u5916\u3057\u3066\u691c\u8a3c\n        $rules = Product::$rules;\n        $rules['sku'] = 'required|unique:products,sku,' . $product-&gt;id;\n\n        $validated = $request-&gt;validate($rules);\n\n        \/\/ \u753b\u50cf\u306e\u66f4\u65b0\u51e6\u7406\n        if ($request-&gt;hasFile('image')) {\n            \/\/ \u53e4\u3044\u753b\u50cf\u306e\u524a\u9664\n            if ($product-&gt;image_path) {\n                Storage::disk('public')-&gt;delete($product-&gt;image_path);\n            }\n            $path = $request-&gt;file('image')-&gt;store('products', 'public');\n            $validated['image_path'] = $path;\n        }\n\n        $product-&gt;update($validated);\n\n        return redirect()\n            -&gt;route('admin.products.index')\n            -&gt;with('success', '\u5546\u54c1\u60c5\u5831\u3092\u66f4\u65b0\u3057\u307e\u3057\u305f\u3002');\n    }\n\n    public function destroy(Product $product)\n    {\n        \/\/ \u753b\u50cf\u306e\u524a\u9664\n        if ($product-&gt;image_path) {\n            Storage::disk('public')-&gt;delete($product-&gt;image_path);\n        }\n\n        $product-&gt;delete();\n\n        return redirect()\n            -&gt;route('admin.products.index')\n            -&gt;with('success', '\u5546\u54c1\u3092\u524a\u9664\u3057\u307e\u3057\u305f\u3002');\n    }\n}<\/pre>\n\n\n\n<p><strong>3. \u30eb\u30fc\u30c6\u30a3\u30f3\u30b0\u306e\u8a2d\u5b9a<\/strong><\/p>\n\n\n\n<pre class=\"EnlighterJSRAW\" data-enlighter-language=\"generic\" data-enlighter-theme=\"\" data-enlighter-highlight=\"\" data-enlighter-linenumbers=\"\" data-enlighter-lineoffset=\"\" data-enlighter-title=\"\" data-enlighter-group=\"\">\/\/ routes\/admin.php\nRoute::middleware(['admin.auth'])-&gt;group(function () {\n    Route::resource('products', ProductController::class);\n});<\/pre>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"i-11\">\u30d3\u30e5\u30fc\u30c6\u30f3\u30d7\u30ec\u30fc\u30c8\u306e\u4f5c\u6210\u3068\u30ab\u30b9\u30bf\u30de\u30a4\u30ba<\/h3>\n\n\n\n<p><strong>1. \u4e00\u89a7\u8868\u793a\u30d3\u30e5\u30fc<\/strong><\/p>\n\n\n\n<pre class=\"EnlighterJSRAW\" data-enlighter-language=\"generic\" data-enlighter-theme=\"\" data-enlighter-highlight=\"\" data-enlighter-linenumbers=\"\" data-enlighter-lineoffset=\"\" data-enlighter-title=\"\" data-enlighter-group=\"\">&lt;!-- resources\/views\/admin\/products\/index.blade.php --&gt;\n&lt;x-app-layout&gt;\n    &lt;x-slot name=\"header\"&gt;\n        &lt;h2 class=\"text-xl font-semibold\"&gt;\u5546\u54c1\u7ba1\u7406&lt;\/h2&gt;\n    &lt;\/x-slot&gt;\n\n    &lt;div class=\"py-12\"&gt;\n        &lt;div class=\"max-w-7xl mx-auto sm:px-6 lg:px-8\"&gt;\n            &lt;!-- \u6210\u529f\u30e1\u30c3\u30bb\u30fc\u30b8 --&gt;\n            @if (session('success'))\n                &lt;div class=\"bg-green-100 border border-green-400 text-green-700 px-4 py-3 rounded mb-4\"&gt;\n                    {{ session('success') }}\n                &lt;\/div&gt;\n            @endif\n\n            &lt;!-- \u5546\u54c1\u4e00\u89a7\u30c6\u30fc\u30d6\u30eb --&gt;\n            &lt;div class=\"bg-white overflow-hidden shadow-sm sm:rounded-lg\"&gt;\n                &lt;div class=\"p-6 bg-white border-b border-gray-200\"&gt;\n                    &lt;div class=\"flex justify-between mb-4\"&gt;\n                        &lt;h3 class=\"text-lg font-medium\"&gt;\u5546\u54c1\u4e00\u89a7&lt;\/h3&gt;\n                        &lt;a href=\"{{ route('admin.products.create') }}\" \n                           class=\"bg-blue-500 hover:bg-blue-700 text-white font-bold py-2 px-4 rounded\"&gt;\n                            \u65b0\u898f\u5546\u54c1\u767b\u9332\n                        &lt;\/a&gt;\n                    &lt;\/div&gt;\n\n                    &lt;table class=\"min-w-full divide-y divide-gray-200\"&gt;\n                        &lt;thead class=\"bg-gray-50\"&gt;\n                            &lt;tr&gt;\n                                &lt;th class=\"px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider\"&gt;\n                                    \u5546\u54c1\u540d\n                                &lt;\/th&gt;\n                                &lt;th class=\"px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider\"&gt;\n                                    \u4fa1\u683c\n                                &lt;\/th&gt;\n                                &lt;th class=\"px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider\"&gt;\n                                    \u5728\u5eab\u6570\n                                &lt;\/th&gt;\n                                &lt;th class=\"px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider\"&gt;\n                                    \u30b9\u30c6\u30fc\u30bf\u30b9\n                                &lt;\/th&gt;\n                                &lt;th class=\"px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider\"&gt;\n                                    \u64cd\u4f5c\n                                &lt;\/th&gt;\n                            &lt;\/tr&gt;\n                        &lt;\/thead&gt;\n                        &lt;tbody class=\"bg-white divide-y divide-gray-200\"&gt;\n                            @foreach ($products as $product)\n                                &lt;tr&gt;\n                                    &lt;td class=\"px-6 py-4 whitespace-nowrap\"&gt;\n                                        &lt;div class=\"flex items-center\"&gt;\n                                            @if ($product-&gt;image_path)\n                                                &lt;img src=\"{{ Storage::url($product-&gt;image_path) }}\" \n                                                     class=\"h-10 w-10 rounded-full object-cover mr-3\"&gt;\n                                            @endif\n                                            &lt;div class=\"text-sm font-medium text-gray-900\"&gt;\n                                                {{ $product-&gt;name }}\n                                            &lt;\/div&gt;\n                                        &lt;\/div&gt;\n                                    &lt;\/td&gt;\n                                    &lt;td class=\"px-6 py-4 whitespace-nowrap\"&gt;\n                                        &lt;div class=\"text-sm text-gray-900\"&gt;\n                                            \u00a5{{ number_format($product-&gt;price) }}\n                                        &lt;\/div&gt;\n                                    &lt;\/td&gt;\n                                    &lt;td class=\"px-6 py-4 whitespace-nowrap\"&gt;\n                                        &lt;div class=\"text-sm text-gray-900\"&gt;\n                                            {{ number_format($product-&gt;stock) }}\n                                        &lt;\/div&gt;\n                                    &lt;\/td&gt;\n                                    &lt;td class=\"px-6 py-4 whitespace-nowrap\"&gt;\n                                        &lt;span class=\"px-2 inline-flex text-xs leading-5 font-semibold rounded-full \n                                                   {{ $product-&gt;is_active ? 'bg-green-100 text-green-800' : 'bg-red-100 text-red-800' }}\"&gt;\n                                            {{ $product-&gt;is_active ? '\u8ca9\u58f2\u4e2d' : '\u505c\u6b62\u4e2d' }}\n                                        &lt;\/span&gt;\n                                    &lt;\/td&gt;\n                                    &lt;td class=\"px-6 py-4 whitespace-nowrap text-sm font-medium\"&gt;\n                                        &lt;a href=\"{{ route('admin.products.edit', $product) }}\" \n                                           class=\"text-indigo-600 hover:text-indigo-900 mr-3\"&gt;\n                                            \u7de8\u96c6\n                                        &lt;\/a&gt;\n                                        &lt;form class=\"inline-block\" method=\"POST\" \n                                              action=\"{{ route('admin.products.destroy', $product) }}\"\n                                              onsubmit=\"return confirm('\u672c\u5f53\u306b\u524a\u9664\u3057\u307e\u3059\u304b\uff1f');\"&gt;\n                                            @csrf\n                                            @method('DELETE')\n                                            &lt;button type=\"submit\" \n                                                    class=\"text-red-600 hover:text-red-900\"&gt;\n                                                \u524a\u9664\n                                            &lt;\/button&gt;\n                                        &lt;\/form&gt;\n                                    &lt;\/td&gt;\n                                &lt;\/tr&gt;\n                            @endforeach\n                        &lt;\/tbody&gt;\n                    &lt;\/table&gt;\n\n                    &lt;div class=\"mt-4\"&gt;\n                        {{ $products-&gt;links() }}\n                    &lt;\/div&gt;\n                &lt;\/div&gt;\n            &lt;\/div&gt;\n        &lt;\/div&gt;\n    &lt;\/div&gt;\n&lt;\/x-app-layout&gt;<\/pre>\n\n\n\n<p>\u3053\u306e\u30bb\u30af\u30b7\u30e7\u30f3\u3067\u306f\u3001\u4ee5\u4e0b\u306e\u5b9f\u88c5\u30dd\u30a4\u30f3\u30c8\u306b\u6ce8\u610f\u3092\u6255\u3063\u3066\u3044\u307e\u3059\uff1a<\/p>\n\n\n\n<ol class=\"wp-block-list\">\n<li>\u30e2\u30c7\u30eb\u306e\u5b9f\u88c5<\/li>\n<\/ol>\n\n\n\n<ul class=\"wp-block-list\">\n<li>\u9069\u5207\u306a\u30d5\u30a3\u30fc\u30eb\u30c9\u5b9a\u7fa9<\/li>\n\n\n\n<li>\u30d0\u30ea\u30c7\u30fc\u30b7\u30e7\u30f3\u30eb\u30fc\u30eb\u306e\u96c6\u4e2d\u7ba1\u7406<\/li>\n\n\n\n<li>\u30b9\u30b3\u30fc\u30d7\u306b\u3088\u308b\u30af\u30a8\u30ea\u306e\u518d\u5229\u7528\u6027\u5411\u4e0a<\/li>\n<\/ul>\n\n\n\n<ol class=\"wp-block-list\">\n<li>\u30b3\u30f3\u30c8\u30ed\u30fc\u30e9\u30fc\u306e\u5b9f\u88c5<\/li>\n<\/ol>\n\n\n\n<ul class=\"wp-block-list\">\n<li>\u30ea\u30bd\u30fc\u30b9\u30b3\u30f3\u30c8\u30ed\u30fc\u30e9\u30fc\u306e\u6d3b\u7528<\/li>\n\n\n\n<li>\u30d5\u30a1\u30a4\u30eb\u30a2\u30c3\u30d7\u30ed\u30fc\u30c9\u51e6\u7406\u306e\u9069\u5207\u306a\u5b9f\u88c5<\/li>\n\n\n\n<li>\u30a8\u30e9\u30fc\u30cf\u30f3\u30c9\u30ea\u30f3\u30b0\u3068\u30d5\u30e9\u30c3\u30b7\u30e5\u30e1\u30c3\u30bb\u30fc\u30b8<\/li>\n<\/ul>\n\n\n\n<ol class=\"wp-block-list\">\n<li>\u30d3\u30e5\u30fc\u306e\u5b9f\u88c5<\/li>\n<\/ol>\n\n\n\n<ul class=\"wp-block-list\">\n<li>\u30e2\u30c0\u30f3\u306aUI\/UX\u306e\u5b9f\u73fe<\/li>\n\n\n\n<li>\u518d\u5229\u7528\u53ef\u80fd\u306a\u30b3\u30f3\u30dd\u30fc\u30cd\u30f3\u30c8\u306e\u6d3b\u7528<\/li>\n\n\n\n<li>\u30ec\u30b9\u30dd\u30f3\u30b7\u30d6\u30c7\u30b6\u30a4\u30f3\u3078\u306e\u5bfe\u5fdc<\/li>\n<\/ul>\n\n\n\n<p>\u3053\u308c\u3089\u306e\u5b9f\u88c5\u306b\u3088\u308a\u3001\u4fdd\u5b88\u6027\u304c\u9ad8\u304f\u3001\u62e1\u5f35\u6027\u306e\u3042\u308bCRUD\u6a5f\u80fd\u3092\u5b9f\u73fe\u3067\u304d\u307e\u3059\u3002<\/p>\n\n\n\n<h2 class=\"wp-block-heading\" id=\"i-12\">\u30bb\u30ad\u30e5\u30ea\u30c6\u30a3\u5bfe\u7b56\u3068\u6a29\u9650\u7ba1\u7406<\/h2>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"i-13\">\u30ed\u30fc\u30eb\u30d9\u30fc\u30b9\u306e\u30a2\u30af\u30bb\u30b9\u5236\u5fa1\u306e\u5b9f\u88c5<\/h3>\n\n\n\n<p>Laravel\u3067\u306f\u3001Spatie\u306elaravel-permission\u30d1\u30c3\u30b1\u30fc\u30b8\u3092\u4f7f\u7528\u3057\u3066\u3001\u67d4\u8edf\u306a\u6a29\u9650\u7ba1\u7406\u30b7\u30b9\u30c6\u30e0\u3092\u5b9f\u88c5\u3067\u304d\u307e\u3059\u3002<\/p>\n\n\n\n<p><strong>1. \u30d1\u30c3\u30b1\u30fc\u30b8\u306e\u30a4\u30f3\u30b9\u30c8\u30fc\u30eb\u3068\u8a2d\u5b9a<\/strong><\/p>\n\n\n\n<pre class=\"EnlighterJSRAW\" data-enlighter-language=\"generic\" data-enlighter-theme=\"\" data-enlighter-highlight=\"\" data-enlighter-linenumbers=\"\" data-enlighter-lineoffset=\"\" data-enlighter-title=\"\" data-enlighter-group=\"\">composer require spatie\/laravel-permission\nphp artisan vendor:publish --provider=\"Spatie\\Permission\\PermissionServiceProvider\"\nphp artisan migrate<\/pre>\n\n\n\n<p><strong>2. \u30e2\u30c7\u30eb\u306e\u8a2d\u5b9a<\/strong><\/p>\n\n\n\n<pre class=\"EnlighterJSRAW\" data-enlighter-language=\"generic\" data-enlighter-theme=\"\" data-enlighter-highlight=\"\" data-enlighter-linenumbers=\"\" data-enlighter-lineoffset=\"\" data-enlighter-title=\"\" data-enlighter-group=\"\">\/\/ app\/Models\/Admin.php\nuse Spatie\\Permission\\Traits\\HasRoles;\n\nclass Admin extends Authenticatable\n{\n    use HasRoles;\n\n    \/\/ ... \u4ed6\u306e\u30b3\u30fc\u30c9\n}<\/pre>\n\n\n\n<p><strong>3. \u57fa\u672c\u7684\u306a\u6a29\u9650\u3068\u30ed\u30fc\u30eb\u306e\u5b9a\u7fa9<\/strong><\/p>\n\n\n\n<pre class=\"EnlighterJSRAW\" data-enlighter-language=\"generic\" data-enlighter-theme=\"\" data-enlighter-highlight=\"\" data-enlighter-linenumbers=\"\" data-enlighter-lineoffset=\"\" data-enlighter-title=\"\" data-enlighter-group=\"\">\/\/ database\/seeders\/RolePermissionSeeder.php\nuse Spatie\\Permission\\Models\\Role;\nuse Spatie\\Permission\\Models\\Permission;\n\nclass RolePermissionSeeder extends Seeder\n{\n    public function run()\n    {\n        \/\/ \u6a29\u9650\u306e\u5b9a\u7fa9\n        $permissions = [\n            'view_dashboard',\n            'manage_products',\n            'manage_users',\n            'manage_orders',\n            'manage_settings',\n            'view_reports'\n        ];\n\n        foreach ($permissions as $permission) {\n            Permission::create(['name' =&gt; $permission, 'guard_name' =&gt; 'admin']);\n        }\n\n        \/\/ \u30ed\u30fc\u30eb\u306e\u5b9a\u7fa9\u3068\u6a29\u9650\u306e\u5272\u308a\u5f53\u3066\n        $roles = [\n            'super_admin' =&gt; $permissions,\n            'manager' =&gt; [\n                'view_dashboard',\n                'manage_products',\n                'manage_orders',\n                'view_reports'\n            ],\n            'editor' =&gt; [\n                'view_dashboard',\n                'manage_products'\n            ]\n        ];\n\n        foreach ($roles as $roleName =&gt; $rolePermissions) {\n            $role = Role::create(['name' =&gt; $roleName, 'guard_name' =&gt; 'admin']);\n            $role-&gt;givePermissionTo($rolePermissions);\n        }\n    }\n}<\/pre>\n\n\n\n<p><strong>4. \u30df\u30c9\u30eb\u30a6\u30a7\u30a2\u306e\u5b9f\u88c5<\/strong><\/p>\n\n\n\n<pre class=\"EnlighterJSRAW\" data-enlighter-language=\"generic\" data-enlighter-theme=\"\" data-enlighter-highlight=\"\" data-enlighter-linenumbers=\"\" data-enlighter-lineoffset=\"\" data-enlighter-title=\"\" data-enlighter-group=\"\">\/\/ app\/Http\/Middleware\/CheckPermission.php\nnamespace App\\Http\\Middleware;\n\nuse Closure;\nuse Illuminate\\Http\\Request;\n\nclass CheckPermission\n{\n    public function handle(Request $request, Closure $next, $permission)\n    {\n        if (!auth('admin')-&gt;user()-&gt;hasPermissionTo($permission)) {\n            abort(403, '\u6a29\u9650\u304c\u3042\u308a\u307e\u305b\u3093\u3002');\n        }\n\n        return $next($request);\n    }\n}\n\n\/\/ app\/Http\/Kernel.php\nprotected $routeMiddleware = [\n    \/\/ ...\n    'permission' =&gt; \\App\\Http\\Middleware\\CheckPermission::class,\n];<\/pre>\n\n\n\n<p><strong>5. \u6a29\u9650\u30c1\u30a7\u30c3\u30af\u306e\u5b9f\u88c5\u4f8b<\/strong><\/p>\n\n\n\n<pre class=\"EnlighterJSRAW\" data-enlighter-language=\"generic\" data-enlighter-theme=\"\" data-enlighter-highlight=\"\" data-enlighter-linenumbers=\"\" data-enlighter-lineoffset=\"\" data-enlighter-title=\"\" data-enlighter-group=\"\">\/\/ routes\/admin.php\nRoute::middleware(['admin.auth'])-&gt;group(function () {\n    Route::get('\/dashboard', [DashboardController::class, 'index'])\n        -&gt;middleware('permission:view_dashboard');\n\n    Route::resource('products', ProductController::class)\n        -&gt;middleware('permission:manage_products');\n});\n\n\/\/ Blade\u3067\u306e\u6a29\u9650\u30c1\u30a7\u30c3\u30af\n@can('manage_products')\n    &lt;a href=\"{{ route('admin.products.create') }}\"&gt;\u65b0\u898f\u5546\u54c1\u767b\u9332&lt;\/a&gt;\n@endcan<\/pre>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"i-14\">\u30af\u30ed\u30b9\u30b5\u30a4\u30c8\u30b9\u30af\u30ea\u30d7\u30c6\u30a3\u30f3\u30b0\u5bfe\u7b56<\/h3>\n\n\n\n<p>XSS\u653b\u6483\u304b\u3089\u30a2\u30d7\u30ea\u30b1\u30fc\u30b7\u30e7\u30f3\u3092\u4fdd\u8b77\u3059\u308b\u305f\u3081\u306e\u5bfe\u7b56\u3092\u5b9f\u88c5\u3057\u307e\u3059\u3002<\/p>\n\n\n\n<p><strong>1. \u81ea\u52d5\u30a8\u30b9\u30b1\u30fc\u30d7\u306e\u6d3b\u7528<\/strong><\/p>\n\n\n\n<pre class=\"EnlighterJSRAW\" data-enlighter-language=\"generic\" data-enlighter-theme=\"\" data-enlighter-highlight=\"\" data-enlighter-linenumbers=\"\" data-enlighter-lineoffset=\"\" data-enlighter-title=\"\" data-enlighter-group=\"\">\/\/ config\/htmlpurifier.php\nreturn [\n    'encoding' =&gt; 'UTF-8',\n    'finalize' =&gt; true,\n    'cachePath' =&gt; storage_path('app\/purifier'),\n    'settings' =&gt; [\n        'default' =&gt; [\n            'HTML.Allowed' =&gt; 'p,b,i,u,strong,em,a[href|title],ul,ol,li',\n            'CSS.AllowedProperties' =&gt; 'font,font-size,font-weight,font-style,text-decoration',\n            'AutoFormat.AutoParagraph' =&gt; true,\n            'AutoFormat.RemoveEmpty' =&gt; true,\n        ],\n    ],\n];\n\n\/\/ app\/Helpers\/HtmlPurifier.php\nuse HTMLPurifier;\nuse HTMLPurifier_Config;\n\nfunction purify($dirty_html)\n{\n    $config = HTMLPurifier_Config::createDefault();\n    $config-&gt;loadArray(config('htmlpurifier'));\n    $purifier = new HTMLPurifier($config);\n    return $purifier-&gt;purify($dirty_html);\n}\n\n\/\/ \u4f7f\u7528\u4f8b\n$description = purify($request-&gt;input('description'));<\/pre>\n\n\n\n<p><strong>2. Content Security Policy (CSP)\u306e\u5b9f\u88c5<\/strong><\/p>\n\n\n\n<pre class=\"EnlighterJSRAW\" data-enlighter-language=\"generic\" data-enlighter-theme=\"\" data-enlighter-highlight=\"\" data-enlighter-linenumbers=\"\" data-enlighter-lineoffset=\"\" data-enlighter-title=\"\" data-enlighter-group=\"\">\/\/ app\/Http\/Middleware\/AddSecurityHeaders.php\nnamespace App\\Http\\Middleware;\n\nuse Closure;\nuse Illuminate\\Http\\Request;\n\nclass AddSecurityHeaders\n{\n    public function handle(Request $request, Closure $next)\n    {\n        $response = $next($request);\n\n        $response-&gt;headers-&gt;set('Content-Security-Policy', \"\n            default-src 'self';\n            script-src 'self' 'unsafe-inline' 'unsafe-eval' https:\/\/cdnjs.cloudflare.com;\n            style-src 'self' 'unsafe-inline' https:\/\/fonts.googleapis.com;\n            font-src 'self' https:\/\/fonts.gstatic.com;\n            img-src 'self' data: https:;\n            frame-src 'none';\n        \");\n\n        return $response;\n    }\n}<\/pre>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"i-15\">\u30bb\u30c3\u30b7\u30e7\u30f3\u7ba1\u7406\u3068CSRF\u5bfe\u7b56<\/h3>\n\n\n\n<p>\u30bb\u30c3\u30b7\u30e7\u30f3\u7ba1\u7406\u3068CSRF\u5bfe\u7b56\u3092\u5f37\u5316\u3059\u308b\u305f\u3081\u306e\u5b9f\u88c5\u3092\u884c\u3044\u307e\u3059\u3002<\/p>\n\n\n\n<p><strong>1. \u30bb\u30c3\u30b7\u30e7\u30f3\u8a2d\u5b9a\u306e\u6700\u9069\u5316<\/strong><\/p>\n\n\n\n<pre class=\"EnlighterJSRAW\" data-enlighter-language=\"generic\" data-enlighter-theme=\"\" data-enlighter-highlight=\"\" data-enlighter-linenumbers=\"\" data-enlighter-lineoffset=\"\" data-enlighter-title=\"\" data-enlighter-group=\"\">\/\/ config\/session.php\nreturn [\n    'driver' =&gt; env('SESSION_DRIVER', 'redis'),  \/\/ Redis\u3092\u63a8\u5968\n    'lifetime' =&gt; env('SESSION_LIFETIME', 120),  \/\/ 2\u6642\u9593\n    'expire_on_close' =&gt; true,\n    'secure' =&gt; env('SESSION_SECURE_COOKIE', true),\n    'http_only' =&gt; true,\n    'same_site' =&gt; 'lax',\n];<\/pre>\n\n\n\n<p><strong>2. \u30bb\u30c3\u30b7\u30e7\u30f3\u306e\u30bb\u30ad\u30e5\u30ea\u30c6\u30a3\u5f37\u5316<\/strong><\/p>\n\n\n\n<pre class=\"EnlighterJSRAW\" data-enlighter-language=\"generic\" data-enlighter-theme=\"\" data-enlighter-highlight=\"\" data-enlighter-linenumbers=\"\" data-enlighter-lineoffset=\"\" data-enlighter-title=\"\" data-enlighter-group=\"\">\/\/ app\/Http\/Middleware\/AdminSessionSecurity.php\nnamespace App\\Http\\Middleware;\n\nuse Closure;\nuse Illuminate\\Http\\Request;\nuse Illuminate\\Support\\Str;\n\nclass AdminSessionSecurity\n{\n    public function handle(Request $request, Closure $next)\n    {\n        \/\/ \u30bb\u30c3\u30b7\u30e7\u30f3ID\u306e\u518d\u751f\u6210\n        if ($request-&gt;session()-&gt;has('last_activity') &amp;&amp; \n            time() - $request-&gt;session()-&gt;get('last_activity') &gt; 1800) {\n            $request-&gt;session()-&gt;flush();\n            return redirect()-&gt;route('admin.login');\n        }\n\n        \/\/ \u30e6\u30fc\u30b6\u30fc\u30a8\u30fc\u30b8\u30a7\u30f3\u30c8\u306e\u691c\u8a3c\n        if ($request-&gt;session()-&gt;has('user_agent')) {\n            if ($request-&gt;session()-&gt;get('user_agent') !== $request-&gt;userAgent()) {\n                $request-&gt;session()-&gt;flush();\n                return redirect()-&gt;route('admin.login');\n            }\n        } else {\n            $request-&gt;session()-&gt;put('user_agent', $request-&gt;userAgent());\n        }\n\n        $request-&gt;session()-&gt;put('last_activity', time());\n\n        return $next($request);\n    }\n}<\/pre>\n\n\n\n<p><strong>3. CSRF\u5bfe\u7b56\u306e\u5b9f\u88c5<\/strong><\/p>\n\n\n\n<pre class=\"EnlighterJSRAW\" data-enlighter-language=\"generic\" data-enlighter-theme=\"\" data-enlighter-highlight=\"\" data-enlighter-linenumbers=\"\" data-enlighter-lineoffset=\"\" data-enlighter-title=\"\" data-enlighter-group=\"\">\/\/ app\/Http\/Middleware\/VerifyCsrfToken.php\nnamespace App\\Http\\Middleware;\n\nuse Illuminate\\Foundation\\Http\\Middleware\\VerifyCsrfToken as Middleware;\n\nclass VerifyCsrfToken extends Middleware\n{\n    protected $except = [\n        \/\/ CSRF\u4fdd\u8b77\u304b\u3089\u9664\u5916\u3059\u308bURL\u3092\u6307\u5b9a\n    ];\n\n    protected function tokensMatch($request)\n    {\n        $token = $request-&gt;input('_token') ?: $request-&gt;header('X-CSRF-TOKEN');\n\n        if (!$token &amp;&amp; $header = $request-&gt;header('X-XSRF-TOKEN')) {\n            $token = $this-&gt;encrypter-&gt;decrypt($header, static::serialized());\n        }\n\n        $sessionToken = $request-&gt;session()-&gt;token();\n\n        if (!is_string($sessionToken)) {\n            return false;\n        }\n\n        return hash_equals($sessionToken, $token);\n    }\n}\n\n\/\/ Ajax\u30ea\u30af\u30a8\u30b9\u30c8\u306e\u4f8b\n$.ajaxSetup({\n    headers: {\n        'X-CSRF-TOKEN': $('meta[name=\"csrf-token\"]').attr('content')\n    }\n});<\/pre>\n\n\n\n<p>\u3053\u308c\u3089\u306e\u30bb\u30ad\u30e5\u30ea\u30c6\u30a3\u5bfe\u7b56\u306b\u3088\u308a\u3001\u4ee5\u4e0b\u306e\u8105\u5a01\u304b\u3089\u7ba1\u7406\u753b\u9762\u3092\u4fdd\u8b77\u3067\u304d\u307e\u3059\uff1a<\/p>\n\n\n\n<ol class=\"wp-block-list\">\n<li>\u4e0d\u6b63\u30a2\u30af\u30bb\u30b9\u3068\u6a29\u9650\u306e\u6607\u683c<\/li>\n\n\n\n<li>\u30af\u30ed\u30b9\u30b5\u30a4\u30c8\u30b9\u30af\u30ea\u30d7\u30c6\u30a3\u30f3\u30b0\uff08XSS\uff09\u653b\u6483<\/li>\n\n\n\n<li>\u30af\u30ed\u30b9\u30b5\u30a4\u30c8\u30ea\u30af\u30a8\u30b9\u30c8\u30d5\u30a9\u30fc\u30b8\u30a7\u30ea\uff08CSRF\uff09\u653b\u6483<\/li>\n\n\n\n<li>\u30bb\u30c3\u30b7\u30e7\u30f3\u30cf\u30a4\u30b8\u30e3\u30c3\u30af<\/li>\n\n\n\n<li>\u30af\u30ea\u30c3\u30af\u30b8\u30e3\u30c3\u30ad\u30f3\u30b0<\/li>\n<\/ol>\n\n\n\n<p>\u307e\u305f\u3001\u3053\u308c\u3089\u306e\u5b9f\u88c5\u306f\u4ee5\u4e0b\u306e\u5229\u70b9\u304c\u3042\u308a\u307e\u3059\uff1a<\/p>\n\n\n\n<ul class=\"wp-block-list\">\n<li>\u67d4\u8edf\u306a\u6a29\u9650\u7ba1\u7406\u30b7\u30b9\u30c6\u30e0<\/li>\n\n\n\n<li>\u5805\u7262\u306a\u30bb\u30ad\u30e5\u30ea\u30c6\u30a3\u5bfe\u7b56<\/li>\n\n\n\n<li>\u30d1\u30d5\u30a9\u30fc\u30de\u30f3\u30b9\u3092\u8003\u616e\u3057\u305f\u5b9f\u88c5<\/li>\n\n\n\n<li>\u4fdd\u5b88\u6027\u306e\u9ad8\u3044\u30b3\u30fc\u30c9\u69cb\u9020<\/li>\n<\/ul>\n\n\n\n<h2 class=\"wp-block-heading\" id=\"i-16\">\u7ba1\u7406\u753b\u9762\u306e\u30ab\u30b9\u30bf\u30de\u30a4\u30ba\u3068\u62e1\u5f35<\/h2>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"i-17\">\u30ec\u30a4\u30a2\u30a6\u30c8\u306e\u30ab\u30b9\u30bf\u30de\u30a4\u30ba\u65b9\u6cd5<\/h3>\n\n\n\n<p>\u7ba1\u7406\u753b\u9762\u306e\u30ec\u30a4\u30a2\u30a6\u30c8\u3092\u30ab\u30b9\u30bf\u30de\u30a4\u30ba\u3057\u3001\u4f7f\u3044\u3084\u3059\u3044\u30a4\u30f3\u30bf\u30fc\u30d5\u30a7\u30fc\u30b9\u3092\u5b9f\u73fe\u3057\u307e\u3059\u3002<\/p>\n\n\n\n<p><strong>1. \u57fa\u672c\u30ec\u30a4\u30a2\u30a6\u30c8\u306e\u4f5c\u6210<\/strong><\/p>\n\n\n\n<pre class=\"EnlighterJSRAW\" data-enlighter-language=\"generic\" data-enlighter-theme=\"\" data-enlighter-highlight=\"\" data-enlighter-linenumbers=\"\" data-enlighter-lineoffset=\"\" data-enlighter-title=\"\" data-enlighter-group=\"\">&lt;!-- resources\/views\/layouts\/admin.blade.php --&gt;\n&lt;!DOCTYPE html&gt;\n&lt;html lang=\"ja\"&gt;\n&lt;head&gt;\n    &lt;meta charset=\"UTF-8\"&gt;\n    &lt;meta name=\"viewport\" content=\"width=device-width, initial-scale=1.0\"&gt;\n    &lt;title&gt;{{ config('app.name') }} - \u7ba1\u7406\u753b\u9762&lt;\/title&gt;\n    @vite(['resources\/css\/app.css', 'resources\/js\/app.js'])\n&lt;\/head&gt;\n&lt;body class=\"bg-gray-100\"&gt;\n    &lt;div class=\"min-h-screen flex\"&gt;\n        &lt;!-- \u30b5\u30a4\u30c9\u30d0\u30fc --&gt;\n        &lt;div class=\"bg-gray-800 text-white w-64 space-y-6 py-7 px-2 absolute inset-y-0 left-0 transform -translate-x-full md:relative md:translate-x-0 transition duration-200 ease-in-out\"&gt;\n            @include('admin.partials.sidebar')\n        &lt;\/div&gt;\n\n        &lt;!-- \u30e1\u30a4\u30f3\u30b3\u30f3\u30c6\u30f3\u30c4 --&gt;\n        &lt;div class=\"flex-1\"&gt;\n            &lt;!-- \u30d8\u30c3\u30c0\u30fc --&gt;\n            &lt;header class=\"bg-white shadow-lg\"&gt;\n                @include('admin.partials.header')\n            &lt;\/header&gt;\n\n            &lt;!-- \u30b3\u30f3\u30c6\u30f3\u30c4\u30a8\u30ea\u30a2 --&gt;\n            &lt;main class=\"p-6\"&gt;\n                @yield('content')\n            &lt;\/main&gt;\n        &lt;\/div&gt;\n    &lt;\/div&gt;\n&lt;\/body&gt;\n&lt;\/html&gt;\n\n&lt;!-- resources\/views\/admin\/partials\/sidebar.blade.php --&gt;\n&lt;nav class=\"space-y-3\"&gt;\n    &lt;div class=\"px-4 py-3\"&gt;\n        &lt;h1 class=\"text-xl font-semibold\"&gt;\u7ba1\u7406\u30d1\u30cd\u30eb&lt;\/h1&gt;\n    &lt;\/div&gt;\n\n    &lt;div class=\"space-y-2\"&gt;\n        &lt;a href=\"{{ route('admin.dashboard') }}\" class=\"block px-4 py-2 rounded transition hover:bg-gray-700 {{ request()-&gt;routeIs('admin.dashboard') ? 'bg-gray-700' : '' }}\"&gt;\n            &lt;i class=\"fas fa-tachometer-alt mr-2\"&gt;&lt;\/i&gt;\u30c0\u30c3\u30b7\u30e5\u30dc\u30fc\u30c9\n        &lt;\/a&gt;\n\n        @can('manage_products')\n        &lt;a href=\"{{ route('admin.products.index') }}\" class=\"block px-4 py-2 rounded transition hover:bg-gray-700 {{ request()-&gt;routeIs('admin.products.*') ? 'bg-gray-700' : '' }}\"&gt;\n            &lt;i class=\"fas fa-box mr-2\"&gt;&lt;\/i&gt;\u5546\u54c1\u7ba1\u7406\n        &lt;\/a&gt;\n        @endcan\n\n        @can('manage_users')\n        &lt;a href=\"{{ route('admin.users.index') }}\" class=\"block px-4 py-2 rounded transition hover:bg-gray-700 {{ request()-&gt;routeIs('admin.users.*') ? 'bg-gray-700' : '' }}\"&gt;\n            &lt;i class=\"fas fa-users mr-2\"&gt;&lt;\/i&gt;\u30e6\u30fc\u30b6\u30fc\u7ba1\u7406\n        &lt;\/a&gt;\n        @endcan\n    &lt;\/div&gt;\n&lt;\/nav&gt;<\/pre>\n\n\n\n<p><strong>2. \u30ec\u30b9\u30dd\u30f3\u30b7\u30d6\u30c7\u30b6\u30a4\u30f3\u306e\u5b9f\u88c5<\/strong><\/p>\n\n\n\n<pre class=\"EnlighterJSRAW\" data-enlighter-language=\"generic\" data-enlighter-theme=\"\" data-enlighter-highlight=\"\" data-enlighter-linenumbers=\"\" data-enlighter-lineoffset=\"\" data-enlighter-title=\"\" data-enlighter-group=\"\">\/\/ resources\/js\/admin.js\ndocument.addEventListener('DOMContentLoaded', function() {\n    const sidebarToggle = document.getElementById('sidebar-toggle');\n    const sidebar = document.querySelector('.sidebar');\n\n    sidebarToggle.addEventListener('click', () =&gt; {\n        sidebar.classList.toggle('-translate-x-full');\n    });\n\n    \/\/ \u30a6\u30a3\u30f3\u30c9\u30a6\u30b5\u30a4\u30ba\u5909\u66f4\u6642\u306e\u51e6\u7406\n    window.addEventListener('resize', () =&gt; {\n        if (window.innerWidth &gt;= 768) {\n            sidebar.classList.remove('-translate-x-full');\n        }\n    });\n});<\/pre>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"i-18\">\u30c0\u30c3\u30b7\u30e5\u30dc\u30fc\u30c9\u30a6\u30a3\u30b8\u30a7\u30c3\u30c8\u306e\u8ffd\u52a0<\/h3>\n\n\n\n<p>\u7ba1\u7406\u753b\u9762\u306e\u30c0\u30c3\u30b7\u30e5\u30dc\u30fc\u30c9\u306b\u3001\u69d8\u3005\u306a\u60c5\u5831\u3092\u8868\u793a\u3059\u308b\u30a6\u30a3\u30b8\u30a7\u30c3\u30c8\u3092\u5b9f\u88c5\u3057\u307e\u3059\u3002<\/p>\n\n\n\n<p><strong>1. \u30a6\u30a3\u30b8\u30a7\u30c3\u30c8\u30b3\u30f3\u30dd\u30fc\u30cd\u30f3\u30c8\u306e\u4f5c\u6210<\/strong><\/p>\n\n\n\n<pre class=\"EnlighterJSRAW\" data-enlighter-language=\"generic\" data-enlighter-theme=\"\" data-enlighter-highlight=\"\" data-enlighter-linenumbers=\"\" data-enlighter-lineoffset=\"\" data-enlighter-title=\"\" data-enlighter-group=\"\">\/\/ app\/View\/Components\/Admin\/Widget.php\nnamespace App\\View\\Components\\Admin;\n\nuse Illuminate\\View\\Component;\n\nclass Widget extends Component\n{\n    public $title;\n    public $value;\n    public $icon;\n    public $color;\n\n    public function __construct($title, $value, $icon = null, $color = 'blue')\n    {\n        $this-&gt;title = $title;\n        $this-&gt;value = $value;\n        $this-&gt;icon = $icon;\n        $this-&gt;color = $color;\n    }\n\n    public function render()\n    {\n        return view('components.admin.widget');\n    }\n}\n\n&lt;!-- resources\/views\/components\/admin\/widget.blade.php --&gt;\n&lt;div class=\"bg-white rounded-lg shadow p-6\"&gt;\n    &lt;div class=\"flex items-center\"&gt;\n        @if($icon)\n        &lt;div class=\"rounded-full p-3 bg-{{ $color }}-100 mr-4\"&gt;\n            &lt;i class=\"fas fa-{{ $icon }} text-{{ $color }}-500 text-xl\"&gt;&lt;\/i&gt;\n        &lt;\/div&gt;\n        @endif\n        &lt;div&gt;\n            &lt;h3 class=\"text-gray-500 text-sm font-medium\"&gt;{{ $title }}&lt;\/h3&gt;\n            &lt;p class=\"text-2xl font-semibold text-gray-700\"&gt;{{ $value }}&lt;\/p&gt;\n        &lt;\/div&gt;\n    &lt;\/div&gt;\n    {{ $slot }}\n&lt;\/div&gt;<\/pre>\n\n\n\n<p><strong>2. \u30c0\u30c3\u30b7\u30e5\u30dc\u30fc\u30c9\u306e\u5b9f\u88c5<\/strong><\/p>\n\n\n\n<pre class=\"EnlighterJSRAW\" data-enlighter-language=\"generic\" data-enlighter-theme=\"\" data-enlighter-highlight=\"\" data-enlighter-linenumbers=\"\" data-enlighter-lineoffset=\"\" data-enlighter-title=\"\" data-enlighter-group=\"\">\/\/ app\/Http\/Controllers\/Admin\/DashboardController.php\nnamespace App\\Http\\Controllers\\Admin;\n\nuse App\\Http\\Controllers\\Controller;\nuse App\\Models\\Order;\nuse App\\Models\\Product;\nuse App\\Models\\User;\nuse Illuminate\\Support\\Facades\\DB;\n\nclass DashboardController extends Controller\n{\n    public function index()\n    {\n        $stats = [\n            'total_sales' =&gt; Order::whereMonth('created_at', now()-&gt;month)\n                                -&gt;sum('total_amount'),\n            'total_products' =&gt; Product::count(),\n            'low_stock_products' =&gt; Product::where('stock', '&lt;=', 5)-&gt;count(),\n            'total_users' =&gt; User::count()\n        ];\n\n        $recentOrders = Order::with('user')\n                            -&gt;latest()\n                            -&gt;take(5)\n                            -&gt;get();\n\n        $monthlySales = Order::select(\n            DB::raw('DATE_FORMAT(created_at, \"%Y-%m\") as month'),\n            DB::raw('SUM(total_amount) as total')\n        )\n        -&gt;groupBy('month')\n        -&gt;orderBy('month', 'DESC')\n        -&gt;take(6)\n        -&gt;get();\n\n        return view('admin.dashboard', compact('stats', 'recentOrders', 'monthlySales'));\n    }\n}<\/pre>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"i-19\">\u30d5\u30a1\u30a4\u30eb\u30a2\u30c3\u30d7\u30ed\u30fc\u30c9\u6a5f\u80fd\u306e\u5b9f\u88c5<\/h3>\n\n\n\n<p>\u753b\u50cf\u3084\u30d5\u30a1\u30a4\u30eb\u306e\u30a2\u30c3\u30d7\u30ed\u30fc\u30c9\u6a5f\u80fd\u3092\u5b9f\u88c5\u3057\u3001\u7ba1\u7406\u753b\u9762\u304b\u3089\u52b9\u7387\u7684\u306b\u30d5\u30a1\u30a4\u30eb\u7ba1\u7406\u304c\u3067\u304d\u308b\u3088\u3046\u306b\u3057\u307e\u3059\u3002<\/p>\n\n\n\n<p><strong>1. \u30d5\u30a1\u30a4\u30eb\u30a2\u30c3\u30d7\u30ed\u30fc\u30c9\u30b5\u30fc\u30d3\u30b9\u306e\u4f5c\u6210<\/strong><\/p>\n\n\n\n<pre class=\"EnlighterJSRAW\" data-enlighter-language=\"generic\" data-enlighter-theme=\"\" data-enlighter-highlight=\"\" data-enlighter-linenumbers=\"\" data-enlighter-lineoffset=\"\" data-enlighter-title=\"\" data-enlighter-group=\"\">\/\/ app\/Services\/FileUploadService.php\nnamespace App\\Services;\n\nuse Illuminate\\Http\\UploadedFile;\nuse Illuminate\\Support\\Facades\\Storage;\nuse Intervention\\Image\\Facades\\Image;\n\nclass FileUploadService\n{\n    public function uploadImage(UploadedFile $file, string $path, array $sizes = []): string\n    {\n        $filename = uniqid() . '.' . $file-&gt;getClientOriginalExtension();\n        $fullPath = $path . '\/' . $filename;\n\n        \/\/ \u30aa\u30ea\u30b8\u30ca\u30eb\u753b\u50cf\u306e\u4fdd\u5b58\n        Storage::disk('public')-&gt;put($fullPath, $file-&gt;get());\n\n        \/\/ \u30ea\u30b5\u30a4\u30ba\u30d0\u30fc\u30b8\u30e7\u30f3\u306e\u4f5c\u6210\n        if (!empty($sizes)) {\n            foreach ($sizes as $size) {\n                $resizedImage = Image::make($file)\n                    -&gt;fit($size['width'], $size['height'])\n                    -&gt;encode();\n\n                Storage::disk('public')-&gt;put(\n                    $path . '\/' . $size['width'] . 'x' . $size['height'] . '_' . $filename,\n                    $resizedImage\n                );\n            }\n        }\n\n        return $fullPath;\n    }\n\n    public function deleteImage(string $path): bool\n    {\n        if (Storage::disk('public')-&gt;exists($path)) {\n            \/\/ \u30aa\u30ea\u30b8\u30ca\u30eb\u753b\u50cf\u306e\u524a\u9664\n            Storage::disk('public')-&gt;delete($path);\n\n            \/\/ \u30ea\u30b5\u30a4\u30ba\u30d0\u30fc\u30b8\u30e7\u30f3\u306e\u524a\u9664\n            $pathInfo = pathinfo($path);\n            $pattern = $pathInfo['dirname'] . '\/*_' . $pathInfo['basename'];\n\n            foreach (Storage::disk('public')-&gt;files($pathInfo['dirname']) as $file) {\n                if (fnmatch($pattern, $file)) {\n                    Storage::disk('public')-&gt;delete($file);\n                }\n            }\n\n            return true;\n        }\n\n        return false;\n    }\n}<\/pre>\n\n\n\n<p><strong>2. \u30d5\u30a1\u30a4\u30eb\u30a2\u30c3\u30d7\u30ed\u30fc\u30c9\u30b3\u30f3\u30dd\u30fc\u30cd\u30f3\u30c8\u306e\u5b9f\u88c5<\/strong><\/p>\n\n\n\n<pre class=\"EnlighterJSRAW\" data-enlighter-language=\"generic\" data-enlighter-theme=\"\" data-enlighter-highlight=\"\" data-enlighter-linenumbers=\"\" data-enlighter-lineoffset=\"\" data-enlighter-title=\"\" data-enlighter-group=\"\">&lt;!-- resources\/views\/components\/admin\/file-upload.blade.php --&gt;\n&lt;div\n    x-data=\"{ \n        isUploading: false,\n        progress: 0,\n        handleFileSelect(event) {\n            const file = event.target.files[0];\n            const formData = new FormData();\n            formData.append('file', file);\n\n            this.isUploading = true;\n            this.progress = 0;\n\n            axios.post('{{ route('admin.upload.file') }}', formData, {\n                headers: {\n                    'Content-Type': 'multipart\/form-data'\n                },\n                onUploadProgress: (progressEvent) =&gt; {\n                    this.progress = Math.round(\n                        (progressEvent.loaded * 100) \/ progressEvent.total\n                    );\n                }\n            })\n            .then(response =&gt; {\n                this.$dispatch('file-uploaded', response.data);\n            })\n            .catch(error =&gt; {\n                console.error('Upload failed:', error);\n                alert('\u30a2\u30c3\u30d7\u30ed\u30fc\u30c9\u306b\u5931\u6557\u3057\u307e\u3057\u305f\u3002');\n            })\n            .finally(() =&gt; {\n                this.isUploading = false;\n            });\n        }\n    }\"\n&gt;\n    &lt;div class=\"relative border-2 border-dashed border-gray-300 rounded-lg p-6\"&gt;\n        &lt;input\n            type=\"file\"\n            class=\"absolute inset-0 w-full h-full opacity-0 cursor-pointer\"\n            @change=\"handleFileSelect\"\n            {{ $attributes }}\n        &gt;\n        &lt;div class=\"text-center\"&gt;\n            &lt;i class=\"fas fa-cloud-upload-alt text-3xl text-gray-400\"&gt;&lt;\/i&gt;\n            &lt;p class=\"mt-2 text-sm text-gray-600\"&gt;\n                \u30d5\u30a1\u30a4\u30eb\u3092\u30c9\u30e9\u30c3\u30b0\uff06\u30c9\u30ed\u30c3\u30d7\u3001\u307e\u305f\u306f&lt;br&gt;\n                \u30af\u30ea\u30c3\u30af\u3057\u3066\u30d5\u30a1\u30a4\u30eb\u3092\u9078\u629e\n            &lt;\/p&gt;\n        &lt;\/div&gt;\n    &lt;\/div&gt;\n\n    &lt;!-- \u30a2\u30c3\u30d7\u30ed\u30fc\u30c9\u9032\u6357\u30d0\u30fc --&gt;\n    &lt;div x-show=\"isUploading\" class=\"mt-4\"&gt;\n        &lt;div class=\"bg-gray-200 rounded-full h-2.5\"&gt;\n            &lt;div\n                class=\"bg-blue-600 h-2.5 rounded-full transition-all duration-300\"\n                :style=\"{ width: `${progress}%` }\"\n            &gt;&lt;\/div&gt;\n        &lt;\/div&gt;\n        &lt;p class=\"text-sm text-gray-600 mt-2\"&gt;\n            \u30a2\u30c3\u30d7\u30ed\u30fc\u30c9\u4e2d... &lt;span x-text=\"progress\"&gt;&lt;\/span&gt;%\n        &lt;\/p&gt;\n    &lt;\/div&gt;\n&lt;\/div&gt;<\/pre>\n\n\n\n<p>\u3053\u308c\u3089\u306e\u5b9f\u88c5\u306b\u3088\u308a\u3001\u4ee5\u4e0b\u306e\u6a5f\u80fd\u304c\u5b9f\u73fe\u3067\u304d\u307e\u3059\uff1a<\/p>\n\n\n\n<ol class=\"wp-block-list\">\n<li>\u30ec\u30b9\u30dd\u30f3\u30b7\u30d6\u5bfe\u5fdc\u306e\u7ba1\u7406\u753b\u9762\u30ec\u30a4\u30a2\u30a6\u30c8<\/li>\n\n\n\n<li>\u30ab\u30b9\u30bf\u30de\u30a4\u30ba\u53ef\u80fd\u306a\u30c0\u30c3\u30b7\u30e5\u30dc\u30fc\u30c9\u30a6\u30a3\u30b8\u30a7\u30c3\u30c8<\/li>\n\n\n\n<li>\u52b9\u7387\u7684\u306a\u30d5\u30a1\u30a4\u30eb\u7ba1\u7406\u30b7\u30b9\u30c6\u30e0<\/li>\n\n\n\n<li>\u6700\u9069\u5316\u3055\u308c\u305f\u753b\u50cf\u30a2\u30c3\u30d7\u30ed\u30fc\u30c9\u6a5f\u80fd<\/li>\n<\/ol>\n\n\n\n<p>\u307e\u305f\u3001\u4ee5\u4e0b\u306e\u5229\u70b9\u304c\u3042\u308a\u307e\u3059\uff1a<\/p>\n\n\n\n<ul class=\"wp-block-list\">\n<li>\u76f4\u611f\u7684\u306a\u30e6\u30fc\u30b6\u30fc\u30a4\u30f3\u30bf\u30fc\u30d5\u30a7\u30fc\u30b9<\/li>\n\n\n\n<li>\u52b9\u7387\u7684\u306a\u30c7\u30fc\u30bf\u7ba1\u7406<\/li>\n\n\n\n<li>\u62e1\u5f35\u6027\u306e\u9ad8\u3044\u8a2d\u8a08<\/li>\n\n\n\n<li>\u30d1\u30d5\u30a9\u30fc\u30de\u30f3\u30b9\u3092\u8003\u616e\u3057\u305f\u5b9f\u88c5<\/li>\n<\/ul>\n\n\n\n<h2 class=\"wp-block-heading\" id=\"i-20\">\u30d1\u30d5\u30a9\u30fc\u30de\u30f3\u30b9\u6700\u9069\u5316\u3068\u30e1\u30f3\u30c6\u30ca\u30f3\u30b9<\/h2>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"i-21\">\u30ad\u30e3\u30c3\u30b7\u30e5\u8a2d\u5b9a\u3068\u30c7\u30fc\u30bf\u30d9\u30fc\u30b9\u6700\u9069\u5316<\/h3>\n\n\n\n<p>\u7ba1\u7406\u753b\u9762\u306e\u30d1\u30d5\u30a9\u30fc\u30de\u30f3\u30b9\u3092\u6700\u9069\u5316\u3057\u3001\u30ec\u30b9\u30dd\u30f3\u30b9\u6642\u9593\u3092\u77ed\u7e2e\u3059\u308b\u65b9\u6cd5\u3092\u89e3\u8aac\u3057\u307e\u3059\u3002<\/p>\n\n\n\n<p><strong>1. \u30ad\u30e3\u30c3\u30b7\u30e5\u306e\u8a2d\u5b9a\u3068\u5229\u7528<\/strong><\/p>\n\n\n\n<pre class=\"EnlighterJSRAW\" data-enlighter-language=\"generic\" data-enlighter-theme=\"\" data-enlighter-highlight=\"\" data-enlighter-linenumbers=\"\" data-enlighter-lineoffset=\"\" data-enlighter-title=\"\" data-enlighter-group=\"\">\/\/ config\/cache.php\nreturn [\n    'default' =&gt; env('CACHE_DRIVER', 'redis'),\n    'stores' =&gt; [\n        'redis' =&gt; [\n            'driver' =&gt; 'redis',\n            'connection' =&gt; 'cache',\n            'lock_connection' =&gt; 'default',\n        ],\n    ],\n    'prefix' =&gt; env('CACHE_PREFIX', 'laravel_cache'),\n];\n\n\/\/ app\/Services\/CacheService.php\nnamespace App\\Services;\n\nuse Illuminate\\Support\\Facades\\Cache;\n\nclass CacheService\n{\n    public static function remember(string $key, $data, int $minutes = 60)\n    {\n        return Cache::remember($key, $minutes, function () use ($data) {\n            return is_callable($data) ? $data() : $data;\n        });\n    }\n\n    public static function clearModelCache(string $modelName)\n    {\n        Cache::tags($modelName)-&gt;flush();\n    }\n}\n\n\/\/ \u4f7f\u7528\u4f8b\nclass ProductController extends Controller\n{\n    public function index()\n    {\n        $products = CacheService::remember('products.all', function() {\n            return Product::with('category')\n                         -&gt;latest()\n                         -&gt;paginate(20);\n        }, 30);  \/\/ 30\u5206\u30ad\u30e3\u30c3\u30b7\u30e5\n\n        return view('admin.products.index', compact('products'));\n    }\n}<\/pre>\n\n\n\n<p><strong>2. \u30c7\u30fc\u30bf\u30d9\u30fc\u30b9\u6700\u9069\u5316<\/strong><\/p>\n\n\n\n<pre class=\"EnlighterJSRAW\" data-enlighter-language=\"generic\" data-enlighter-theme=\"\" data-enlighter-highlight=\"\" data-enlighter-linenumbers=\"\" data-enlighter-lineoffset=\"\" data-enlighter-title=\"\" data-enlighter-group=\"\">\/\/ app\/Console\/Commands\/OptimizeDatabase.php\nnamespace App\\Console\\Commands;\n\nuse Illuminate\\Console\\Command;\nuse Illuminate\\Support\\Facades\\DB;\n\nclass OptimizeDatabase extends Command\n{\n    protected $signature = 'db:optimize';\n    protected $description = 'Optimize database tables';\n\n    public function handle()\n    {\n        $tables = DB::select('SHOW TABLES');\n\n        foreach ($tables as $table) {\n            $tableName = array_values((array) $table)[0];\n\n            $this-&gt;info(\"Optimizing table: {$tableName}\");\n\n            DB::statement(\"ANALYZE TABLE {$tableName}\");\n            DB::statement(\"OPTIMIZE TABLE {$tableName}\");\n        }\n\n        $this-&gt;info('Database optimization completed!');\n    }\n}\n\n\/\/ \u30c7\u30fc\u30bf\u30d9\u30fc\u30b9\u30a4\u30f3\u30c7\u30c3\u30af\u30b9\u306e\u6700\u9069\u5316\nclass CreateProductsIndexes extends Migration\n{\n    public function up()\n    {\n        Schema::table('products', function (Blueprint $table) {\n            \/\/ \u8907\u5408\u30a4\u30f3\u30c7\u30c3\u30af\u30b9\u306e\u4f5c\u6210\n            $table-&gt;index(['category_id', 'is_active']);\n            \/\/ \u5168\u6587\u691c\u7d22\u30a4\u30f3\u30c7\u30c3\u30af\u30b9\u306e\u4f5c\u6210\n            $table-&gt;fullText(['name', 'description']);\n        });\n    }\n}<\/pre>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"i-22\">\u5b9a\u671f\u7684\u306a\u30e1\u30f3\u30c6\u30ca\u30f3\u30b9\u4f5c\u696d\u306e\u81ea\u52d5\u5316<\/h3>\n\n\n\n<p>\u30b7\u30b9\u30c6\u30e0\u306e\u5b9a\u671f\u7684\u306a\u30e1\u30f3\u30c6\u30ca\u30f3\u30b9\u4f5c\u696d\u3092\u81ea\u52d5\u5316\u3057\u3001\u5b89\u5b9a\u3057\u305f\u904b\u7528\u3092\u5b9f\u73fe\u3057\u307e\u3059\u3002<\/p>\n\n\n\n<p><strong>1. \u30e1\u30f3\u30c6\u30ca\u30f3\u30b9\u30bf\u30b9\u30af\u306e\u30b9\u30b1\u30b8\u30e5\u30fc\u30ea\u30f3\u30b0<\/strong><\/p>\n\n\n\n<pre class=\"EnlighterJSRAW\" data-enlighter-language=\"generic\" data-enlighter-theme=\"\" data-enlighter-highlight=\"\" data-enlighter-linenumbers=\"\" data-enlighter-lineoffset=\"\" data-enlighter-title=\"\" data-enlighter-group=\"\">\/\/ app\/Console\/Kernel.php\nprotected function schedule(Schedule $schedule)\n{\n    \/\/ \u30c7\u30fc\u30bf\u30d9\u30fc\u30b9\u6700\u9069\u5316\uff08\u6bce\u9031\u65e5\u66dc\u65e5\u306e\u6df1\u591c3\u6642\uff09\n    $schedule-&gt;command('db:optimize')\n            -&gt;weekly()\n            -&gt;sundays()\n            -&gt;at('03:00')\n            -&gt;emailOutputTo('admin@example.com');\n\n    \/\/ \u30ad\u30e3\u30c3\u30b7\u30e5\u306e\u30af\u30ea\u30a2\uff08\u6bce\u65e5\u6df1\u591c1\u6642\uff09\n    $schedule-&gt;command('cache:clear')\n            -&gt;daily()\n            -&gt;at('01:00');\n\n    \/\/ \u4e00\u6642\u30d5\u30a1\u30a4\u30eb\u306e\u524a\u9664\uff08\u6bce\u65e5\u6df1\u591c2\u6642\uff09\n    $schedule-&gt;command('temp:clean')\n            -&gt;daily()\n            -&gt;at('02:00');\n\n    \/\/ \u30d0\u30c3\u30af\u30a2\u30c3\u30d7\u306e\u4f5c\u6210\uff08\u6bce\u65e5\u6df1\u591c4\u6642\uff09\n    $schedule-&gt;command('backup:run')\n            -&gt;daily()\n            -&gt;at('04:00');\n}<\/pre>\n\n\n\n<p><strong>2. \u81ea\u52d5\u30d0\u30c3\u30af\u30a2\u30c3\u30d7\u306e\u8a2d\u5b9a<\/strong><\/p>\n\n\n\n<pre class=\"EnlighterJSRAW\" data-enlighter-language=\"generic\" data-enlighter-theme=\"\" data-enlighter-highlight=\"\" data-enlighter-linenumbers=\"\" data-enlighter-lineoffset=\"\" data-enlighter-title=\"\" data-enlighter-group=\"\">\/\/ app\/Console\/Commands\/BackupDatabase.php\nnamespace App\\Console\\Commands;\n\nuse Illuminate\\Console\\Command;\nuse Illuminate\\Support\\Facades\\Storage;\nuse Carbon\\Carbon;\n\nclass BackupDatabase extends Command\n{\n    protected $signature = 'backup:run';\n\n    public function handle()\n    {\n        $filename = 'backup-' . Carbon::now()-&gt;format('Y-m-d-H-i-s') . '.sql';\n\n        \/\/ MySQL\u30c7\u30fc\u30bf\u30d9\u30fc\u30b9\u306e\u30c0\u30f3\u30d7\u3092\u4f5c\u6210\n        $command = sprintf(\n            'mysqldump -u%s -p%s %s &gt; %s',\n            config('database.connections.mysql.username'),\n            config('database.connections.mysql.password'),\n            config('database.connections.mysql.database'),\n            storage_path('app\/backups\/' . $filename)\n        );\n\n        exec($command);\n\n        \/\/ \u30d0\u30c3\u30af\u30a2\u30c3\u30d7\u30d5\u30a1\u30a4\u30eb\u3092S3\u306b\u4fdd\u5b58\n        Storage::disk('s3')-&gt;put(\n            'backups\/' . $filename,\n            Storage::disk('local')-&gt;get('backups\/' . $filename)\n        );\n\n        \/\/ \u53e4\u3044\u30d0\u30c3\u30af\u30a2\u30c3\u30d7\u306e\u524a\u9664\uff0830\u65e5\u4ee5\u4e0a\u524d\u306e\u3082\u306e\uff09\n        $oldBackups = Storage::disk('s3')\n            -&gt;files('backups')\n            -&gt;filter(function($file) {\n                $timestamp = Carbon::createFromFormat(\n                    'Y-m-d-H-i-s',\n                    substr($file, 7, 19)\n                );\n                return $timestamp-&gt;diffInDays(now()) &gt; 30;\n            });\n\n        Storage::disk('s3')-&gt;delete($oldBackups);\n    }\n}<\/pre>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"i-23\">\u30a8\u30e9\u30fc\u30ed\u30b0\u306e\u76e3\u8996\u3068\u5bfe\u5fdc\u65b9\u6cd5<\/h3>\n\n\n\n<p>\u30b7\u30b9\u30c6\u30e0\u306e\u30a8\u30e9\u30fc\u3092\u52b9\u7387\u7684\u306b\u76e3\u8996\u3057\u3001\u8fc5\u901f\u306b\u5bfe\u5fdc\u3059\u308b\u65b9\u6cd5\u3092\u5b9f\u88c5\u3057\u307e\u3059\u3002<\/p>\n\n\n\n<p><strong>1. \u30a8\u30e9\u30fc\u30ed\u30b0\u306e\u8a2d\u5b9a<\/strong><\/p>\n\n\n\n<pre class=\"EnlighterJSRAW\" data-enlighter-language=\"generic\" data-enlighter-theme=\"\" data-enlighter-highlight=\"\" data-enlighter-linenumbers=\"\" data-enlighter-lineoffset=\"\" data-enlighter-title=\"\" data-enlighter-group=\"\">\/\/ config\/logging.php\n'channels' =&gt; [\n    'stack' =&gt; [\n        'driver' =&gt; 'stack',\n        'channels' =&gt; ['daily', 'slack'],\n        'ignore_exceptions' =&gt; false,\n    ],\n    'daily' =&gt; [\n        'driver' =&gt; 'daily',\n        'path' =&gt; storage_path('logs\/laravel.log'),\n        'level' =&gt; env('LOG_LEVEL', 'debug'),\n        'days' =&gt; 14,\n    ],\n    'slack' =&gt; [\n        'driver' =&gt; 'slack',\n        'url' =&gt; env('LOG_SLACK_WEBHOOK_URL'),\n        'username' =&gt; 'Laravel Log',\n        'emoji' =&gt; ':boom:',\n        'level' =&gt; env('LOG_LEVEL', 'critical'),\n    ],\n];<\/pre>\n\n\n\n<p><strong>2. \u30a8\u30e9\u30fc\u76e3\u8996\u30b5\u30fc\u30d3\u30b9\u306e\u5b9f\u88c5<\/strong><\/p>\n\n\n\n<pre class=\"EnlighterJSRAW\" data-enlighter-language=\"generic\" data-enlighter-theme=\"\" data-enlighter-highlight=\"\" data-enlighter-linenumbers=\"\" data-enlighter-lineoffset=\"\" data-enlighter-title=\"\" data-enlighter-group=\"\">\/\/ app\/Services\/ErrorMonitoringService.php\nnamespace App\\Services;\n\nuse Illuminate\\Support\\Facades\\Log;\nuse Exception;\n\nclass ErrorMonitoringService\n{\n    public static function logError(Exception $exception, array $context = [])\n    {\n        $data = [\n            'message' =&gt; $exception-&gt;getMessage(),\n            'file' =&gt; $exception-&gt;getFile(),\n            'line' =&gt; $exception-&gt;getLine(),\n            'trace' =&gt; $exception-&gt;getTraceAsString(),\n            'url' =&gt; request()-&gt;fullUrl(),\n            'method' =&gt; request()-&gt;method(),\n            'ip' =&gt; request()-&gt;ip(),\n            'user_agent' =&gt; request()-&gt;userAgent(),\n        ];\n\n        if (auth()-&gt;check()) {\n            $data['user_id'] = auth()-&gt;id();\n        }\n\n        $data = array_merge($data, $context);\n\n        Log::error('Application Error', $data);\n\n        if (app()-&gt;environment('production')) {\n            \/\/ Slack\u306b\u901a\u77e5\n            self::notifySlack($data);\n        }\n    }\n\n    private static function notifySlack(array $data)\n    {\n        $fields = collect($data)-&gt;map(function ($value, $key) {\n            return [\n                'title' =&gt; ucfirst(str_replace('_', ' ', $key)),\n                'value' =&gt; is_array($value) ? json_encode($value) : $value,\n                'short' =&gt; true\n            ];\n        })-&gt;values()-&gt;all();\n\n        $client = new \\GuzzleHttp\\Client();\n        $client-&gt;post(config('logging.channels.slack.url'), [\n            'json' =&gt; [\n                'attachments' =&gt; [\n                    [\n                        'color' =&gt; 'danger',\n                        'title' =&gt; 'Application Error Detected',\n                        'fields' =&gt; $fields,\n                        'footer' =&gt; config('app.name'),\n                        'ts' =&gt; time()\n                    ]\n                ]\n            ]\n        ]);\n    }\n}<\/pre>\n\n\n\n<p><strong>3. \u30a8\u30e9\u30fc\u30cf\u30f3\u30c9\u30ea\u30f3\u30b0\u306e\u5b9f\u88c5<\/strong><\/p>\n\n\n\n<pre class=\"EnlighterJSRAW\" data-enlighter-language=\"generic\" data-enlighter-theme=\"\" data-enlighter-highlight=\"\" data-enlighter-linenumbers=\"\" data-enlighter-lineoffset=\"\" data-enlighter-title=\"\" data-enlighter-group=\"\">\/\/ app\/Exceptions\/Handler.php\nnamespace App\\Exceptions;\n\nuse Illuminate\\Foundation\\Exceptions\\Handler as ExceptionHandler;\nuse App\\Services\\ErrorMonitoringService;\nuse Throwable;\n\nclass Handler extends ExceptionHandler\n{\n    public function register()\n    {\n        $this-&gt;reportable(function (Throwable $e) {\n            if ($this-&gt;shouldReport($e)) {\n                ErrorMonitoringService::logError($e);\n            }\n        });\n    }\n\n    protected function shouldReport(Throwable $e)\n    {\n        return parent::shouldReport($e) &amp;&amp; \n               !$this-&gt;isHttpException($e);\n    }\n}<\/pre>\n\n\n\n<p>\u3053\u308c\u3089\u306e\u5b9f\u88c5\u306b\u3088\u308a\u3001\u4ee5\u4e0b\u306e\u5229\u70b9\u304c\u5f97\u3089\u308c\u307e\u3059\uff1a<\/p>\n\n\n\n<ol class=\"wp-block-list\">\n<li>\u30b7\u30b9\u30c6\u30e0\u30d1\u30d5\u30a9\u30fc\u30de\u30f3\u30b9\u306e\u5411\u4e0a<\/li>\n<\/ol>\n\n\n\n<ul class=\"wp-block-list\">\n<li>\u52b9\u7387\u7684\u306a\u30ad\u30e3\u30c3\u30b7\u30e5\u7ba1\u7406<\/li>\n\n\n\n<li>\u6700\u9069\u5316\u3055\u308c\u305f\u30c7\u30fc\u30bf\u30d9\u30fc\u30b9\u30a2\u30af\u30bb\u30b9<\/li>\n\n\n\n<li>\u30af\u30a8\u30ea\u306e\u5b9f\u884c\u901f\u5ea6\u5411\u4e0a<\/li>\n<\/ul>\n\n\n\n<ol class=\"wp-block-list\">\n<li>\u5b89\u5b9a\u3057\u305f\u904b\u7528\u4f53\u5236\u306e\u78ba\u7acb<\/li>\n<\/ol>\n\n\n\n<ul class=\"wp-block-list\">\n<li>\u81ea\u52d5\u5316\u3055\u308c\u305f\u30e1\u30f3\u30c6\u30ca\u30f3\u30b9\u4f5c\u696d<\/li>\n\n\n\n<li>\u5b9a\u671f\u7684\u306a\u30d0\u30c3\u30af\u30a2\u30c3\u30d7<\/li>\n\n\n\n<li>\u30a8\u30e9\u30fc\u306e\u65e9\u671f\u767a\u898b\u3068\u5bfe\u5fdc<\/li>\n<\/ul>\n\n\n\n<ol class=\"wp-block-list\">\n<li>\u904b\u7528\u30b3\u30b9\u30c8\u306e\u524a\u6e1b<\/li>\n<\/ol>\n\n\n\n<ul class=\"wp-block-list\">\n<li>\u624b\u52d5\u4f5c\u696d\u306e\u81ea\u52d5\u5316<\/li>\n\n\n\n<li>\u30a8\u30e9\u30fc\u5bfe\u5fdc\u6642\u9593\u306e\u77ed\u7e2e<\/li>\n\n\n\n<li>\u30b7\u30b9\u30c6\u30e0\u76e3\u8996\u306e\u52b9\u7387\u5316<\/li>\n<\/ul>\n\n\n\n<h2 class=\"wp-block-heading\" id=\"i-24\">\u5b9f\u88c5\u4f8b\u3068\u5fdc\u7528\u4e8b\u4f8b<\/h2>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"i-25\">EC\u30b5\u30a4\u30c8\u7ba1\u7406\u753b\u9762\u306e\u5b9f\u88c5\u4f8b<\/h3>\n\n\n\n<p>EC\u30b5\u30a4\u30c8\u306e\u7ba1\u7406\u753b\u9762\u306b\u304a\u3051\u308b\u5177\u4f53\u7684\u306a\u5b9f\u88c5\u4f8b\u3092\u89e3\u8aac\u3057\u307e\u3059\u3002<\/p>\n\n\n\n<p><strong>1. \u6ce8\u6587\u7ba1\u7406\u30b7\u30b9\u30c6\u30e0\u306e\u5b9f\u88c5<\/strong><\/p>\n\n\n\n<pre class=\"EnlighterJSRAW\" data-enlighter-language=\"generic\" data-enlighter-theme=\"\" data-enlighter-highlight=\"\" data-enlighter-linenumbers=\"\" data-enlighter-lineoffset=\"\" data-enlighter-title=\"\" data-enlighter-group=\"\">\/\/ app\/Models\/Order.php\nnamespace App\\Models;\n\nuse Illuminate\\Database\\Eloquent\\Model;\n\nclass Order extends Model\n{\n    protected $fillable = [\n        'order_number',\n        'customer_id',\n        'status',\n        'total_amount',\n        'payment_status',\n        'shipping_status'\n    ];\n\n    protected $casts = [\n        'total_amount' =&gt; 'decimal:2',\n        'ordered_at' =&gt; 'datetime'\n    ];\n\n    \/\/ \u30b9\u30c6\u30fc\u30bf\u30b9\u306e\u5b9a\u6570\u5b9a\u7fa9\n    const STATUS_PENDING = 'pending';\n    const STATUS_CONFIRMED = 'confirmed';\n    const STATUS_SHIPPED = 'shipped';\n    const STATUS_DELIVERED = 'delivered';\n    const STATUS_CANCELLED = 'cancelled';\n\n    public static $statuses = [\n        self::STATUS_PENDING =&gt; '\u51e6\u7406\u5f85\u3061',\n        self::STATUS_CONFIRMED =&gt; '\u78ba\u8a8d\u6e08\u307f',\n        self::STATUS_SHIPPED =&gt; '\u767a\u9001\u6e08\u307f',\n        self::STATUS_DELIVERED =&gt; '\u914d\u9054\u5b8c\u4e86',\n        self::STATUS_CANCELLED =&gt; '\u30ad\u30e3\u30f3\u30bb\u30eb'\n    ];\n}\n\n\/\/ app\/Http\/Controllers\/Admin\/OrderController.php\nnamespace App\\Http\\Controllers\\Admin;\n\nuse App\\Http\\Controllers\\Controller;\nuse App\\Models\\Order;\nuse App\\Services\\OrderService;\nuse Illuminate\\Http\\Request;\n\nclass OrderController extends Controller\n{\n    private $orderService;\n\n    public function __construct(OrderService $orderService)\n    {\n        $this-&gt;orderService = $orderService;\n    }\n\n    public function index(Request $request)\n    {\n        $orders = Order::with(['customer', 'items'])\n            -&gt;when($request-&gt;status, function($query, $status) {\n                return $query-&gt;where('status', $status);\n            })\n            -&gt;when($request-&gt;search, function($query, $search) {\n                return $query-&gt;where('order_number', 'like', \"%{$search}%\")\n                    -&gt;orWhereHas('customer', function($q) use ($search) {\n                        $q-&gt;where('name', 'like', \"%{$search}%\")\n                          -&gt;orWhere('email', 'like', \"%{$search}%\");\n                    });\n            })\n            -&gt;latest()\n            -&gt;paginate(20);\n\n        return view('admin.orders.index', compact('orders'));\n    }\n\n    public function updateStatus(Order $order, Request $request)\n    {\n        $validated = $request-&gt;validate([\n            'status' =&gt; 'required|in:' . implode(',', array_keys(Order::$statuses))\n        ]);\n\n        $this-&gt;orderService-&gt;updateStatus($order, $validated['status']);\n\n        return response()-&gt;json([\n            'message' =&gt; '\u30b9\u30c6\u30fc\u30bf\u30b9\u3092\u66f4\u65b0\u3057\u307e\u3057\u305f\u3002',\n            'order' =&gt; $order-&gt;fresh()\n        ]);\n    }\n}<\/pre>\n\n\n\n<p><strong>2. \u5728\u5eab\u7ba1\u7406\u30b7\u30b9\u30c6\u30e0\u306e\u5b9f\u88c5<\/strong><\/p>\n\n\n\n<pre class=\"EnlighterJSRAW\" data-enlighter-language=\"generic\" data-enlighter-theme=\"\" data-enlighter-highlight=\"\" data-enlighter-linenumbers=\"\" data-enlighter-lineoffset=\"\" data-enlighter-title=\"\" data-enlighter-group=\"\">\/\/ app\/Services\/InventoryService.php\nnamespace App\\Services;\n\nuse App\\Models\\Product;\nuse App\\Models\\InventoryLog;\nuse Illuminate\\Support\\Facades\\DB;\nuse App\\Exceptions\\InsufficientStockException;\n\nclass InventoryService\n{\n    public function adjustStock(Product $product, int $quantity, string $reason)\n    {\n        return DB::transaction(function () use ($product, $quantity, $reason) {\n            \/\/ \u5728\u5eab\u6570\u306e\u66f4\u65b0\n            $newStock = $product-&gt;stock + $quantity;\n\n            if ($newStock &lt; 0) {\n                throw new InsufficientStockException('\u5728\u5eab\u304c\u4e0d\u8db3\u3057\u3066\u3044\u307e\u3059\u3002');\n            }\n\n            $product-&gt;update(['stock' =&gt; $newStock]);\n\n            \/\/ \u5728\u5eab\u5c65\u6b74\u306e\u8a18\u9332\n            InventoryLog::create([\n                'product_id' =&gt; $product-&gt;id,\n                'quantity' =&gt; $quantity,\n                'reason' =&gt; $reason,\n                'before_stock' =&gt; $product-&gt;stock,\n                'after_stock' =&gt; $newStock\n            ]);\n\n            \/\/ \u5728\u5eab\u30a2\u30e9\u30fc\u30c8\u306e\u78ba\u8a8d\n            $this-&gt;checkLowStockAlert($product);\n\n            return $product;\n        });\n    }\n\n    private function checkLowStockAlert(Product $product)\n    {\n        if ($product-&gt;stock &lt;= $product-&gt;stock_alert_threshold) {\n            event(new LowStockAlert($product));\n        }\n    }\n}<\/pre>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"i-26\">\u30e6\u30fc\u30b6\u30fc\u7ba1\u7406\u30b7\u30b9\u30c6\u30e0\u306e\u5b9f\u88c5\u4f8b<\/h3>\n\n\n\n<p>\u5927\u898f\u6a21\u306a\u30e6\u30fc\u30b6\u30fc\u7ba1\u7406\u30b7\u30b9\u30c6\u30e0\u306e\u5b9f\u88c5\u4f8b\u3092\u89e3\u8aac\u3057\u307e\u3059\u3002<\/p>\n\n\n\n<p><strong>1. \u30e6\u30fc\u30b6\u30fc\u7ba1\u7406\u57fa\u672c\u6a5f\u80fd\u306e\u5b9f\u88c5<\/strong><\/p>\n\n\n\n<pre class=\"EnlighterJSRAW\" data-enlighter-language=\"generic\" data-enlighter-theme=\"\" data-enlighter-highlight=\"\" data-enlighter-linenumbers=\"\" data-enlighter-lineoffset=\"\" data-enlighter-title=\"\" data-enlighter-group=\"\">\/\/ app\/Http\/Controllers\/Admin\/UserManagementController.php\nnamespace App\\Http\\Controllers\\Admin;\n\nuse App\\Http\\Controllers\\Controller;\nuse App\\Models\\User;\nuse App\\Services\\UserService;\nuse Illuminate\\Http\\Request;\n\nclass UserManagementController extends Controller\n{\n    private $userService;\n\n    public function __construct(UserService $userService)\n    {\n        $this-&gt;userService = $userService;\n    }\n\n    public function index(Request $request)\n    {\n        $users = User::with(['roles', 'lastLogin'])\n            -&gt;when($request-&gt;role, function($query, $role) {\n                return $query-&gt;whereHas('roles', function($q) use ($role) {\n                    $q-&gt;where('name', $role);\n                });\n            })\n            -&gt;when($request-&gt;status, function($query, $status) {\n                return $query-&gt;where('status', $status);\n            })\n            -&gt;latest()\n            -&gt;paginate(20);\n\n        return view('admin.users.index', compact('users'));\n    }\n\n    public function ban(User $user)\n    {\n        $this-&gt;userService-&gt;banUser($user);\n        return back()-&gt;with('success', '\u30e6\u30fc\u30b6\u30fc\u3092BAN\u3057\u307e\u3057\u305f\u3002');\n    }\n\n    public function export(Request $request)\n    {\n        return $this-&gt;userService-&gt;exportUsers($request-&gt;all());\n    }\n}<\/pre>\n\n\n\n<p><strong>2. \u30e6\u30fc\u30b6\u30fc\u30a2\u30af\u30c6\u30a3\u30d3\u30c6\u30a3\u76e3\u8996\u30b7\u30b9\u30c6\u30e0<\/strong><\/p>\n\n\n\n<pre class=\"EnlighterJSRAW\" data-enlighter-language=\"generic\" data-enlighter-theme=\"\" data-enlighter-highlight=\"\" data-enlighter-linenumbers=\"\" data-enlighter-lineoffset=\"\" data-enlighter-title=\"\" data-enlighter-group=\"\">\/\/ app\/Services\/UserActivityService.php\nnamespace App\\Services;\n\nuse App\\Models\\UserActivity;\nuse Illuminate\\Http\\Request;\n\nclass UserActivityService\n{\n    public function logActivity($user, $action, $details = null)\n    {\n        return UserActivity::create([\n            'user_id' =&gt; $user-&gt;id,\n            'action' =&gt; $action,\n            'ip_address' =&gt; request()-&gt;ip(),\n            'user_agent' =&gt; request()-&gt;userAgent(),\n            'details' =&gt; $details\n        ]);\n    }\n\n    public function getUserActivities($userId, $options = [])\n    {\n        return UserActivity::where('user_id', $userId)\n            -&gt;when(isset($options['action']), function($query) use ($options) {\n                return $query-&gt;where('action', $options['action']);\n            })\n            -&gt;when(isset($options['date_from']), function($query) use ($options) {\n                return $query-&gt;where('created_at', '&gt;=', $options['date_from']);\n            })\n            -&gt;latest()\n            -&gt;paginate($options['per_page'] ?? 20);\n    }\n}<\/pre>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"i-27\">\u30b3\u30f3\u30c6\u30f3\u30c4\u7ba1\u7406\u30b7\u30b9\u30c6\u30e0\u306e\u5b9f\u88c5\u4f8b<\/h3>\n\n\n\n<p>\u30d6\u30ed\u30b0\u3084\u30cb\u30e5\u30fc\u30b9\u8a18\u4e8b\u3092\u7ba1\u7406\u3059\u308bCMS\u306e\u5b9f\u88c5\u4f8b\u3092\u89e3\u8aac\u3057\u307e\u3059\u3002<\/p>\n\n\n\n<p><strong>1. \u8a18\u4e8b\u7ba1\u7406\u30b7\u30b9\u30c6\u30e0\u306e\u5b9f\u88c5<\/strong><\/p>\n\n\n\n<pre class=\"EnlighterJSRAW\" data-enlighter-language=\"generic\" data-enlighter-theme=\"\" data-enlighter-highlight=\"\" data-enlighter-linenumbers=\"\" data-enlighter-lineoffset=\"\" data-enlighter-title=\"\" data-enlighter-group=\"\">\/\/ app\/Models\/Article.php\nnamespace App\\Models;\n\nuse Illuminate\\Database\\Eloquent\\Model;\nuse Illuminate\\Database\\Eloquent\\SoftDeletes;\nuse Spatie\\MediaLibrary\\HasMedia;\nuse Spatie\\MediaLibrary\\InteractsWithMedia;\n\nclass Article extends Model implements HasMedia\n{\n    use SoftDeletes, InteractsWithMedia;\n\n    protected $fillable = [\n        'title',\n        'slug',\n        'content',\n        'excerpt',\n        'author_id',\n        'status',\n        'published_at'\n    ];\n\n    protected $casts = [\n        'published_at' =&gt; 'datetime',\n        'meta' =&gt; 'array'\n    ];\n\n    public function author()\n    {\n        return $this-&gt;belongsTo(User::class, 'author_id');\n    }\n\n    public function categories()\n    {\n        return $this-&gt;belongsToMany(Category::class);\n    }\n\n    public function tags()\n    {\n        return $this-&gt;belongsToMany(Tag::class);\n    }\n}\n\n\/\/ app\/Http\/Controllers\/Admin\/ArticleController.php\nnamespace App\\Http\\Controllers\\Admin;\n\nuse App\\Http\\Controllers\\Controller;\nuse App\\Models\\Article;\nuse App\\Services\\ArticleService;\nuse Illuminate\\Http\\Request;\n\nclass ArticleController extends Controller\n{\n    private $articleService;\n\n    public function __construct(ArticleService $articleService)\n    {\n        $this-&gt;articleService = $articleService;\n    }\n\n    public function store(Request $request)\n    {\n        $validated = $request-&gt;validate([\n            'title' =&gt; 'required|max:255',\n            'content' =&gt; 'required',\n            'category_ids' =&gt; 'array',\n            'tag_ids' =&gt; 'array',\n            'status' =&gt; 'required|in:draft,published,scheduled',\n            'published_at' =&gt; 'required_if:status,scheduled|date',\n            'featured_image' =&gt; 'image|max:2048'\n        ]);\n\n        $article = $this-&gt;articleService-&gt;createArticle($validated);\n\n        return redirect()\n            -&gt;route('admin.articles.edit', $article)\n            -&gt;with('success', '\u8a18\u4e8b\u3092\u4f5c\u6210\u3057\u307e\u3057\u305f\u3002');\n    }\n}<\/pre>\n\n\n\n<p><strong>2. \u30e1\u30c7\u30a3\u30a2\u7ba1\u7406\u30b7\u30b9\u30c6\u30e0\u306e\u5b9f\u88c5<\/strong><\/p>\n\n\n\n<pre class=\"EnlighterJSRAW\" data-enlighter-language=\"generic\" data-enlighter-theme=\"\" data-enlighter-highlight=\"\" data-enlighter-linenumbers=\"\" data-enlighter-lineoffset=\"\" data-enlighter-title=\"\" data-enlighter-group=\"\">\/\/ app\/Services\/MediaLibraryService.php\nnamespace App\\Services;\n\nuse Spatie\\MediaLibrary\\MediaCollections\\Models\\Media;\nuse Illuminate\\Http\\UploadedFile;\nuse Illuminate\\Support\\Str;\n\nclass MediaLibraryService\n{\n    public function uploadMedia($model, UploadedFile $file, $collection = 'default')\n    {\n        $media = $model\n            -&gt;addMedia($file)\n            -&gt;usingName(Str::slug($file-&gt;getClientOriginalName()))\n            -&gt;withResponsiveImages()\n            -&gt;toMediaCollection($collection);\n\n        return $this-&gt;getMediaDetails($media);\n    }\n\n    public function getMediaDetails(Media $media)\n    {\n        return [\n            'id' =&gt; $media-&gt;id,\n            'name' =&gt; $media-&gt;name,\n            'file_name' =&gt; $media-&gt;file_name,\n            'mime_type' =&gt; $media-&gt;mime_type,\n            'size' =&gt; $media-&gt;size,\n            'url' =&gt; $media-&gt;getUrl(),\n            'thumbnail' =&gt; $media-&gt;getUrl('thumbnail'),\n            'responsive_images' =&gt; $media-&gt;getResponsiveImageUrls()\n        ];\n    }\n}<\/pre>\n\n\n\n<p>\u3053\u308c\u3089\u306e\u5b9f\u88c5\u4f8b\u306f\u3001\u4ee5\u4e0b\u306e\u7279\u5fb4\u3092\u6301\u3063\u3066\u3044\u307e\u3059\uff1a<\/p>\n\n\n\n<ol class=\"wp-block-list\">\n<li>EC\u30b5\u30a4\u30c8\u7ba1\u7406\u753b\u9762<\/li>\n<\/ol>\n\n\n\n<ul class=\"wp-block-list\">\n<li>\u6ce8\u6587\u72b6\u614b\u306e\u67d4\u8edf\u306a\u7ba1\u7406<\/li>\n\n\n\n<li>\u30ea\u30a2\u30eb\u30bf\u30a4\u30e0\u306e\u5728\u5eab\u7ba1\u7406<\/li>\n\n\n\n<li>\u5728\u5eab\u30a2\u30e9\u30fc\u30c8\u30b7\u30b9\u30c6\u30e0<\/li>\n<\/ul>\n\n\n\n<ol class=\"wp-block-list\">\n<li>\u30e6\u30fc\u30b6\u30fc\u7ba1\u7406\u30b7\u30b9\u30c6\u30e0<\/li>\n<\/ol>\n\n\n\n<ul class=\"wp-block-list\">\n<li>\u8a73\u7d30\u306a\u30e6\u30fc\u30b6\u30fc\u60c5\u5831\u306e\u7ba1\u7406<\/li>\n\n\n\n<li>\u30a2\u30af\u30c6\u30a3\u30d3\u30c6\u30a3\u30ed\u30b0\u306e\u8a18\u9332<\/li>\n\n\n\n<li>\u30e6\u30fc\u30b6\u30fc\u30c7\u30fc\u30bf\u306e\u30a8\u30af\u30b9\u30dd\u30fc\u30c8\u6a5f\u80fd<\/li>\n<\/ul>\n\n\n\n<ol class=\"wp-block-list\">\n<li>\u30b3\u30f3\u30c6\u30f3\u30c4\u7ba1\u7406\u30b7\u30b9\u30c6\u30e0<\/li>\n<\/ol>\n\n\n\n<ul class=\"wp-block-list\">\n<li>\u67d4\u8edf\u306a\u8a18\u4e8b\u7ba1\u7406<\/li>\n\n\n\n<li>\u30e1\u30c7\u30a3\u30a2\u30e9\u30a4\u30d6\u30e9\u30ea\u306e\u7d71\u5408<\/li>\n\n\n\n<li>\u30ec\u30b9\u30dd\u30f3\u30b7\u30d6\u753b\u50cf\u306e\u81ea\u52d5\u751f\u6210<\/li>\n<\/ul>\n\n\n\n<p>\u3053\u308c\u3089\u306e\u5b9f\u88c5\u4f8b\u3092\u53c2\u8003\u306b\u3001\u30d7\u30ed\u30b8\u30a7\u30af\u30c8\u306e\u8981\u4ef6\u306b\u5fdc\u3058\u3066\u30ab\u30b9\u30bf\u30de\u30a4\u30ba\u3059\u308b\u3053\u3068\u3067\u3001\u52b9\u7387\u7684\u306a\u7ba1\u7406\u753b\u9762\u3092\u69cb\u7bc9\u3067\u304d\u307e\u3059\u3002<\/p>\n","protected":false},"excerpt":{"rendered":"<p>Warning: Undefined array key &#8220;is_admin&#8221; in \/home\/xs392991\/dexall.co.jp\/public_html\/articles\/wp-content\/themes\/ &#8230; <\/p>\n","protected":false},"author":1,"featured_media":0,"comment_status":"closed","ping_status":"open","sticky":false,"template":"","format":"standard","meta":{"footnotes":""},"categories":[33,12],"tags":[],"class_list":{"0":"post-3291","1":"post","2":"type-post","3":"status-publish","4":"format-standard","6":"category-php-laravel","7":"category-php","8":"nothumb"},"_links":{"self":[{"href":"https:\/\/dexall.co.jp\/articles\/index.php?rest_route=\/wp\/v2\/posts\/3291","targetHints":{"allow":["GET"]}}],"collection":[{"href":"https:\/\/dexall.co.jp\/articles\/index.php?rest_route=\/wp\/v2\/posts"}],"about":[{"href":"https:\/\/dexall.co.jp\/articles\/index.php?rest_route=\/wp\/v2\/types\/post"}],"author":[{"embeddable":true,"href":"https:\/\/dexall.co.jp\/articles\/index.php?rest_route=\/wp\/v2\/users\/1"}],"replies":[{"embeddable":true,"href":"https:\/\/dexall.co.jp\/articles\/index.php?rest_route=%2Fwp%2Fv2%2Fcomments&post=3291"}],"version-history":[{"count":2,"href":"https:\/\/dexall.co.jp\/articles\/index.php?rest_route=\/wp\/v2\/posts\/3291\/revisions"}],"predecessor-version":[{"id":3293,"href":"https:\/\/dexall.co.jp\/articles\/index.php?rest_route=\/wp\/v2\/posts\/3291\/revisions\/3293"}],"wp:attachment":[{"href":"https:\/\/dexall.co.jp\/articles\/index.php?rest_route=%2Fwp%2Fv2%2Fmedia&parent=3291"}],"wp:term":[{"taxonomy":"category","embeddable":true,"href":"https:\/\/dexall.co.jp\/articles\/index.php?rest_route=%2Fwp%2Fv2%2Fcategories&post=3291"},{"taxonomy":"post_tag","embeddable":true,"href":"https:\/\/dexall.co.jp\/articles\/index.php?rest_route=%2Fwp%2Fv2%2Ftags&post=3291"}],"curies":[{"name":"wp","href":"https:\/\/api.w.org\/{rel}","templated":true}]}}