{"id":3782,"date":"2025-10-28T09:04:54","date_gmt":"2025-10-28T06:04:54","guid":{"rendered":"https:\/\/shedov.top\/ru\/?p=3782"},"modified":"2026-03-15T20:34:39","modified_gmt":"2026-03-15T17:34:39","slug":"opisanie-i-vozmozhnosti-crystal-v2-0","status":"publish","type":"post","link":"https:\/\/shedov.top\/ru\/opisanie-i-vozmozhnosti-crystal-v2-0\/","title":{"rendered":"\u041e\u043f\u0438\u0441\u0430\u043d\u0438\u0435 \u0438 \u0432\u043e\u0437\u043c\u043e\u0436\u043d\u043e\u0441\u0442\u0438 CRYSTAL v2.0"},"content":{"rendered":"<div class=\"single_contents\">\n   <h2>\u0421\u043e\u0434\u0435\u0440\u0436\u0430\u043d\u0438\u0435:<\/h2>\n   <nav>\n      <ul>\n         <li>1. <a href=\"#paragraph_1\">\u0410\u0440\u0445\u0438\u0442\u0435\u043a\u0442\u0443\u0440\u0430, \u0441\u0442\u0440\u043e\u0435\u043d\u0438\u0435 \u0438 \u0441\u0442\u0440\u0443\u043a\u0442\u0443\u0440\u0430 \u043f\u0440\u043e\u0435\u043a\u0442\u0430<\/a><\/li>\n         <li>2. <a href=\"#paragraph_2\">\u041e\u043f\u0438\u0441\u0430\u043d\u0438\u0435 \u0438 \u0432\u043e\u0437\u043c\u043e\u0436\u043d\u043e\u0441\u0442\u0438<\/a><\/li>\n         <li>3. <a href=\"#paragraph_3\">\u0412\u0438\u0434\u0435\u043e \u0441 \u0434\u0435\u043c\u043e\u043d\u0441\u0442\u0440\u0430\u0446\u0438\u0435\u0439 \u0432\u043e\u0437\u043c\u043e\u0436\u043d\u043e\u0441\u0442\u0435\u0439 CRYSTAL v2.0<\/a><\/li>\n      <\/ul>\n   <\/nav>\n<\/div>   \n\n<div class=\"single_top\"> \n<h2 id=\"paragraph_1\">\u0410\u0440\u0445\u0438\u0442\u0435\u043a\u0442\u0443\u0440\u0430:<\/h2>\n<p>SPA, RESTful API, FSD.<\/p>\n \n<h2>\u0421\u0442\u0440\u043e\u0435\u043d\u0438\u0435:<\/h2>\n<p><a href=\"https:\/\/github.com\/CrystalSystems\/crystal-v2.0\" target=\"_blank\" rel=\"noopener\" >\n   Full code<\/a> |\n   Package.json:\n   <a href=\"https:\/\/github.com\/CrystalSystems\/crystal-v2.0\/blob\/main\/main\/frontend\/package.json\" target=\"_blank\"\n      rel=\"noopener\" >\n   frontend<\/a> &#x2234; <a href=\"https:\/\/github.com\/CrystalSystems\/crystal-v2.0\/blob\/main\/main\/backend\/package.json\"\n      target=\"_blank\" rel=\"noopener\">\n   backend<\/a>\n<\/p>\n \n<h2>\u0421\u0442\u0440\u0443\u043a\u0442\u0443\u0440\u0430:<\/h2>\n<ul class=\"stack_color\">\n    <li>\n        <strong>M<\/strong>ongoDB v8.0.4 \n<\/li> \n<li> \n<strong>E<\/strong>xpress.js v4.21.2 \n<\/li> \n<li> \n<strong>R<\/strong>eact v19.0.0 \n<\/li> \n<li> \n<strong>N<\/strong>ode.js v24.0.2\n   <\/li>\n   <li>\n      NPM v11.3.0\n   <\/li>\n   <li>\n      PM2 v5.4.3\n   <\/li>\n   <li>\n      Vite v6.1.0\n   <\/li>\n<\/ul>\n<\/div>\n\n<h2 id=\"paragraph_2\" style=\"text-align:center; margin-bottom: 9px;\">\u041e\u043f\u0438\u0441\u0430\u043d\u0438\u0435 \u0438 \u0432\u043e\u0437\u043c\u043e\u0436\u043d\u043e\u0441\u0442\u0438<\/h2>\n<p style=\"margin-bottom: 9px;\">\u0424\u0443\u043d\u043a\u0446\u0438\u043e\u043d\u0430\u043b\u044c\u043d\u043e, \u0434\u0430\u043d\u043d\u0430\u044f \u0432\u0435\u0440\u0441\u0438\u044f \u043f\u0440\u0430\u043a\u0442\u0438\u0447\u0435\u0441\u043a\u0438 \u043f\u043e\u043b\u043d\u043e\u0441\u0442\u044c\u044e \u0438\u0434\u0435\u043d\u0442\u0438\u0447\u043d\u0430 <a href=\"https:\/\/shedov.top\/ru\/opisanie-i-vozmozhnosti-crystal-v1-0\/\" rel=\"noopener\" target=\"_blank\">CRYSTAL v1.0<\/a>, \u043d\u043e \u043e\u0431\u043b\u0430\u0434\u0430\u0435\u0442 \u0440\u044f\u0434\u043e\u043c \u043a\u043b\u044e\u0447\u0435\u0432\u044b\u0445 \u0443\u043b\u0443\u0447\u0448\u0435\u043d\u0438\u0439:<\/p>\n<ul>\n\n<li>1. \u0423\u043b\u0443\u0447\u0448\u0435\u043d UX\/UI \u0434\u0438\u0437\u0430\u0439\u043d, \u0434\u043b\u044f \u0431\u043e\u043b\u044c\u0448\u0438\u0445 \u043f\u043b\u0430\u043d\u0448\u0435\u0442\u043d\u044b\u0445 \u044d\u043a\u0440\u0430\u043d\u043e\u0432 (iPad Pro \u0438 \u043f\u043e\u0434\u043e\u0431\u043d\u044b\u0435). \u0411\u043e\u043a\u043e\u0432\u0430\u044f \u043d\u0430\u0432\u0438\u0433\u0430\u0446\u0438\u043e\u043d\u043d\u0430\u044f \u043f\u0430\u043d\u0435\u043b\u044c, \u0441\u0442\u0430\u043b\u0430 \u0431\u043e\u043b\u0435\u0435 \u043a\u043e\u043c\u043f\u0430\u043a\u0442\u043d\u043e\u0439, \u0447\u0442\u043e \u043f\u043e\u0437\u0432\u043e\u043b\u0438\u043b\u043e \u0443\u0432\u0435\u043b\u0438\u0447\u0438\u0442\u044c \u043e\u0431\u043b\u0430\u0441\u0442\u044c \u043e\u0442\u043e\u0431\u0440\u0430\u0436\u0435\u043d\u0438\u044f \u043e\u0441\u043d\u043e\u0432\u043d\u043e\u0433\u043e \u043a\u043e\u043d\u0442\u0435\u043d\u0442\u0430:\n \n<div style=\"margin-top: 15px;\" class=\"blog_post_image_wrap\">\n    <div class=\"blog_post_image\">\n        <a href=\"https:\/\/shedov.top\/wp-content\/images\/description_crystal_v2.0_gif_2.gif\" data-lbwps-width=\"5575\" data-lbwps-height=\"3717\" data-lbwps-srcsmall=\"https:\/\/shedov.top\/wp-content\/images\/description_crystal_v2.0_gif_2.gif\"><img decoding=\"async\" src=\"https:\/\/shedov.top\/wp-content\/images\/description_crystal_v2.0_gif_2.gif\" alt=\"gif 2\"><\/a>\n    <\/div>\n<\/div>\n<div style=\"margin-bottom: 20px;\" class=\"blog_post_screenshot_number\">\n <p>iPad Pro 13\" 2025 (iOS v26, Safari)<\/p>\n<\/div>\n<\/li>\n\n<li>2. Mongoose \u0431\u044b\u043b \u0443\u0434\u0430\u043b\u0451\u043d \u0438 \u0437\u0430\u043c\u0435\u043d\u0451\u043d <a href=\"https:\/\/www.npmjs.com\/package\/mongodb\" rel=\"noopener\" target=\"_blank\">\u043d\u0430\u0442\u0438\u0432\u043d\u044b\u043c \u0434\u0440\u0430\u0439\u0432\u0435\u0440\u043e\u043c<\/a> MongoDB.<\/li>\n<li>3. \u0421\u0445\u0435\u043c\u044b \u0434\u0430\u043d\u043d\u044b\u0445 \u0434\u043b\u044f \u0432\u0441\u0435\u0445 \u043a\u043e\u043b\u043b\u0435\u043a\u0446\u0438\u0439 (<code class=\"inline-code\"><a href=\"https:\/\/github.com\/CrystalSystems\/crystal-v2.0\/blob\/main\/main\/backend\/src\/modules\/user\/user.schema.js\" rel=\"noopener\" target=\"_blank\">users<\/a><\/code>, <code class=\"inline-code\"><a href=\"https:\/\/github.com\/CrystalSystems\/crystal-v2.0\/blob\/main\/main\/backend\/src\/modules\/post\/post.schema.js\" rel=\"noopener\" target=\"_blank\">posts<\/a><\/code>, <code class=\"inline-code\"><a href=\"https:\/\/github.com\/CrystalSystems\/crystal-v2.0\/blob\/main\/main\/backend\/src\/modules\/like\/like.schema.js\" rel=\"noopener\" target=\"_blank\">likes<\/a><\/code>, <code class=\"inline-code\"><a href=\"https:\/\/github.com\/CrystalSystems\/crystal-v2.0\/blob\/main\/main\/backend\/src\/modules\/hashtag\/hashtag.schema.js\" rel=\"noopener\" target=\"_blank\">hashtags<\/a><\/code>), \u043e\u043f\u0440\u0435\u0434\u0435\u043b\u0435\u043d\u044b \u0441 \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u043d\u0438\u0435\u043c \u0441\u0442\u0430\u043d\u0434\u0430\u0440\u0442\u0430 JSON Schema \u0438 <a href=\"https:\/\/github.com\/CrystalSystems\/crystal-v2.0\/blob\/main\/main\/backend\/src\/core\/engine\/db\/initializeCollections.js\" rel=\"noopener\" target=\"_blank\">\u0438\u043d\u0438\u0446\u0438\u0430\u043b\u0438\u0437\u0438\u0440\u0443\u044e\u0442\u0441\u044f<\/a> \u0432 MongoDB \u0441 \u043f\u043e\u043c\u043e\u0449\u044c\u044e \u043e\u043f\u0435\u0440\u0430\u0442\u043e\u0440\u0430 <a href=\"https:\/\/www.mongodb.com\/docs\/manual\/reference\/operator\/query\/jsonSchema\/#mongodb-query-op.-jsonSchema\" rel=\"noopener\" target=\"_blank\">$jsonSchema<\/a>. \u0422\u0430\u043a\u043e\u0439 \u043f\u043e\u0434\u0445\u043e\u0434 \u043e\u0431\u0435\u0441\u043f\u0435\u0447\u0438\u0432\u0430\u0435\u0442 \u0441\u043e\u0433\u043b\u0430\u0441\u043e\u0432\u0430\u043d\u043d\u043e\u0441\u0442\u044c \u0438 \u0435\u0434\u0438\u043d\u0443\u044e \u0441\u0442\u0440\u0443\u043a\u0442\u0443\u0440\u0443 \u0434\u043b\u044f \u0434\u043e\u043a\u0443\u043c\u0435\u043d\u0442\u043e\u0432 \u0432 \u043a\u043e\u043b\u043b\u0435\u043a\u0446\u0438\u044f\u0445.<\/li>\n\n<li>4. \u0414\u043b\u044f <a href=\"https:\/\/github.com\/CrystalSystems\/crystal-v2.0\/blob\/main\/main\/backend\/src\/modules\/hashtag\/hashtag.schema.js\" rel=\"noopener\" target=\"_blank\">\u0445\u0435\u0448\u0442\u0435\u0433\u043e\u0432<\/a> \u0438 <a href=\"https:\/\/github.com\/CrystalSystems\/crystal-v2.0\/blob\/main\/main\/backend\/src\/modules\/like\/like.schema.js\" rel=\"noopener\" target=\"_blank\">\u043b\u0430\u0439\u043a\u043e\u0432<\/a> \u0431\u044b\u043b\u0438 \u0441\u043e\u0437\u0434\u0430\u043d\u044b \u043e\u0442\u0434\u0435\u043b\u044c\u043d\u044b\u0435 \u043a\u043e\u043b\u043b\u0435\u043a\u0446\u0438\u0438 \u0441 \u0434\u0435\u043d\u043e\u0440\u043c\u0430\u043b\u0438\u0437\u0430\u0446\u0438\u0435\u0439 \u0438 \u0438\u043d\u0434\u0435\u043a\u0441\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u0435\u043c, \u0447\u0442\u043e \u043e\u0431\u0435\u0441\u043f\u0435\u0447\u0438\u0442 \u0431\u043e\u043b\u0435\u0435 \u0432\u044b\u0441\u043e\u043a\u0443\u044e \u043f\u0440\u043e\u0438\u0437\u0432\u043e\u0434\u0438\u0442\u0435\u043b\u044c\u043d\u043e\u0441\u0442\u044c \u043f\u0440\u0438 \u0431\u043e\u043b\u044c\u0448\u043e\u043c \u043e\u0431\u044a\u0435\u043c\u0435 \u0434\u0430\u043d\u043d\u044b\u0445.<\/li>\n\n<li>5. \u0414\u043b\u044f \u043f\u043e\u0438\u0441\u043a\u0430 \u043f\u043e \u043a\u043e\u043d\u0442\u0435\u043d\u0442\u0443 \u043f\u043e\u0441\u0442\u043e\u0432, \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u0435\u0442\u0441\u044f \u043f\u043e\u043b\u043d\u043e\u0442\u0435\u043a\u0441\u0442\u043e\u0432\u044b\u0439 \u043f\u043e\u0438\u0441\u043a MongoDB \u043d\u0430 \u043e\u0441\u043d\u043e\u0432\u0435 \u043e\u043f\u0435\u0440\u0430\u0442\u043e\u0440\u0430 <a href=\"https:\/\/www.mongodb.com\/docs\/manual\/reference\/operator\/query\/text\/\" rel=\"noopener\" target=\"_blank\">$text<\/a>. Frontend (<a href=\"https:\/\/github.com\/CrystalSystems\/crystal-v2.0\/blob\/main\/main\/frontend\/src\/pages\/SearchPage\/SearchPage.jsx\" rel=\"noopener\" target=\"_blank\">SearchPage.jsx<\/a>, <a href=\"https:\/\/github.com\/CrystalSystems\/crystal-v2.0\/blob\/main\/main\/frontend\/src\/shared\/ui\/Search\/Search.jsx\" rel=\"noopener\" target=\"_blank\">Search.jsx<\/a>) | Backend (<a href=\"https:\/\/github.com\/CrystalSystems\/crystal-v2.0\/blob\/98498943156ac63dd3f00c2012ff45712aecfc99\/backend\/src\/modules\/post\/post.controller.js#L689\" rel=\"noopener\" target=\"_blank\">searchPosts<\/a>):\n\n<div style=\"margin-top: 15px;\" class=\"blog_post_image_wrap\"> \n<div class=\"blog_post_image\">\n<a href=\"https:\/\/shedov.top\/wp-content\/images\/description_crystal_v2.0_gif_1.gif\" data-lbwps-width=\"5575\" data-lbwps-height=\"3717\" data-lbwps-srcsmall=\"https:\/\/shedov.top\/wp-content\/images\/description_crystal_v2.0_gif_1.gif\"><img decoding=\"async\" src=\"https:\/\/shedov.top\/wp-content\/images\/description_crystal_v2.0_gif_1.gif\" alt=\"\u0414\u0435\u043c\u043e\u043d\u0441\u0442\u0440\u0430\u0446\u0438\u044f \u043f\u043e\u0438\u0441\u043a\u043e\u0432\u043e\u0439 \u0441\u0438\u0441\u0442\u0435\u043c\u044b\"><\/a> \n<\/div>\n<\/div>\n<div style=\"margin-bottom: 20px;\" class=\"blog_post_screenshot_number\">\n<p>\u0414\u0435\u043c\u043e\u043d\u0441\u0442\u0440\u0430\u0446\u0438\u044f \u043f\u043e\u0438\u0441\u043a\u043e\u0432\u043e\u0439 \u0441\u0438\u0441\u0442\u0435\u043c\u044b<\/p>\n<\/div><\/li>\n\n<li>6. \u0414\u043e\u0431\u0430\u0432\u043b\u0435\u043d \u0441\u0442\u0430\u0442\u0443\u0441 \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044f online\/offline. \u041b\u043e\u0433\u0438\u043a\u0430 \u0440\u0435\u0430\u043b\u0438\u0437\u043e\u0432\u0430\u043d\u0430 \u0441 \u043f\u043e\u043c\u043e\u0449\u044c\u044e WebSocket (<a href=\"https:\/\/github.com\/CrystalSystems\/crystal-v2.0\/blob\/main\/main\/frontend\/src\/shared\/hooks\/useWebSocket\/useWebSocket.js\" rel=\"noopener\" target=\"_blank\">frontend<\/a> | <a href=\"https:\/\/github.com\/CrystalSystems\/crystal-v2.0\/blob\/main\/main\/backend\/src\/core\/engine\/web\/websocket.js\" rel=\"noopener\" target=\"_blank\">backend<\/a>).<\/li>\n<li>7. \u0412 \u0441\u0442\u0430\u0442\u0443\u0441\u0435 \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044f (\u043f\u0440\u0438 offline), \u0434\u043e\u0431\u0430\u0432\u043b\u0435\u043d \u043f\u043e\u043a\u0430\u0437 \u0432\u0440\u0435\u043c\u0435\u043d\u0438 \u043f\u043e\u0441\u043b\u0435\u0434\u043d\u0435\u0433\u043e \u043f\u043e\u0441\u0435\u0449\u0435\u043d\u0438\u044f \u0441\u0430\u0439\u0442\u0430.<\/li>\n<li>8. <a href=\"https:\/\/www.npmjs.com\/package\/multer\" rel=\"noopener\" target=\"_blank\">Multer<\/a> \u0437\u0430\u043c\u0435\u043d\u0435\u043d \u043d\u0430 <a href=\"https:\/\/www.npmjs.com\/package\/sharp\" rel=\"noopener\" target=\"_blank\">Sharp<\/a>. \u0412 <a href=\"https:\/\/github.com\/CrystalSystems\/crystal-v2.0\/blob\/main\/main\/backend\/src\/shared\/utils\/sharp\/sharp-upload.js\" rel=\"noopener\" target=\"_blank\">sharp-upload.js<\/a> \u0431\u044b\u043b\u0438 \u0434\u043e\u0431\u0430\u0432\u043b\u0435\u043d\u044b \u0441\u043b\u0435\u0434\u0443\u044e\u0449\u0438\u0435 \u043c\u0435\u0445\u0430\u043d\u0438\u0437\u043c\u044b \u0443\u043f\u0440\u0430\u0432\u043b\u0435\u043d\u0438\u044f \u0437\u0430\u0433\u0440\u0443\u0437\u043a\u043e\u0439 \u0438\u0437\u043e\u0431\u0440\u0430\u0436\u0435\u043d\u0438\u0439 \u0438 \u043e\u0431\u0435\u0441\u043f\u0435\u0447\u0435\u043d\u0438\u044f \u043a\u0438\u0431\u0435\u0440\u0431\u0435\u0437\u043e\u043f\u0430\u0441\u043d\u043e\u0441\u0442\u0438:\n\n<p><strong>\u2014 \u041e\u0433\u0440\u0430\u043d\u0438\u0447\u0435\u043d\u0438\u0435 \u043e\u0434\u043d\u043e\u0432\u0440\u0435\u043c\u0435\u043d\u043d\u043e\u0439 \u043e\u0431\u0440\u0430\u0431\u043e\u0442\u043a\u0438 \u0438\u0437\u043e\u0431\u0440\u0430\u0436\u0435\u043d\u0438\u0439 (<a href=\"https:\/\/github.com\/CrystalSystems\/crystal-v2.0\/blob\/ca54aec0bc7a5ef96b9172d005aca608829bd05e\/backend\/src\/shared\/utils\/sharp\/sharp-upload.js#L38\" rel=\"noopener\" target=\"_blank\">Semaphore<\/a>)<\/strong><\/p>\n <p>\u0414\u043b\u044f \u043f\u0440\u0435\u0434\u043e\u0442\u0432\u0440\u0430\u0449\u0435\u043d\u0438\u044f \u043f\u0435\u0440\u0435\u0433\u0440\u0443\u0437\u043a\u0438 \u043f\u0440\u043e\u0446\u0435\u0441\u0441\u043e\u0440\u0430 \u0432\u043e \u0432\u0440\u0435\u043c\u044f \u0440\u0435\u0441\u0443\u0440\u0441\u043e\u0435\u043c\u043a\u043e\u0439 \u043e\u0431\u0440\u0430\u0431\u043e\u0442\u043a\u0438 \u0438\u0437\u043e\u0431\u0440\u0430\u0436\u0435\u043d\u0438\u0439, \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u0435\u0442\u0441\u044f \u043c\u0435\u0445\u0430\u043d\u0438\u0437\u043c \u0441\u0435\u043c\u0430\u0444\u043e\u0440\u0430.<\/p>\n\n<p><strong>\u2014 \u041e\u0433\u0440\u0430\u043d\u0438\u0447\u0435\u043d\u0438\u0435 \u0447\u0430\u0441\u0442\u043e\u0442\u044b \u0437\u0430\u043f\u0440\u043e\u0441\u043e\u0432 (<a href=\"https:\/\/github.com\/CrystalSystems\/crystal-v2.0\/blob\/ca54aec0bc7a5ef96b9172d005aca608829bd05e\/backend\/src\/shared\/utils\/sharp\/sharp-upload.js#L83\" rel=\"noopener\" target=\"_blank\">Rate Limiting<\/a>)<\/strong><\/p>\n<p>\u0414\u043b\u044f \u0437\u0430\u0449\u0438\u0442\u044b \u043e\u0442 DDoS-\u0430\u0442\u0430\u043a \u0438 \u0441\u043f\u0430\u043c\u0430, \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u0435\u0442\u0441\u044f \u043e\u0433\u0440\u0430\u043d\u0438\u0447\u0435\u043d\u0438\u0435 \u043a\u043e\u043b\u0438\u0447\u0435\u0441\u0442\u0432\u0430 \u0437\u0430\u043f\u0440\u043e\u0441\u043e\u0432 \u043d\u0430 \u0437\u0430\u0433\u0440\u0443\u0437\u043a\u0443 \u0441 \u043e\u0434\u043d\u043e\u0433\u043e IP-\u0430\u0434\u0440\u0435\u0441\u0430.<\/p>\n\n<p><strong>\u2014 \u041e\u0433\u0440\u0430\u043d\u0438\u0447\u0435\u043d\u0438\u0435 \u0440\u0430\u0437\u043c\u0435\u0440\u0430 \u0437\u0430\u0433\u0440\u0443\u0436\u0430\u0435\u043c\u043e\u0433\u043e \u0444\u0430\u0439\u043b\u0430<\/strong><\/p>\n<p>\u041f\u0440\u043e\u0432\u0435\u0440\u043a\u0430 <a href=\"https:\/\/github.com\/CrystalSystems\/crystal-v2.0\/blob\/ca54aec0bc7a5ef96b9172d005aca608829bd05e\/backend\/src\/shared\/utils\/sharp\/sharp-upload.js#L123\" rel=\"noopener\" target=\"_blank\">\u043b\u0438\u043c\u0438\u0442\u0430<\/a> \u043f\u0440\u043e\u0438\u0441\u0445\u043e\u0434\u0438\u0442 \u043d\u0430 \u0440\u0430\u043d\u043d\u0435\u0439 \u0441\u0442\u0430\u0434\u0438\u0438 \u0437\u0430\u0433\u0440\u0443\u0437\u043a\u0438, \u0447\u0442\u043e\u0431\u044b \u0438\u0437\u0431\u0435\u0436\u0430\u0442\u044c \u0447\u0442\u0435\u043d\u0438\u044f \u0441\u043b\u0438\u0448\u043a\u043e\u043c \u0431\u043e\u043b\u044c\u0448\u0438\u0445 \u0444\u0430\u0439\u043b\u043e\u0432 \u0432 \u043f\u0430\u043c\u044f\u0442\u044c.<\/p>\n\n<p><strong>\u2014 \u041f\u0440\u043e\u0446\u0435\u0441\u0441 \u0432\u0430\u043b\u0438\u0434\u0430\u0446\u0438\u0438 \u0437\u0430\u0433\u0440\u0443\u0436\u0430\u0435\u043c\u043e\u0433\u043e \u0444\u0430\u0439\u043b\u0430<\/strong><\/p>\n<p>\u041f\u043e\u0441\u043b\u0435 \u043f\u0440\u043e\u0445\u043e\u0436\u0434\u0435\u043d\u0438\u044f \u043f\u0435\u0440\u0432\u0438\u0447\u043d\u044b\u0445 \u043f\u0440\u043e\u0432\u0435\u0440\u043e\u043a (Semaphore \u0438 Rate Limiting), \u0437\u0430\u0433\u0440\u0443\u0436\u0435\u043d\u043d\u044b\u0439 \u0444\u0430\u0439\u043b \u043f\u043e\u0434\u0432\u0435\u0440\u0433\u0430\u0435\u0442\u0441\u044f \u0434\u0432\u043e\u0439\u043d\u043e\u0439 \u043f\u0440\u043e\u0432\u0435\u0440\u043a\u0435 (<code class=\"inline-code\"><a href=\"https:\/\/github.com\/CrystalSystems\/crystal-v2.0\/blob\/ca54aec0bc7a5ef96b9172d005aca608829bd05e\/backend\/src\/shared\/utils\/sharp\/sharp-upload.js#L193\" rel=\"noopener\" target=\"_blank\">!isImageExtension<\/a><\/code> \u0438 <code class=\"inline-code\"><a href=\"https:\/\/github.com\/CrystalSystems\/crystal-v2.0\/blob\/ca54aec0bc7a5ef96b9172d005aca608829bd05e\/backend\/src\/shared\/utils\/sharp\/sharp-upload.js#L194\" rel=\"noopener\" target=\"_blank\">!isImageMime<\/a><\/code>), \u0447\u0442\u043e\u0431\u044b \u0433\u0430\u0440\u0430\u043d\u0442\u0438\u0440\u043e\u0432\u0430\u0442\u044c, \u0447\u0442\u043e \u044d\u0442\u043e \u0434\u0435\u0439\u0441\u0442\u0432\u0438\u0442\u0435\u043b\u044c\u043d\u043e \u0431\u0435\u0437\u043e\u043f\u0430\u0441\u043d\u043e\u0435 \u0438\u0437\u043e\u0431\u0440\u0430\u0436\u0435\u043d\u0438\u0435. \u0421\u0438\u0441\u0442\u0435\u043c\u0430 \u043e\u0434\u043d\u043e\u0432\u0440\u0435\u043c\u0435\u043d\u043d\u043e \u043f\u0440\u043e\u0432\u0435\u0440\u044f\u0435\u0442 \u0434\u0432\u0435 \u043d\u0435\u0437\u0430\u0432\u0438\u0441\u0438\u043c\u044b\u0435 \u0445\u0430\u0440\u0430\u043a\u0442\u0435\u0440\u0438\u0441\u0442\u0438\u043a\u0438 \u0444\u0430\u0439\u043b\u0430: \u0440\u0430\u0441\u0448\u0438\u0440\u0435\u043d\u0438\u0435 (\u043f\u0440\u043e\u0432\u0435\u0440\u043a\u0430 \u0438\u043c\u0435\u043d\u0438 \u0444\u0430\u0439\u043b\u0430 \u043d\u0430 \u043d\u0430\u043b\u0438\u0447\u0438\u0435 \u043e\u0434\u043d\u043e\u0433\u043e \u0438\u0437 \u0440\u0430\u0437\u0440\u0435\u0448\u0435\u043d\u043d\u044b\u0445 \u0440\u0430\u0441\u0448\u0438\u0440\u0435\u043d\u0438\u0439: <code class=\"inline-code\">jpe?g|png|webp|gif<\/code>) \u0438 MIME-\u0442\u0438\u043f, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u0434\u043e\u043b\u0436\u0435\u043d \u0441\u043e\u043e\u0442\u0432\u0435\u0442\u0441\u0442\u0432\u043e\u0432\u0430\u0442\u044c: <code class=\"inline-code\">image\\\/(jpeg|png|webp|gif)<\/code>. \u0415\u0441\u043b\u0438 \u0445\u043e\u0442\u044f \u0431\u044b \u043e\u0434\u043d\u0430 \u0438\u0437 \u044d\u0442\u0438\u0445 \u043f\u0440\u043e\u0432\u0435\u0440\u043e\u043a \u043d\u0435 \u043f\u0440\u043e\u0439\u0434\u0435\u043d\u0430, \u0444\u0430\u0439\u043b \u043f\u0435\u0440\u0435\u0445\u043e\u0434\u0438\u0442 \u043a <a href=\"https:\/\/github.com\/CrystalSystems\/crystal-v2.0\/blob\/ca54aec0bc7a5ef96b9172d005aca608829bd05e\/backend\/src\/shared\/utils\/sharp\/sharp-upload.js#L198\" rel=\"noopener\" target=\"_blank\">\u0441\u043f\u0435\u0446\u0438\u0430\u043b\u044c\u043d\u043e\u0439 \u043f\u0440\u043e\u0432\u0435\u0440\u043a\u0435 GIF<\/a>, \u0438 \u043f\u043e\u0441\u043b\u0435\u0434\u0443\u044e\u0449\u0435\u0439 \u0444\u0443\u043d\u043a\u0446\u0438\u0438 <code class=\"inline-code\"><a href=\"https:\/\/github.com\/CrystalSystems\/crystal-v2.0\/blob\/ca54aec0bc7a5ef96b9172d005aca608829bd05e\/backend\/src\/shared\/utils\/sharp\/sharp-upload.js#L8\" rel=\"noopener\" target=\"_blank\">isValidGif(fileBuffer)<\/a><\/code>, \u043a\u043e\u0442\u043e\u0440\u0430\u044f \u043f\u0440\u043e\u0432\u0435\u0440\u044f\u0435\u0442 \"Magic Bytes\" \u0432 \u0437\u0430\u0433\u043e\u043b\u043e\u0432\u043a\u0435 \u0444\u0430\u0439\u043b\u0430 (<code class=\"inline-code\">GIF87a<\/code> \u0438\u043b\u0438 <code class=\"inline-code\">GIF89a<\/code>).<\/p>\n<\/li>\n<li>9. GIF \u043f\u0440\u043e\u0445\u043e\u0434\u044f\u0442 \u0441\u0430\u043d\u0438\u0442\u0438\u0437\u0430\u0446\u0438\u044e \u0447\u0435\u0440\u0435\u0437 <a href=\"https:\/\/github.com\/CrystalSystems\/crystal-v2.0\/blob\/6c3478f5ea8bb037b4de42b0feff5f6560c753c4\/backend\/src\/shared\/utils\/sharp\/sharp-upload.js#L228\" rel=\"noopener\" target=\"_blank\">\u0441\u043f\u0435\u0446\u0438\u0430\u043b\u044c\u043d\u0443\u044e \u043b\u043e\u0433\u0438\u043a\u0443<\/a>. \u0412\u0441\u0435 \u0438\u0437\u043e\u0431\u0440\u0430\u0436\u0435\u043d\u0438\u044f \u043a\u0440\u043e\u043c\u0435 GIF, <a href=\"https:\/\/github.com\/CrystalSystems\/crystal-v2.0\/blob\/6c3478f5ea8bb037b4de42b0feff5f6560c753c4\/backend\/src\/shared\/utils\/sharp\/sharp-upload.js#L251\" rel=\"noopener\" target=\"_blank\">\u043a\u043e\u043d\u0432\u0435\u0440\u0442\u0438\u0440\u0443\u044e\u0442\u0441\u044f<\/a> \u0432 WebP.<\/li>\n\n<li>10. \u0414\u043e\u0431\u0430\u0432\u043b\u0435\u043d\u0430 \u043d\u0430\u0441\u0442\u0440\u043e\u0439\u043a\u0430 \u0438\u043d\u0442\u0435\u0440\u0444\u0435\u0439\u0441\u0430, \u043a\u043e\u0442\u043e\u0440\u0430\u044f \u043f\u043e\u0437\u0432\u043e\u043b\u044f\u0435\u0442 \u0441\u043a\u0440\u044b\u0442\u044c \u0432\u0441\u0435 GIF \u0438\u0437\u043e\u0431\u0440\u0430\u0436\u0435\u043d\u0438\u044f \u043d\u0430 \u0441\u0430\u0439\u0442\u0435:\n<div style=\"margin-top: 15px;\" class=\"blog_post_image_wrap\">\n    <div class=\"blog_post_image\">\n        <a href=\"https:\/\/shedov.top\/wp-content\/images\/description_crystal_v2.0_screenshot_3.webp\" data-lbwps-width=\"5575\" data-lbwps-height=\"3717\" data-lbwps-srcsmall=\"https:\/\/shedov.top\/wp-content\/images\/description_crystal_v2.0_screenshot_3.webp\"><img decoding=\"async\" src=\"https:\/\/shedov.top\/wp-content\/images\/description_crystal_v2.0_screenshot_3.webp\" alt=\"\u0441\u043a\u0440\u0438\u043d 3\"><\/a>\n    <\/div>\n<\/div>\n<div style=\"margin-bottom: 20px;\" class=\"blog_post_screenshot_number\">\n <p>\u0421\u043a\u0440\u044b\u0442\u044b\u0435 GIF \u0438\u0437\u043e\u0431\u0440\u0430\u0436\u0435\u043d\u0438\u044f, \u0441\u0432\u0435\u0442\u043b\u0430\u044f \u0442\u0435\u043c\u0430<\/p>\n<\/div>\n\n<div class=\"blog_post_image_wrap\">\n    <div class=\"blog_post_image\">\n        <a href=\"https:\/\/shedov.top\/wp-content\/images\/description_crystal_v2.0_screenshot_4.webp\" data-lbwps-width=\"5575\" data-lbwps-height=\"3717\" data-lbwps-srcsmall=\"https:\/\/shedov.top\/wp-content\/images\/description_crystal_v2.0_screenshot_4.webp\"><img decoding=\"async\" src=\"https:\/\/shedov.top\/wp-content\/images\/description_crystal_v2.0_screenshot_4.webp\" alt=\"\u0441\u043a\u0440\u0438\u043d 4\"><\/a>\n    <\/div>\n<\/div>\n<div style=\"margin-bottom: 15px;\" class=\"blog_post_screenshot_number\">\n <p>\u0421\u043a\u0440\u044b\u0442\u044b\u0435 GIF \u0438\u0437\u043e\u0431\u0440\u0430\u0436\u0435\u043d\u0438\u044f, \u0442\u0435\u043c\u043d\u0430\u044f \u0442\u0435\u043c\u0430<\/p>\n<\/div>\n\n<\/li>\n\n<li>11. \u0414\u043e\u0431\u0430\u0432\u043b\u0435\u043d\u0430 \u0432\u043e\u0437\u043c\u043e\u0436\u043d\u043e\u0441\u0442\u044c \u0443\u043a\u0430\u0437\u0430\u0442\u044c \u043f\u043e\u043b \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044f.<\/li>\n<li>12. \u041d\u0430 \u0441\u0442\u0440\u0430\u043d\u0438\u0446\u0435 \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044f, \u0434\u043e\u0431\u0430\u0432\u043b\u0435\u043d \u0440\u0430\u0437\u0434\u0435\u043b \u0441 \u043f\u043e\u0434\u0440\u043e\u0431\u043d\u043e\u0439 \u0438\u043d\u0444\u043e\u0440\u043c\u0430\u0446\u0438\u0435\u0439 \u043e \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u0435: \u043f\u043e\u043b, \u0434\u0430\u0442\u0430 \u0440\u0435\u0433\u0438\u0441\u0442\u0440\u0430\u0446\u0438\u0438.<\/li>\n<li>13. \u0414\u043e\u0431\u0430\u0432\u043b\u0435\u043d\u0430 \u043d\u0430\u0441\u0442\u0440\u043e\u0439\u043a\u0430 \u043a\u043e\u043d\u0444\u0438\u0434\u0435\u043d\u0446\u0438\u0430\u043b\u044c\u043d\u043e\u0441\u0442\u0438, \u043a\u043e\u0442\u043e\u0440\u0430\u044f \u043f\u043e\u0437\u0432\u043e\u043b\u044f\u0435\u0442 \u0441\u043a\u0440\u044b\u0442\u044c \u043f\u043e\u043b.<\/li>\n<li>14. \u0414\u043b\u044f \u0443\u0432\u0435\u043b\u0438\u0447\u0435\u043d\u0438\u044f \u043f\u0440\u043e\u0438\u0437\u0432\u043e\u0434\u0438\u0442\u0435\u043b\u044c\u043d\u043e\u0441\u0442\u0438, offset-\u043f\u0430\u0433\u0438\u043d\u0430\u0446\u0438\u044f \u0431\u044b\u043b\u0430 \u0437\u0430\u043c\u0435\u043d\u0435\u043d\u0430 \u043d\u0430 \u043a\u0443\u0440\u0441\u043e\u0440\u043d\u0443\u044e, \u0432 \u0440\u0430\u0437\u0434\u0435\u043b\u0430\u0445 \u0432\u044b\u0432\u043e\u0434\u0430 \u043b\u0430\u0439\u043a\u043e\u0432 \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044f, \u043f\u043e\u0441\u0442\u043e\u0432 \u0441 \u043e\u043f\u0440\u0435\u0434\u0435\u043b\u0451\u043d\u043d\u044b\u043c \u0445\u0435\u0448\u0442\u0435\u0433\u043e\u043c \u0438 \u043f\u043e\u0438\u0441\u043a\u0435 \u043f\u043e\u0441\u0442\u043e\u0432.<\/li>\n<li>15. \u0414\u043e\u0431\u0430\u0432\u043b\u0435\u043d\u0430 \u043a\u043d\u043e\u043f\u043a\u0430 \u2014 '\u041d\u0430\u0437\u0430\u0434'.<\/li>\n<li>16. \u0414\u043e\u0431\u0430\u0432\u043b\u0435\u043d\u0430 \u043b\u043e\u0433\u0438\u043a\u0430 \u0443\u0434\u0430\u043b\u0435\u043d\u0438\u044f \u0441\u0442\u0430\u0440\u044b\u0445 \u0438\u0437\u043e\u0431\u0440\u0430\u0436\u0435\u043d\u0438\u0439 \u0443 \u043f\u043e\u0441\u0442\u043e\u0432 \u0438 \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u0435\u0439: \u043f\u043e\u0441\u043b\u0435 \u0443\u0434\u0430\u043b\u0435\u043d\u0438\u044f\/\u0437\u0430\u043c\u0435\u043d\u044b \u0438\u0437\u043e\u0431\u0440\u0430\u0436\u0435\u043d\u0438\u0439, \u043f\u043e\u0441\u043b\u0435 \u0443\u0434\u0430\u043b\u0435\u043d\u0438\u044f \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044f \u0438\u043b\u0438 \u043f\u043e\u0441\u0442\u0430.<\/li>\n\n<li>17. \u0414\u043e\u0431\u0430\u0432\u043b\u0435\u043d\u0430 <a href=\"https:\/\/github.com\/CrystalSystems\/crystal-v2.0\/blob\/main\/main\/backend\/src\/shared\/helpers\/extract-hashtags-from-text\/extract-hashtags-from-text.js\" target=\"_blank\" rel=\"noopener\" >\u0432\u0430\u043b\u0438\u0434\u0430\u0446\u0438\u044f<\/a> \u0445\u0435\u0448\u0442\u0435\u0433\u043e\u0432 \u0432 backend, \u043a\u043e\u0442\u043e\u0440\u0430\u044f \u043d\u0435 \u043f\u043e\u0437\u0432\u043e\u043b\u044f\u0435\u0442 \u0441\u043e\u0445\u0440\u0430\u043d\u044f\u0442\u044c \u0445\u0435\u0448\u0442\u0435\u0433\u0438 \u0432\u0438\u0434\u0430: <code class=\"inline-code\">##Test<\/code>, <code class=\"inline-code\">#Te#st<\/code>, <code class=\"inline-code\">#Te?st<\/code> \u0438 \u0442.\u0434. \u041f\u0440\u043e\u0432\u0435\u0440\u043a\u0430 \u043f\u0440\u043e\u0438\u0441\u0445\u043e\u0434\u0438\u0442 \u0441 \u043f\u043e\u043c\u043e\u0449\u044c\u044e \u0440\u0435\u0433\u0443\u043b\u044f\u0440\u043d\u043e\u0433\u043e \u0432\u044b\u0440\u0430\u0436\u0435\u043d\u0438\u044f \u2014 <code class=\"inline-code\">\/^[\\p{L}0-9_-]+$\/u<\/code> (\u0420\u0430\u0437\u0440\u0435\u0448\u0430\u0435\u0442 \u043b\u044e\u0431\u044b\u0435 \u0431\u0443\u043a\u0432\u044b Unicode, \u0446\u0438\u0444\u0440\u044b, \u0434\u0435\u0444\u0438\u0441 \u0438 \u043d\u0438\u0436\u043d\u0435\u0435 \u043f\u043e\u0434\u0447\u0435\u0440\u043a\u0438\u0432\u0430\u043d\u0438\u0435). \u0422\u0430\u043a \u0436\u0435, \u043c\u043e\u0436\u043d\u043e \u0437\u0430\u0434\u0430\u0442\u044c \u0434\u043e\u043f\u0443\u0441\u0442\u0438\u043c\u043e\u0435 \u043a\u043e\u043b\u0438\u0447\u0435\u0441\u0442\u0432\u043e \u0445\u0435\u0448\u0442\u0435\u0433\u043e\u0432 \u0432 \u043e\u0434\u043d\u043e\u043c \u043f\u043e\u0441\u0442\u0435 \u0438 \u0434\u043b\u0438\u043d\u0443 \u0445\u0435\u0448\u0442\u0435\u0433\u0430 \u0441 \u043f\u043e\u043c\u043e\u0449\u044c\u044e \u043a\u043e\u043d\u0441\u0442\u0430\u043d\u0442: <code class=\"inline-code\">MAX_HASHTAGS_COUNT<\/code> \u0438 <code class=\"inline-code\">MAX_HASHTAG_LENGTH<\/code>. \u0415\u0441\u043b\u0438 \u0445\u0435\u0448\u0442\u0435\u0433 \u043d\u0435 \u043f\u0440\u043e\u0448\u0435\u043b \u043f\u0440\u043e\u0432\u0435\u0440\u043a\u0443, \u043e\u043d \u043d\u0435 \u043f\u043e\u043f\u0430\u0434\u0430\u0435\u0442 \u0432 \u0431\u0430\u0437\u0443 \u0434\u0430\u043d\u043d\u044b\u0445, \u043d\u043e \u043f\u0440\u0438 \u044d\u0442\u043e\u043c, \u043f\u043e\u0441\u0442 \u0441\u043e\u0437\u0434\u0430\u0435\u0442\u0441\u044f \u0438 \u0432 \u0435\u0433\u043e \u0442\u0435\u043a\u0441\u0442\u0435 \u0431\u0443\u0434\u0435\u0442 \u043d\u0435\u0432\u0430\u043b\u0438\u0434\u043d\u044b\u0439 \u0445\u0435\u0448\u0442\u0435\u0433 \u2014 <code class=\"inline-code\">#Te#st<\/code>. \u041f\u043e\u0441\u043b\u0435 \u0443\u0441\u043f\u0435\u0448\u043d\u043e\u0433\u043e \u043f\u0440\u043e\u0445\u043e\u0436\u0434\u0435\u043d\u0438\u044f \u043f\u0440\u043e\u0432\u0435\u0440\u043a\u0438, \u0445\u0435\u0448\u0442\u0435\u0433 <code class=\"inline-code\">#Test<\/code> \u0434\u043e\u0431\u0430\u0432\u043b\u044f\u0435\u0442\u0441\u044f \u0432 \u043f\u043e\u043b\u0435 <code class=\"inline-code\">name<\/code> \u043a\u043e\u043b\u043b\u0435\u043a\u0446\u0438\u0438 <code class=\"inline-code\">hashtags<\/code>, \u0432 \u043d\u0438\u0436\u043d\u0435\u043c \u0440\u0435\u0433\u0438\u0441\u0442\u0440\u0435 \u2014 <code class=\"inline-code\">test<\/code>.<\/li>\n\n<li>18. \u0414\u043e\u0431\u0430\u0432\u043b\u0435\u043d\u0430 <a href=\"https:\/\/github.com\/CrystalSystems\/crystal-v2.0\/blob\/main\/main\/frontend\/src\/shared\/helpers\/formatting\/formatLinksInText.jsx\" target=\"_blank\" rel=\"noopener\" >\u0432\u0430\u043b\u0438\u0434\u0430\u0446\u0438\u044f<\/a> \u0445\u0435\u0448\u0442\u0435\u0433\u043e\u0432 \u0432 frontend. \u0414\u043b\u044f \u043e\u0442\u043e\u0431\u0440\u0430\u0436\u0435\u043d\u0438\u044f \u0432 \u0432\u0438\u0434\u0435 \u043a\u043b\u0438\u043a\u0430\u0431\u0435\u043b\u044c\u043d\u043e\u0439 \u0441\u0441\u044b\u043b\u043a\u0438, \u0445\u0435\u0448\u0442\u0435\u0433 \u0434\u043e\u043b\u0436\u0435\u043d \u043f\u0440\u043e\u0439\u0442\u0438 \u0432\u0430\u043b\u0438\u0434\u0430\u0446\u0438\u044e \u0441 \u043f\u043e\u043c\u043e\u0449\u044c\u044e \u0440\u0435\u0433\u0443\u043b\u044f\u0440\u043d\u043e\u0433\u043e \u0432\u044b\u0440\u0430\u0436\u0435\u043d\u0438\u044f \u2014 <code class=\"inline-code\">\/^[\\p{L}0-9_-]+$\/u<\/code> (\u0420\u0430\u0437\u0440\u0435\u0448\u0430\u0435\u0442 \u043b\u044e\u0431\u044b\u0435 \u0431\u0443\u043a\u0432\u044b Unicode, \u0446\u0438\u0444\u0440\u044b, \u0434\u0435\u0444\u0438\u0441 \u0438 \u043d\u0438\u0436\u043d\u0435\u0435 \u043f\u043e\u0434\u0447\u0435\u0440\u043a\u0438\u0432\u0430\u043d\u0438\u0435).<\/li>\n<li>19. \u0414\u043e\u0431\u0430\u0432\u043b\u0435\u043d\u043e \u0431\u043e\u043b\u0435\u0435 \u0438\u043d\u0444\u043e\u0440\u043c\u0430\u0442\u0438\u0432\u043d\u043e\u0435 \u043e\u0442\u043e\u0431\u0440\u0430\u0436\u0435\u043d\u0438\u0435 \u0434\u0430\u0442\u044b \u0441\u043e\u0437\u0434\u0430\u043d\u0438\u044f \u0438 \u043e\u0431\u043d\u043e\u0432\u043b\u0435\u043d\u0438\u044f \u043f\u043e\u0441\u0442\u0430 \u0432 \n<a href=\"https:\/\/github.com\/CrystalSystems\/crystal-v2.0\/blob\/main\/frontend\/src\/pages\/FullPostPage\/FullPostPage.jsx\" target=\"_blank\" rel=\"noopener\">\u043f\u043e\u043b\u043d\u043e\u0439<\/a> \n\u0438 \n<a href=\"https:\/\/github.com\/CrystalSystems\/crystal-v2.0\/blob\/main\/main\/frontend\/src\/widgets\/PostPreview\/PostPreview.jsx\" target=\"_blank\" rel=\"noopener\">\u043f\u0440\u0435\u0432\u044c\u044e<\/a> \n\u0432\u0435\u0440\u0441\u0438\u0438. \u0424\u043e\u0440\u043c\u0430\u0442\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u0435 \u0434\u0430\u0442\u044b \u043f\u0440\u043e\u0438\u0441\u0445\u043e\u0434\u0438\u0442 \u0432 \u0441\u043f\u0435\u0446\u0438\u0430\u043b\u044c\u043d\u043e\u043c \u0445\u0443\u043a\u0435 \u2014 \n<a href=\"https:\/\/github.com\/CrystalSystems\/crystal-v2.0\/blob\/main\/main\/frontend\/src\/shared\/hooks\/formatting\/useFormattedPostDate\/useFormattedPostDate.jsx\" target=\"_blank\" rel=\"noopener\">useFormattedPostDate<\/a>, \n\u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u0432\u044b\u043f\u043e\u043b\u043d\u044f\u0435\u0442 \u043b\u043e\u043a\u0430\u043b\u0438\u0437\u043e\u0432\u0430\u043d\u043d\u043e\u0435 \u0444\u043e\u0440\u043c\u0430\u0442\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u0435 \u0434\u0430\u0442\u044b \u0438 \u0432\u0440\u0435\u043c\u0435\u043d\u0438 \u043d\u0430 \u0434\u0432\u0443\u0445 \u044f\u0437\u044b\u043a\u0430\u0445 (\u0440\u0443\u0441\u0441\u043a\u043e\u043c \u0438 \u0430\u043d\u0433\u043b\u0438\u0439\u0441\u043a\u043e\u043c) \u0441 \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u043d\u0438\u0435\u043c \n<code class=\"inline-code\">toLocaleDateString<\/code> \u0438 <code class=\"inline-code\">toLocaleTimeString<\/code>. \n\u0414\u043b\u044f \u0430\u043d\u0433\u043b\u0438\u0439\u0441\u043a\u043e\u0439 \u043b\u043e\u043a\u0430\u043b\u0438 \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u0435\u0442\u0441\u044f 12-\u0447\u0430\u0441\u043e\u0432\u043e\u0439 \u0444\u043e\u0440\u043c\u0430\u0442 (<code class=\"inline-code\">Jul 4, 2025 \u2219 10:45 PM<\/code>), \n\u0430 \u0434\u043b\u044f \u0440\u0443\u0441\u0441\u043a\u043e\u0439 \u2014 24-\u0447\u0430\u0441\u043e\u0432\u043e\u0439 (<code class=\"inline-code\">4 \u0438\u044e\u043b\u044f 2025 \u2219 22:45<\/code>). \n\u0425\u0443\u043a \u0430\u0432\u0442\u043e\u043c\u0430\u0442\u0438\u0447\u0435\u0441\u043a\u0438 \u043e\u043f\u0440\u0435\u0434\u0435\u043b\u044f\u0435\u0442 \u0442\u0435\u043a\u0443\u0449\u0438\u0439 \u044f\u0437\u044b\u043a \u0438\u043d\u0442\u0435\u0440\u0444\u0435\u0439\u0441\u0430 \u0447\u0435\u0440\u0435\u0437 <code class=\"inline-code\">i18n.language<\/code> \n\u0438 \u043e\u0442\u043e\u0431\u0440\u0430\u0436\u0430\u0435\u0442 \u0433\u043e\u0434, \u0442\u043e\u043b\u044c\u043a\u043e \u0435\u0441\u043b\u0438 \u0434\u0430\u0442\u0430 \u043e\u0442\u043d\u043e\u0441\u0438\u0442\u0441\u044f \u043a \u043f\u0440\u0435\u0434\u044b\u0434\u0443\u0449\u0435\u043c\u0443 \u0433\u043e\u0434\u0443.<\/li>\n<li>20. \u0421\u0438\u0441\u0442\u0435\u043c\u0430 \u043a\u0438\u0431\u0435\u0440\u0431\u0435\u0437\u043e\u043f\u0430\u0441\u043d\u043e\u0441\u0442\u0438 \u0431\u0430\u0437\u044b \u0434\u0430\u043d\u043d\u044b\u0445, \u0441\u043e\u043e\u0442\u0432\u0435\u0442\u0441\u0442\u0432\u0443\u0435\u0442 <a href=\"https:\/\/shedov.top\/ru\/opisanie-i-vozmozhnosti-crystal-v1-0\/#paragraph_7\" rel=\"noopener\" target=\"_blank\">CRYSTAL v1.0 (Production)<\/a>.<\/li>\n<\/ul>\n\n<h2 id=\"paragraph_3\" style=\"text-align:center; margin-bottom: 17px; margin-top: 25px;\">\u0414\u0435\u043c\u043e\u043d\u0441\u0442\u0440\u0430\u0446\u0438\u044f \u0432\u043e\u0437\u043c\u043e\u0436\u043d\u043e\u0441\u0442\u0435\u0439 CRYSTAL v2.0<\/h2>\n<div style=\"margin-bottom: 37px;\" class=\"single_external_video\">\n   <iframe src=\"https:\/\/vk.com\/video_ext.php?oid=-161152224&id=456239083&hd=2&autoplay=0\"\n      allow=\"autoplay; encrypted-media; fullscreen; picture-in-picture; screen-wake-lock;\" style=\"border:0;\"\n      allowfullscreen><\/iframe>\n<\/div>\n<div class=\"browserstack_testing__announcement___wrap\">\n         <div class=\"browserstack_testing__announcement\">\n            <div class=\"browserstack_testing__announcement___top\">\n               <p>CRYSTAL \u0442\u0435\u0441\u0442\u0438\u0440\u0443\u0435\u0442\u0441\u044f \u0432 <\/p>\n            <\/div>\n            <div class=\"browserstack_testing__announcement___bottom\">\n               <svg xmlns=\"http:\/\/www.w3.org\/2000\/svg\" xmlns:xlink=\"http:\/\/www.w3.org\/1999\/xlink\" viewBox=\"0 -1.5 31 33\">\n               <g fill=\"none\" fill-rule=\"evenodd\"> <\/g>\n               <g>\n                  <path d=\"m31.0344828 15.5172414c0 8.5701882-6.9470532 15.5172414-15.5172414 15.5172414-8.56989423 0-15.5172414-6.9470532-15.5172414-15.5172414 0-8.56989423 6.94734717-15.5172414 15.5172414-15.5172414 8.5701882 0 15.5172414 6.94734717 15.5172414 15.5172414\" fill=\"#ecb360\"><\/path>\n                  <path d=\"m26.8965517 13.9655172c0 7.7128847-6.0213369 13.9655173-13.4482758 13.9655173-7.42722227 0-13.4482759-6.2526326-13.4482759-13.9655173 0-7.71288459 6.02105363-13.9655172 13.4482759-13.9655172 7.4269389 0 13.4482758 6.25263261 13.4482758 13.9655172\" fill=\"#d76835\"><\/path>\n                  <path d=\"m27.9310345 12.4136422c0 6.8560652-5.3263006 12.413944-11.8969855 12.413944-6.5698174 0-11.89611797-5.5578788-11.89611797-12.413944 0-6.85576335 5.32630057-12.4136422 11.89611797-12.4136422 6.5706849 0 11.8969855 5.55787885 11.8969855 12.4136422\" fill=\"#d33a41\"><\/path>\n                  <path d=\"m26.8965517 13.4484262c0 6.2843054-4.8632703 11.37916-10.8622124 11.37916-5.9989422 0-10.86192551-5.0948546-10.86192551-11.37916 0-6.28490677 4.86298331-11.37946068 10.86192551-11.37946068 5.9989421 0 10.8622124 5.09455391 10.8622124 11.37946068\" fill=\"#b6cb46\"><\/path>\n                  <path d=\"m25.862069 14.4826111c0 5.7133106-4.6316645 10.3449751-10.3449751 10.3449751-5.71301565 0-10.34468011-4.6316645-10.34468011-10.3449751 0-5.71331065 4.63166446-10.34468007 10.34468011-10.34468007 5.7133106 0 10.3449751 4.63136942 10.3449751 10.34468007\" fill=\"#66ad4a\"><\/path>\n                  <path d=\"m21.7241379 12.4137931c0 4.5706247-3.7052373 8.2758621-8.275862 8.2758621-4.57090515 0-8.27586211-3.7052374-8.27586211-8.2758621 0-4.57062469 3.70495696-8.27586207 8.27586211-8.27586207 4.5706247 0 8.275862 3.70523738 8.275862 8.27586207\" fill=\"#aed7dc\"><\/path>\n                  <path d=\"m22.7586207 11.3793103c0 3.9991764-3.2419755 7.2413794-7.2415259 7.2413794-3.9995505 0-7.24123273-3.242203-7.24123273-7.2413794 0-3.99946945 3.24168223-7.24137927 7.24123273-7.24137927 3.9995504 0 7.2415259 3.24190982 7.2415259 7.24137927\" fill=\"#5bb1cf\"><\/path>\n                  <path d=\"m22.7586207 11.8963967c0 3.7136422-2.7788402 6.724293-6.2068966 6.724293s-6.2068965-3.0106508-6.2068965-6.724293c0-3.71333217 2.7788401-6.72398291 6.2068965-6.72398291s6.2068966 3.01065074 6.2068966 6.72398291\" fill=\"#25a8c3\"><\/path>\n                  <path d=\"m20.6896552 12.9308813c0 3.1423524-2.315869 5.6898084-5.1725531 5.6898084-2.8564055 0-5.1722745-2.547456-5.1722745-5.6898084 0-3.14235241 2.315869-5.68950199 5.1722745-5.68950199 2.8566841 0 5.1725531 2.54714958 5.1725531 5.68950199\"><\/path>\n                  <path d=\"m15.5171021 18.6206897c-2.8564055 0-5.1722745-2.547456-5.1722745-5.6898084 0-3.14235241 2.315869-5.68950199 5.1722745-5.68950199 2.8566841 0 5.1725531 2.54714958 5.1725531 5.68950199 0 3.1423524-2.315869 5.6898084-5.1725531 5.6898084z\" fill=\"#1b1a18\"><\/path>\n                  <path d=\"m18.3395432 11.2904163c-.4920811.8231975-1.4595626 1.299291-2.16129 1.062668-.7017274-.2363383-.8715674-1.0956984-.3798654-1.9191807.4920811-.82348231 1.4595626-1.29929107 2.16129-1.06295279s.8715674 1.09569839.3798654 1.91946549\" fill=\"#fffffe\"><\/path>\n               <\/g>\n               <\/svg>\n               <p>BrowserStack<\/p>\n            <\/div>\n            <a href=\"https:\/\/www.browserstack.com\/\" target=\"blank\" rel=\"noopener\" aria-label=\"\u0421\u0441\u044b\u043b\u043a\u0430 \u043d\u0430 browserstack\">\n            <\/a>\n         <\/div>\n      <\/div>","protected":false},"excerpt":{"rendered":"\u0421\u043e\u0434\u0435\u0440\u0436\u0430\u043d\u0438\u0435: 1. \u0410\u0440\u0445\u0438\u0442\u0435\u043a\u0442\u0443\u0440\u0430, \u0441\u0442\u0440\u043e\u0435\u043d\u0438\u0435 \u0438 \u0441\u0442\u0440\u0443\u043a\u0442\u0443\u0440\u0430 \u043f\u0440\u043e\u0435\u043a\u0442\u0430 2. \u041e\u043f\u0438\u0441\u0430\u043d\u0438\u0435 \u0438 \u0432\u043e\u0437\u043c\u043e\u0436\u043d\u043e\u0441\u0442\u0438 3. \u0412\u0438\u0434\u0435\u043e \u0441 \u0434\u0435\u043c\u043e\u043d\u0441\u0442\u0440\u0430\u0446\u0438\u0435\u0439 \u0432\u043e\u0437\u043c\u043e\u0436\u043d\u043e\u0441\u0442\u0435\u0439 CRYSTAL v2.0 \u0410\u0440\u0445\u0438\u0442\u0435\u043a\u0442\u0443\u0440\u0430: SPA, RESTful API, FSD. \u0421\u0442\u0440\u043e\u0435\u043d\u0438\u0435: Full code | Package.json: frontend &#x2234; backend \u0421\u0442\u0440\u0443\u043a\u0442\u0443\u0440\u0430: MongoDB v8.0.4 Express.js v4.21.2 React v19.0.0 Node.js v24.0.2 NPM v11.3.0 PM2 v5.4.3 Vite v6.1.0 \u041e\u043f\u0438\u0441\u0430\u043d\u0438\u0435 \u0438 \u0432\u043e\u0437\u043c\u043e\u0436\u043d\u043e\u0441\u0442\u0438 \u0424\u0443\u043d\u043a\u0446\u0438\u043e\u043d\u0430\u043b\u044c\u043d\u043e, \u0434\u0430\u043d\u043d\u0430\u044f \u0432\u0435\u0440\u0441\u0438\u044f \u043f\u0440\u0430\u043a\u0442\u0438\u0447\u0435\u0441\u043a\u0438 \u043f\u043e\u043b\u043d\u043e\u0441\u0442\u044c\u044e \u0438\u0434\u0435\u043d\u0442\u0438\u0447\u043d\u0430 [&hellip;]","protected":false},"author":1,"featured_media":4167,"comment_status":"closed","ping_status":"closed","sticky":false,"template":"","format":"standard","meta":{"footnotes":""},"categories":[20,42,31],"tags":[],"class_list":["post-3782","post","type-post","status-publish","format-standard","has-post-thumbnail","hentry","category-crystal","category-crystal-v2-0","category-dokumentaciya-crystal"],"_links":{"self":[{"href":"https:\/\/shedov.top\/ru\/wp-json\/wp\/v2\/posts\/3782","targetHints":{"allow":["GET"]}}],"collection":[{"href":"https:\/\/shedov.top\/ru\/wp-json\/wp\/v2\/posts"}],"about":[{"href":"https:\/\/shedov.top\/ru\/wp-json\/wp\/v2\/types\/post"}],"author":[{"embeddable":true,"href":"https:\/\/shedov.top\/ru\/wp-json\/wp\/v2\/users\/1"}],"replies":[{"embeddable":true,"href":"https:\/\/shedov.top\/ru\/wp-json\/wp\/v2\/comments?post=3782"}],"version-history":[{"count":116,"href":"https:\/\/shedov.top\/ru\/wp-json\/wp\/v2\/posts\/3782\/revisions"}],"predecessor-version":[{"id":4737,"href":"https:\/\/shedov.top\/ru\/wp-json\/wp\/v2\/posts\/3782\/revisions\/4737"}],"wp:featuredmedia":[{"embeddable":true,"href":"https:\/\/shedov.top\/ru\/wp-json\/wp\/v2\/media\/4167"}],"wp:attachment":[{"href":"https:\/\/shedov.top\/ru\/wp-json\/wp\/v2\/media?parent=3782"}],"wp:term":[{"taxonomy":"category","embeddable":true,"href":"https:\/\/shedov.top\/ru\/wp-json\/wp\/v2\/categories?post=3782"},{"taxonomy":"post_tag","embeddable":true,"href":"https:\/\/shedov.top\/ru\/wp-json\/wp\/v2\/tags?post=3782"}],"curies":[{"name":"wp","href":"https:\/\/api.w.org\/{rel}","templated":true}]}}