Server Side Rendering – اختصارا SSR – هي عملية يتم خلالها توليد كود HTML الخاص بتطبيقات جافاسكريبت الحديثة في الخادم عوضا عن المتصفح.
لماذا ؟ وكيف ؟ هذان هما السؤالان اللذان سنجيب عليهما في هذا الدرس
لماذا أصبحنا نتكلم عن Server Side Rendering
في تطبيقات الويب التقليدية، حالما يقوم المستخدم بالدخول إليها عن طريق عنوان URL، يتم إرسال طلب Http إلى الخادم حيث التطبيق مستضاف. بعد ذلك يقوم خادم الويب في تلك الإستضافة (مثلا خادم Python ،PHP أو غيرهما) بتوليد كود HTML الخاص بتلك الصفحة بشكل كامل ويقدمه إلى المتصفح ليقوم بعرضه.
هذه العملية تتكرر مع كل صفحة يتم طلب من طرف المستخدم.
المتصفح كان عمله بسيطا وسهلا مقارنة بحاله اليوم
ولكن في أيامنا هذه، انتشر ما يعرف بتطبيقات الويب أحادية الصفحة، وأصبح معظم العمل يتم على مستوى المتصفح. هذا الأخير يقوم فقط بطلب البيانات من الخادم الذي يلبي الطلب على شكل واجهة برمجية API يتم استهلاكها في المتصفح بطبيعة الحال وبناء عليها يتم توليد وتقديم كود HTML للمستخدم عن طريق جافاسكريبت.
في مدونة توتومينا تحدثنا مرارا عن عدد من أطر عمل جافاسكريبت التي تساعد المطورين على بناء تطبيقات ويب حديثة بأقل مجهود.
React.js واحد من أشهر هذه الأطر وأكثرها استخداما.
في تطبيق مبني على React.js مثلا، عندما نطلب صفحة من الخادم فإن كل ما نحصل عليه هو مايلي :
<!DOCTYPE html> <html lang="en"> <head> <meta charset="utf-8"> <link rel="shortcut icon" href="/favicon.ico"> <title>React Application</title> </head> <body> <div id="root"></div> <script src="/app.js"></script> </body> </html>
1
2
3
4
5
6
7
8
9
10
11
12
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="utf-8">
<link rel="shortcut icon" href="/favicon.ico">
<title>React Application</title>
</head>
<body>
<div id="root"></div>
<script src="/app.js"></script>
</body>
</html>
نلاحظ أنها صفحة تقريبا فارغة، لا تتضمن سوى <div> وأسفل منه ملف جافاسكريبت app.js يتم استدعاؤه.
هكذا تبدو تطبيقات الويب أحادية الصفحة في كل أطر عمل جافاسكريبت الأخر (Vue.js ،Angular إلخ…).
عندما تصل هذه الصفحة إلى المتصفح، يقوم الأخير بواسطة جافاسكريبت الموجود في الملف app.js، بتحديد المكونات (Components) التي يجب توليدها وتركيبها في الصفحة بناء على URL (أو Route) المطلوب. كل هذا العمل يتم في المتصفح (Client) بعد أن تصل إليه الصفحة الفارغة قادمة من الخادم (Server).
الآثار الجانبية
تطبيقات الويب ذات الصفحة الواحدة جميلة وتجعل المستخدم يحسن كأنه في تطبيق مكتبي أصلي، ولكن لكل شيء جميل آثار جانبية لا يجب إهمالها، أو على الأقل يجب أن نكون على دراية بها.
إليكم أهم الآثار الجانبية أو النواقص التي تعاني منها تطبيقات SPA :
تم بناء عدة أطر عمل جافاسكريبت حول أطر عمل الواجهات الأمامية المعروفة لكي تصبح قادرة على العمل في الخوادم وليس في المتصفحات فقط.
مثال عملي ل Server Side Rendering
في هذا الدرس سنرى كيفية تقديم تطبيق ويب (تطبيق React.js) من الخادم، ولن نقوم باستخدام أي من أطر العمل التي سبقت الإشارة إليها، بل سنرى كيفية فعل ذلك من الصفر وباستخدام إطار العمل Express فقط.
سنقوم بإعداد مشروع React.js من الصفر لأن مشاريع create-react-app لا تدعم Server side rendering لذلك سننشئ المشروع بناء على البنية
– server فيه الكود الذي نريد تشغيله على الخادم (Express).
– shared الأكواد المشتركة بين الخادم والمتصفح، وهي في الغالب مكونات React.js.
shared/App.js
في الملف shared/App.js نجد مكون React عادي :
import React from "react"; function App() { return ( <div className="App"> <h1>React SSR Application</h1> </div> ); } export default App;
1
2
3
4
5
6
7
8
9
10
11
12
import React from "react";
function App() {
return (
<div className="App">
<h1>React SSR Application</h1>
</div>
);
}
export default App;
server/index.js
أما الملف server/index.js فيسكون على هيئة كهذه :
import React from "react"; import { renderToString } from "react-dom/server"; import express from "express"; import App from "../shared/App"; const app = express(); app.use(express.static("public")); app.get("*", (req, res) => { const reactDom = renderToString( <App /> ); res.send( htmlTemplate(reactDom) ); }); app.listen(process.env.PORT || 3000, () => { console.log("Server is listening"); }); function htmlTemplate( reactDom ) { return ` <!DOCTYPE html> <head> <title>Isomorphic React Application</title> </head> <body> <div id="root">${reactDom}</div> <script src="/bundle.js"></script> </body> </html> `; }
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
import React from "react";
import { renderToString } from "react-dom/server";
import express from "express";
import App from "../shared/App";
const app = express();
app.use(express.static("public"));
app.get("*", (req, res) => {
const reactDom = renderToString( <App /> );
res.send( htmlTemplate(reactDom) );
});
app.listen(process.env.PORT || 3000, () => {
console.log("Server is listening");
});
function htmlTemplate( reactDom ) {
return `
<!DOCTYPE html>
<head>
<title>Isomorphic React Application</title>
</head>
<body>
<div id="root">${reactDom}</div>
<script src="/bundle.js"></script>
</body>
</html>
`;
}
سأقوم بشرح ما قد يبدو غير مفهوم في هذا الملف.
import { renderToString } from "react-dom/server";
1
import { renderToString } from "react-dom/server";
هنا قمنا باستيراد الوظيفة renderToString من react-dom/server، وتلعب دور render في الخادم. الفرق أنها تقوم بإرجاع ال HTML على شكل String وليس شجرة DOM كما جرت عليه العادة في ناحية المتصفح.
const app = express(); app.use(express.static("public"));
1
2
3
const app = express();
app.use(express.static("public"));
السطر الأول هنا يقوم بإنشاء تطبيق إكسبريس، بينما الثاني يخبر Express بأنه سيجد الملفات الساكنة في المجلد public. الملف الساكن الوحيد في مثالنا هو build.js الذي يتم توليده من طرف Webpack.
app.get("*", (req, res) => { const reactDom = renderToString( <App /> ); res.send( htmlTemplate(reactDom) ); });
1
2
3
4
5
6
7
app.get("*", (req, res) => {
const reactDom = renderToString( <App /> );
res.send( htmlTemplate(reactDom) );
});
هنا نقول لإكسبريس أن يقوم بتحويل المكون <App/> إلى كود Html على شكل String وذلك عن طريق الوظيفة renderToString.
بعد ذلك يتم تقديم هذا Html للمتصفح عن طريق res.send.
browser/index.js
الملف الأخير الذي يخصنا في هذا المثال هو browser/index.js:
import React from "react"; import ReactDOM from "react-dom"; import App from "../shared/App"; ReactDOM.hydrate(<App />, document.getElementById("root"));
1
2
3
4
5
6
import React from "react";
import ReactDOM from "react-dom";
import App from "../shared/App";
ReactDOM.hydrate(<App />, document.getElementById("root"));
نرى بأن هذا الملف بسيط للغاية، فقط قمنا باستيراد المكون <App/> وركبناه في العنصر div#root.
النقطة الوحيدة المثيرة للإنتباه في هذا الملف هي أننا استخدمنا الوظيفة ReactDOM.hydrate عوض ReactDOM.render. وفي الحقيقة الوظيفتان تقومان بنفس الدور، الفرق أن hydrate مصممة خصيصا للحالة التي يتم فيها تقديم كود HTML المكون بواسطة renderToString، يعني حالة Server Side Rendering.
ستجدون الشفرة المصدرية لهذا المثال كاملة (بما فيها ملف webpack.config.js ) على Github في هذا الرابط.
لماذا ؟ وكيف ؟ هذان هما السؤالان اللذان سنجيب عليهما في هذا الدرس
لماذا أصبحنا نتكلم عن Server Side Rendering
في تطبيقات الويب التقليدية، حالما يقوم المستخدم بالدخول إليها عن طريق عنوان URL، يتم إرسال طلب Http إلى الخادم حيث التطبيق مستضاف. بعد ذلك يقوم خادم الويب في تلك الإستضافة (مثلا خادم Python ،PHP أو غيرهما) بتوليد كود HTML الخاص بتلك الصفحة بشكل كامل ويقدمه إلى المتصفح ليقوم بعرضه.
هذه العملية تتكرر مع كل صفحة يتم طلب من طرف المستخدم.
المتصفح كان عمله بسيطا وسهلا مقارنة بحاله اليوم
ولكن في أيامنا هذه، انتشر ما يعرف بتطبيقات الويب أحادية الصفحة، وأصبح معظم العمل يتم على مستوى المتصفح. هذا الأخير يقوم فقط بطلب البيانات من الخادم الذي يلبي الطلب على شكل واجهة برمجية API يتم استهلاكها في المتصفح بطبيعة الحال وبناء عليها يتم توليد وتقديم كود HTML للمستخدم عن طريق جافاسكريبت.
في مدونة توتومينا تحدثنا مرارا عن عدد من أطر عمل جافاسكريبت التي تساعد المطورين على بناء تطبيقات ويب حديثة بأقل مجهود.
React.js واحد من أشهر هذه الأطر وأكثرها استخداما.
في تطبيق مبني على React.js مثلا، عندما نطلب صفحة من الخادم فإن كل ما نحصل عليه هو مايلي :
<!DOCTYPE html> <html lang="en"> <head> <meta charset="utf-8"> <link rel="shortcut icon" href="/favicon.ico"> <title>React Application</title> </head> <body> <div id="root"></div> <script src="/app.js"></script> </body> </html>
1
2
3
4
5
6
7
8
9
10
11
12
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="utf-8">
<link rel="shortcut icon" href="/favicon.ico">
<title>React Application</title>
</head>
<body>
<div id="root"></div>
<script src="/app.js"></script>
</body>
</html>
نلاحظ أنها صفحة تقريبا فارغة، لا تتضمن سوى <div> وأسفل منه ملف جافاسكريبت app.js يتم استدعاؤه.
هكذا تبدو تطبيقات الويب أحادية الصفحة في كل أطر عمل جافاسكريبت الأخر (Vue.js ،Angular إلخ…).
عندما تصل هذه الصفحة إلى المتصفح، يقوم الأخير بواسطة جافاسكريبت الموجود في الملف app.js، بتحديد المكونات (Components) التي يجب توليدها وتركيبها في الصفحة بناء على URL (أو Route) المطلوب. كل هذا العمل يتم في المتصفح (Client) بعد أن تصل إليه الصفحة الفارغة قادمة من الخادم (Server).
الآثار الجانبية
تطبيقات الويب ذات الصفحة الواحدة جميلة وتجعل المستخدم يحسن كأنه في تطبيق مكتبي أصلي، ولكن لكل شيء جميل آثار جانبية لا يجب إهمالها، أو على الأقل يجب أن نكون على دراية بها.
إليكم أهم الآثار الجانبية أو النواقص التي تعاني منها تطبيقات SPA :
- تحميل الموقع يكون ثقيلا خاصة عندما يكون الإنترنت بطيئا، ويعاني من هذا البطء في العادة مستخدمو الأجهزة المحمولة من الدول النامية، حيث اتصال الإنترنت سواء من الجيلين الثالث والرابع يكون دون المستوى الإعتيادي.
بطء التحميل ناتج عن ملف الجافاسكريبت الذي يكون كبيرا، فحتى بعد تحميله يستغرق المتصفح وقتا لا يستهان به في قراءته من أجل تجهيز وعرض الصفحة للمستخدم، ويكون الأخيرا قادرا على التفاعل معها.
يمكن تسريع هذه العملية عبر اتباع طرق مثل تقسيم كود الجافاسكريبت وتحميل الأكواد التي نحتاجها في الصفحة فقط باستخدام ال Code Splitting و Lazy Loading، ولكن تطبيق هذه الأفكار لا يكون دائما عملية سهلة بالنسبة لمطوري الويب الذين لا يتمتعون بمستوى معين من الخبرة. - محركات تقوم بأرشفة صفحات الويب عن طريق ما يعرف بالعناكب (Crawlers)، هذه العناكب عندما تدخل لصفحة SPA تجدها فارغة كما رأينا أعلاه، وبالتالي لا تصل إلى محتوى الصفحة لأنها لا تقرأ الجافاسكريبت مثلما تفعل المتصفحات.
في السنوات الأخيرة أصبحت تلك العناكب أكثر ذكاءً وتطورا، ومحرك Google أعلن فيما سبق أنه يستطيع أرشفة وقراءة الصفحات التي يتم توليدها عبر جافاسكريبت. ولكن رغم هذا كله ما يزال هذا النوع من التطبيقات يعاني من إشكاليات أخرى مع SEO، ما يؤثر بشكل سلبي على ظهورها على محركات البحث. - نفس ما قيل عن محركات البحث يمكن أن يقال على منصات التواصل الإجتماعي، حيث إن الصفحات تصبح مبدئيا غير قابلة للمشاركة بشكل جيد، لأن موقع التواصل (فيسبوك مثلا) لا يستطيع الوصول إلى معلومات meta التي تظهر للمستخدين عند مشاركة الرابط (مثلا عنوان الصفحة، وصف الصفحة، الصورة إلخ…)
تم بناء عدة أطر عمل جافاسكريبت حول أطر عمل الواجهات الأمامية المعروفة لكي تصبح قادرة على العمل في الخوادم وليس في المتصفحات فقط.
- Next.js تم بناؤه حول React.js.
- Nuxt.js تم بناؤه حول Vue.js.
- Angular Universal حول إطار العمل Angular.. هذا الإطار مدعوم رسميا من فريق Angular.
مثال عملي ل Server Side Rendering
في هذا الدرس سنرى كيفية تقديم تطبيق ويب (تطبيق React.js) من الخادم، ولن نقوم باستخدام أي من أطر العمل التي سبقت الإشارة إليها، بل سنرى كيفية فعل ذلك من الصفر وباستخدام إطار العمل Express فقط.
سنقوم بإعداد مشروع React.js من الصفر لأن مشاريع create-react-app لا تدعم Server side rendering لذلك سننشئ المشروع بناء على البنية
- المجلد public: سنقوم بإنشائه ونتركه فارغا، وسيتم توليد الملفات بداخله (مثلا bundle.js) بشكل آلي، بناء على إعدادات webpack.
يمكن كذلك أن يحتوي على الصور وملفات Css إلخ… - المجلد src: هذا هو المجلد الذي سيضم كافة أكوادنا البرمجية الخاصة. سنقوم بتقسيمها إلى ثلاثة أجزاء : server ،browser و shared.
– server فيه الكود الذي نريد تشغيله على الخادم (Express).
– shared الأكواد المشتركة بين الخادم والمتصفح، وهي في الغالب مكونات React.js.
shared/App.js
في الملف shared/App.js نجد مكون React عادي :
import React from "react"; function App() { return ( <div className="App"> <h1>React SSR Application</h1> </div> ); } export default App;
1
2
3
4
5
6
7
8
9
10
11
12
import React from "react";
function App() {
return (
<div className="App">
<h1>React SSR Application</h1>
</div>
);
}
export default App;
server/index.js
أما الملف server/index.js فيسكون على هيئة كهذه :
import React from "react"; import { renderToString } from "react-dom/server"; import express from "express"; import App from "../shared/App"; const app = express(); app.use(express.static("public")); app.get("*", (req, res) => { const reactDom = renderToString( <App /> ); res.send( htmlTemplate(reactDom) ); }); app.listen(process.env.PORT || 3000, () => { console.log("Server is listening"); }); function htmlTemplate( reactDom ) { return ` <!DOCTYPE html> <head> <title>Isomorphic React Application</title> </head> <body> <div id="root">${reactDom}</div> <script src="/bundle.js"></script> </body> </html> `; }
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
import React from "react";
import { renderToString } from "react-dom/server";
import express from "express";
import App from "../shared/App";
const app = express();
app.use(express.static("public"));
app.get("*", (req, res) => {
const reactDom = renderToString( <App /> );
res.send( htmlTemplate(reactDom) );
});
app.listen(process.env.PORT || 3000, () => {
console.log("Server is listening");
});
function htmlTemplate( reactDom ) {
return `
<!DOCTYPE html>
<head>
<title>Isomorphic React Application</title>
</head>
<body>
<div id="root">${reactDom}</div>
<script src="/bundle.js"></script>
</body>
</html>
`;
}
سأقوم بشرح ما قد يبدو غير مفهوم في هذا الملف.
import { renderToString } from "react-dom/server";
1
import { renderToString } from "react-dom/server";
هنا قمنا باستيراد الوظيفة renderToString من react-dom/server، وتلعب دور render في الخادم. الفرق أنها تقوم بإرجاع ال HTML على شكل String وليس شجرة DOM كما جرت عليه العادة في ناحية المتصفح.
const app = express(); app.use(express.static("public"));
1
2
3
const app = express();
app.use(express.static("public"));
السطر الأول هنا يقوم بإنشاء تطبيق إكسبريس، بينما الثاني يخبر Express بأنه سيجد الملفات الساكنة في المجلد public. الملف الساكن الوحيد في مثالنا هو build.js الذي يتم توليده من طرف Webpack.
app.get("*", (req, res) => { const reactDom = renderToString( <App /> ); res.send( htmlTemplate(reactDom) ); });
1
2
3
4
5
6
7
app.get("*", (req, res) => {
const reactDom = renderToString( <App /> );
res.send( htmlTemplate(reactDom) );
});
هنا نقول لإكسبريس أن يقوم بتحويل المكون <App/> إلى كود Html على شكل String وذلك عن طريق الوظيفة renderToString.
بعد ذلك يتم تقديم هذا Html للمتصفح عن طريق res.send.
browser/index.js
الملف الأخير الذي يخصنا في هذا المثال هو browser/index.js:
import React from "react"; import ReactDOM from "react-dom"; import App from "../shared/App"; ReactDOM.hydrate(<App />, document.getElementById("root"));
1
2
3
4
5
6
import React from "react";
import ReactDOM from "react-dom";
import App from "../shared/App";
ReactDOM.hydrate(<App />, document.getElementById("root"));
نرى بأن هذا الملف بسيط للغاية، فقط قمنا باستيراد المكون <App/> وركبناه في العنصر div#root.
النقطة الوحيدة المثيرة للإنتباه في هذا الملف هي أننا استخدمنا الوظيفة ReactDOM.hydrate عوض ReactDOM.render. وفي الحقيقة الوظيفتان تقومان بنفس الدور، الفرق أن hydrate مصممة خصيصا للحالة التي يتم فيها تقديم كود HTML المكون بواسطة renderToString، يعني حالة Server Side Rendering.
- هذا الرابط يعطي معلومات وتفاصيل أكثر عن ReactDOM.hydrate.
ستجدون الشفرة المصدرية لهذا المثال كاملة (بما فيها ملف webpack.config.js ) على Github في هذا الرابط.