الفرق بين المراجعتين لصفحة: «Node.js/stream»
رهف-النجار (نقاش | مساهمات) القسم الأخير من الصفحة |
رهف-النجار (نقاش | مساهمات) ط تنسيق |
||
(15 مراجعات متوسطة بواسطة نفس المستخدم غير معروضة) | |||
سطر 1: | سطر 1: | ||
<noinclude>{{DISPLAYTITLE:الوحدة Stream}}</noinclude> | |||
الاستقرار: 2-مستقر | الاستقرار: 2-مستقر | ||
المجرى هو واجهة مجرّدة للعمل مع البيانات المتدفقة في Node.js. توفّر الوحدة stream واجهة برمجية (API) أساسية تجعل من السهل بناء كائنات تتعامل مع واجهة المجرى. | المجرى هو واجهة مجرّدة للعمل مع البيانات المتدفقة في Node.js. توفّر الوحدة <code>stream</code> واجهة برمجية (API) أساسية تجعل من السهل بناء كائنات تتعامل مع واجهة المجرى. | ||
يوجد العديد من كائنات المجرى التي توفرها Node.js. على سبيل المثال، http.IncomingMessage (طلبيات الخادم HTTP) و process.stdout هما نسخ من الصنف stream. | يوجد العديد من كائنات المجرى التي توفرها Node.js. على سبيل المثال، [[Node.js/http#.D8.A7.D9.84.D8.B5.D9.86.D9.81 http.IncomingMessage|http.IncomingMessage (طلبيات الخادم HTTP)]] و [[Node.js/process#process.stdout|process.stdout]] هما نسخ من الصنف stream. | ||
يمكن أن تكون المجاري قابلة للقراءة (readable)، أو قابلة للكتابة (writable)، أو كليهما. كل المجاري هي نسخ من الصنف EventEmitter. | يمكن أن تكون المجاري قابلة للقراءة (readable)، أو قابلة للكتابة (writable)، أو كليهما. كل المجاري هي نسخ من الصنف [[Node.js/events#.D8.A7.D9.84.D8.B5.D9.86.D9.81: EventEmitter|EventEmitter]]. | ||
يمكن الوصول إلى الوحدة stream باستخدام:<syntaxhighlight lang="javascript"> | يمكن الوصول إلى الوحدة <code>stream</code> باستخدام:<syntaxhighlight lang="javascript"> | ||
const stream = require('stream'); | const stream = require('stream'); | ||
</syntaxhighlight>لمَّا كان من الضروري فهم كيفية عمل المجاري، فإنَّ الوحدة stream بحد ذاتها هي أكثر فائدةً للمطورين الذين ينشئون أنواعًا جديدةً من نسخ المجاري. المطورون الذين يستهلكون كائنات المجاري بالمقام الأول نادرًا ما يحتاجون لاستخدام الوحدة stream بشكل مباشر. | </syntaxhighlight>لمَّا كان من الضروري فهم كيفية عمل المجاري، فإنَّ الوحدة <code>stream</code> بحد ذاتها هي أكثر فائدةً للمطورين الذين ينشئون أنواعًا جديدةً من نسخ المجاري. المطورون الذين يستهلكون كائنات المجاري بالمقام الأول نادرًا ما يحتاجون لاستخدام الوحدة <code>stream</code> بشكل مباشر. | ||
= تنظيم هذا المستند = | == تنظيم هذا المستند == | ||
يُقسّم هذا التوثيق إلى قسمين رئيسيين مع قسم ثالث للملاحظات الإضافية. يشرح القسم الأول عناصر الواجهة البرمجية للمجرى (stream API) التي تكون مطلوبة لاستخدام المجاري ضمن أي تطبيق. يشرح القسم الثاني عناصر الواجهة البرمجية (API) التي تُكون مطلوبةً لتطبيق أنواع جديدة من المجاري. | يُقسّم هذا التوثيق إلى قسمين رئيسيين مع قسم ثالث للملاحظات الإضافية. يشرح القسم الأول عناصر الواجهة البرمجية للمجرى (stream API) التي تكون مطلوبة لاستخدام المجاري ضمن أي تطبيق. يشرح القسم الثاني عناصر الواجهة البرمجية (API) التي تُكون مطلوبةً لتطبيق أنواع جديدة من المجاري. | ||
== أنواع المجاري == | == أنواع المجاري == | ||
يوجد أربعة أنواع رئيسية للمجاري ضمن Node.js هي: | يوجد أربعة أنواع رئيسية للمجاري ضمن Node.js هي: | ||
* Writable: المجاري التي يمكن أن تُكتب عليها البيانات (مثل fs.createWriteStream())، و | * [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Writable|Writable]]: المجاري التي يمكن أن تُكتب عليها البيانات (مثل [[Node.js/fs#fs.createWriteStream.28path.5B.2C options.5D.29.E2.80.8E|fs.createWriteStream()]])، و | ||
* Readable: المجاري التي يمكن أن تُقرأ منها البيانات. (مثل fs.createReadStream())، و | * [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Readable|Readable]]: المجاري التي يمكن أن تُقرأ منها البيانات. (مثل [[Node.js/fs#fs.createReadStream.28path.5B.2C options.5D.29.E2.80.8E|fs.createReadStream()]])، و | ||
* Duplex: المجاري التي تجمع بين النوع Readable والنوع Writable (مثل net.Socket) | * [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Duplex|Duplex]]: المجاري التي تجمع بين النوع <code>Readable</code> والنوع <code>Writable</code> (مثل [[Node.js/net#.D8.A7.D9.84.D8.B5.D9.86.D9.81 net.Socket|net.Socket]]) | ||
* Transform: المجاري Duplex التي يمكن أن تعدّل أو تحوّل البيانات حسبما تُكتب أو تُقرأ (مثل zlib.createDeflate()). | * [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81: stream.Transform|Transform]]: المجاري <code>Duplex</code> التي يمكن أن تعدّل أو تحوّل البيانات حسبما تُكتب أو تُقرأ (مثل [[Node.js/zlib#zlib.createDeflate.28.5Boptions.5D.29.E2.80.8E|zlib.createDeflate()]]). | ||
بالإضافة إلى ذلك، تحوي هذه الوحدة الدوال الخدمية pipeline و finished. | بالإضافة إلى ذلك، تحوي هذه الوحدة الدوال الخدمية pipeline و finished. | ||
=== نمط الكائن === | === نمط الكائن === | ||
كل المجاري المُنشأة من قبل واجهات Node.js تعمل حصريًا على السلاسل النصية والكائنات Buffer (أو Uint8Array). مع ذلك، فمن الممكن لتطبيقات المجاري أن تعمل مع أنواع بيانات أخرى في JavaScript (باستثناء النوع | كل المجاري المُنشأة من قبل واجهات Node.js تعمل حصريًا على السلاسل النصية والكائنات <code>Buffer</code> (أو <code>Uint8Array</code>). مع ذلك، فمن الممكن لتطبيقات المجاري أن تعمل مع أنواع بيانات أخرى في JavaScript (باستثناء النوع <code>null</code>، الذي يخدم غرضًا خاصًا ضمن المجاري). مثل هذه المجاري يؤخذ بالحسبان تشغيلها في "نمط الكائن" (object mode). | ||
تُبدَّل نُسَخ المجاري إلى نمط الكائن باستخدام الخيار objectMode عند إنشاء المجرى. محاولة قلب مجرى موجود إلى نمط الكائن ليست آمنةً. | تُبدَّل نُسَخ المجاري إلى نمط الكائن باستخدام الخيار <code>objectMode</code> عند إنشاء المجرى. محاولة قلب مجرى موجود إلى نمط الكائن ليست آمنةً. | ||
=== Buffering === | === التخزين المؤقت Buffering === | ||
سوف يخزن كلا المجريين Writable و Readable البيانات في مخزن مؤقت داخلي لكي يمكن استعادتها باستخدام التابع writable.writableBuffer أو readable.readableBuffer على التوالي. | سوف يخزن كلا المجريين [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Writable|Writable]] و [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Readable|Readable]] البيانات في مخزن مؤقت داخلي لكي يمكن استعادتها باستخدام التابع <code>writable.writableBuffer</code> أو <code>readable.readableBuffer</code> على التوالي. | ||
كمية البيانات القابلة للتخزين تعتمد على الخيار highWaterMark المُمرر إلى باني المجرى. من أجل المجاري القياسية، يحدد الخيار highWaterMark العدد الكلي من البايتات. أمَّا من أجل المجاري المُشغلة في نمط الكائن، فيحدد الخيار highWaterMark العدد الكلي من الكائنات. | كمية البيانات القابلة للتخزين تعتمد على الخيار <code>highWaterMark</code> المُمرر إلى باني المجرى. من أجل المجاري القياسية، يحدد الخيار <code>highWaterMark</code> [[Node.js/stream#.D8.A7.D8.AE.D8.AA.D9.84.D8.A7.D9.81 highWaterMark .D8.A8.D8.B9.D8.AF .D8.A7.D8.B3.D8.AA.D8.AF.D8.B9.D8.A7.D8.A1 readable.setEncoding.28.29.E2.80.8E|العدد الكلي من البايتات]]. أمَّا من أجل المجاري المُشغلة في نمط الكائن، فيحدد الخيار <code>highWaterMark</code> العدد الكلي من الكائنات. | ||
تُخزّن البيانات مؤقتًا في المجاري التي من النوع Readable عندما يستدعي التابع stream.push(chunk). إذا لم يستدعي من يستخدم المجرى التابع | تُخزّن البيانات مؤقتًا في المجاري التي من النوع <code>Readable</code> عندما يستدعي التابع [[Node.js/stream#readable.push.28chunk.5B.2C encoding.5D.E2.80.8E.29.E2.80.8E|stream.push(chunk)]]. إذا لم يستدعي من يستخدم المجرى التابع [[Node.js/stream#readable.read.28.5Bsize.5D.E2.80.8E.29.E2.80.8E|stream.read()]]، فستتوضع البيانات ضمن طابور داخلي إلى أن تُستخدَم. | ||
حالما يصل الحجم الكلي لذاكرة القراءة الداخلية المؤقتة عتبةَ محددة من قبل | حالما يصل الحجم الكلي لذاكرة القراءة الداخلية المؤقتة عتبةَ محددة من قبل <code>highWaterMark</code>، سيوقفُ المجرى مؤقتًا قراءة البيانات من المصادر الأساسية حتى تُستهلك البيانات الحالية المخزنة مؤقتًا (ذلك أنّ المجرى سيوقف استدعاء التابع <code>readable._read()</code> المحلي الذي يُستخدم لوضع البيانات في ذاكرة القراءة المؤقتة). | ||
تُخزّن البيانات مؤقتًا في المجاري التي من النوع Writable عندما يُستدعى التابع writable.write(chunk) مرارًا وتكرارًا. طالما أنّ الحجم الكلي لذاكرة الكتابة المؤقتة الداخلية هو أدنى من العتبة المضبوطة بمقدار | تُخزّن البيانات مؤقتًا في المجاري التي من النوع <code>Writable</code> عندما يُستدعى التابع [[يسبئتلئكتلبهخئسهت|writable.write(chunk)]] مرارًا وتكرارًا. طالما أنّ الحجم الكلي لذاكرة الكتابة المؤقتة الداخلية هو أدنى من العتبة المضبوطة بمقدار <code>highWaterMark</code>، سوف يعيد استدعاء <code>writable.write()</code> القيمة <code>true</code>. حالما يصل أو يتجاوز حجم ذاكرة التخزين المؤقت الداخلية القيمة <code>highWaterMark</code>، سوف تُعاد القيمة <code>false</code>. | ||
الهدف الأساسي من واجهات الوحدة stream البرمجية - على وجه الخصوص التابع stream.pipe() - هو تقييد حجم التخزين المؤقت للبيانات إلى مستويات مقبولة من أجل التوافق بين المصادر (source) والوجهات (destination) ذات السرعات المختلفة لكي لا تمتلئ الذاكرة المتوافرة. | الهدف الأساسي من واجهات الوحدة <code>stream</code> البرمجية - على وجه الخصوص التابع [[Node.js/stream#readable.pipe.28destination.5B.2C options.5D.E2.80.8E.29.E2.80.8E|stream.pipe()]] - هو تقييد حجم التخزين المؤقت للبيانات إلى مستويات مقبولة من أجل التوافق بين المصادر (source) والوجهات (destination) ذات السرعات المختلفة لكي لا تمتلئ الذاكرة المتوافرة. | ||
بما أن كلا المجريين Duplex و Transform يجمعان بين النوعين Readable و | بما أن كلا المجريين [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Duplex|Duplex]] و [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81: stream.Transform|Transform]] يجمعان بين النوعين <code>Readable</code> و <code>Writable</code>، يمتلك كل واحد منها ذاكرتي تخزين مؤقتة. هاتان الذاكراتان داخليتين ومنفصلتين عن بعضهما وتستخدمان للقراءة والكتابة يف آن واحد، مما يسمح لكل طرف من المجرى بالتشغيل بشكل مستقل عن الآخر بينما يضمن تدفق بيانات مناسب وفعّال. على سبيل المثال، النُسخ [[Node.js/net#.D8.A7.D9.84.D8.B5.D9.86.D9.81 net.Socket|net.Socket]] هي مجاري من النوع [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Duplex|Duplex]] طرفها القابل للقراءة (<code>Readable</code>) يسمح باستهلاك البيانات المُستقبَلة من المقبس وطرفها القابل للكتابة (<code>Writable</code>) يسمح بكتابة البيانات على المقبس. بسبب أن البيانات قد تُكتب إلى المقبس بمعدل أسرع أو أبطأ من البيانات التي تُستقبل وتكتب على المجرى، فمن الضروري لكل طرف أن يعمل (ويخزِّن البيانات) بشكل مستقل عن الآخر. | ||
== الواجهات البرمجية لمستخدمي المجرى == | == الواجهات البرمجية لمستخدمي المجرى == | ||
سطر 50: | سطر 51: | ||
// والذي هو مجرى قابل للقراءة ،http.IncomingMessage هو Req | |||
//والذي هو مجرى قابل للكتابة http.ServerResponse هو Req | |||
سطر 57: | سطر 58: | ||
احصل على البيانات كسلاسل | //utf8 احصل على البيانات كسلاسل | ||
إذا لم يُجهّز الترميز ستُستقبل كائنات ذاكرة مؤقتة | //إذا لم يُجهّز الترميز ستُستقبل كائنات ذاكرة مؤقتة | ||
req.setEncoding('utf8'); | req.setEncoding('utf8'); | ||
سطر 64: | سطر 65: | ||
تطلق المجاري القابلة للقراءة أحداث | //حالما يُضاف مستمع 'data'تطلق المجاري القابلة للقراءة أحداث | ||
req.on('data', (chunk) => { | req.on('data', (chunk) => { | ||
سطر 74: | سطر 75: | ||
try { | try { | ||
const data = JSON.parse(body); | const data = JSON.parse(body); | ||
// | // اكتب شيئًا يثير اهتمام المستخدم | ||
res.write(typeof data); | res.write(typeof data); | ||
res.end(); | res.end(); | ||
سطر 81: | سطر 82: | ||
// !سيئة json ،أوه | |||
res.statusCode = 400; | res.statusCode = 400; | ||
return res.end(`error: ${er.message}`); | return res.end(`error: ${er.message}`); | ||
سطر 96: | سطر 97: | ||
// $ curl localhost:1337 -d "not json" | // $ curl localhost:1337 -d "not json" | ||
// error: Unexpected token o in JSON at position 1 | // error: Unexpected token o in JSON at position 1 | ||
</syntaxhighlight>المجاري ذات النوع Writable (مثل res في المثال) توفر توابع مثل write() و end() والتي تُستخدم لكتابة البيانات على المجرى. | </syntaxhighlight>المجاري ذات النوع [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Writable|Writable]] (مثل <code>res</code> في المثال) توفر توابع مثل <code>write()</code> و <code>end()</code> والتي تُستخدم لكتابة البيانات على المجرى. | ||
المجاري ذات النوع Readable تستخدم واجهات الصنف EventEmitter البرمجية من أجل اشعار شيفرة التطبيق عندما تكون البيانات متوفرة للقراءة من المجرى. يمكن قراءة هذه البيانات المتوفرة من المجرى بعدة طرق. | المجاري ذات النوع [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Readable|Readable]] تستخدم واجهات الصنف [[Node.js/events#.D8.A7.D9.84.D8.B5.D9.86.D9.81: EventEmitter|EventEmitter]] البرمجية من أجل اشعار شيفرة التطبيق عندما تكون البيانات متوفرة للقراءة من المجرى. يمكن قراءة هذه البيانات المتوفرة من المجرى بعدة طرق. | ||
تستخدم المجاري التي من النوع Writable و Readable واجهات الصنف EventEmitter البرمجية بطرق متنوعة للبقاء على اتصال دائم بالحالة الحالية للمجرى. | تستخدم المجاري التي من النوع [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Writable|Writable]] و [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Readable|Readable]] واجهات الصنف [[Node.js/events#.D8.A7.D9.84.D8.B5.D9.86.D9.81: EventEmitter|EventEmitter]] البرمجية بطرق متنوعة للبقاء على اتصال دائم بالحالة الحالية للمجرى. | ||
تذكر أن المجاري التي من النوع Duplex و Transform هي مجاري قابلة للكتاية (Writable) والقراءة (Readable). | تذكر أن المجاري التي من النوع [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Duplex|Duplex]] و [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81: stream.Transform|Transform]] هي مجاري قابلة للكتاية ([[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Writable|Writable]]) والقراءة ([[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Readable|Readable]]). | ||
لا يُطلب من التطبيقات التي إمّا تكتب البيانات أو تقرؤها من المجرى تنفيذ واجهات الوحدة stream بشكل مباشر، وبذلك لا يوجد عمومًا سبب لاستدعاء الوحدة stream عبر require('stream'). | لا يُطلب من التطبيقات التي إمّا تكتب البيانات أو تقرؤها من المجرى تنفيذ واجهات الوحدة stream بشكل مباشر، وبذلك لا يوجد عمومًا سبب لاستدعاء الوحدة stream عبر <code>require('stream')</code>. | ||
ينبغي على المطورين الراغبين بإنشاء أنواع جديدة من المجاري الرجوع إلى القسم الواجهات البرمجية لمنفذي المجاري. | ينبغي على المطورين الراغبين بإنشاء أنواع جديدة من المجاري الرجوع إلى القسم [[Node.js/stream#.D8.A7.D9.84.D9.88.D8.A7.D8.AC.D9.87.D8.A7.D8.AA .D8.A7.D9.84.D8.A8.D8.B1.D9.85.D8.AC.D9.8A.D8.A9 .D9.84.D9.85.D9.86.D9.81.D8.B0.D9.8A .D8.A7.D9.84.D9.85.D8.AC.D8.A7.D8.B1.D9.8A|الواجهات البرمجية لمنفذي المجاري]]. | ||
=== المجاري القابلة للكتابة === | === المجاري القابلة للكتابة === | ||
المجاري القابلة للكتابة إجمالًا تمثِّل مكانًا قابلًا لكتابة البيانات فيه. | المجاري القابلة للكتابة إجمالًا تمثِّل مكانًا قابلًا لكتابة البيانات فيه. | ||
تتضمن الأمثلة التالية مجارٍ قابلة للكتابة: | تتضمن الأمثلة التالية مجارٍ [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Writable|قابلة للكتابة]]: | ||
* طلب HTTP، من طرف العميل. | * [[Node.js/http#.D8.A7.D9.84.D8.B5.D9.86.D9.81 http.ClientRequest|طلب HTTP، من طرف العميل]]. | ||
* استجابة HTTP، من طرف الخادم. | * [[Node.js/http#.D8.A7.D9.84.D8.B5.D9.86.D9.81 http.ServerResponse|استجابة HTTP، من طرف الخادم]]. | ||
* مجاري الوحدة fs القابلة للكتابة. | * [[Node.js/fs#.D8.A7.D9.84.D8.B5.D9.86.D9.81 fs.WriteStream.E2.80.8E|مجاري الوحدة fs القابلة للكتابة]]. | ||
* مجاري الوحدة zlib. | * [[Node.js/zlib|مجاري الوحدة zlib]]. | ||
* مجاري الوحدة crypto. | * [[Node.js#.D8.A7.D9.84.D9.88.D8.AD.D8.AF.D8.A9 Crypto|مجاري الوحدة crypto]]. | ||
* المقابس TCP. | * [[Node.js/net#.D8.A7.D9.84.D8.B5.D9.86.D9.81 net.Socket|المقابس TCP]]. | ||
* العملية الابن للمجرى stdin (هي subprocess.stdin). | * العملية الابن للمجرى stdin (هي [[Node.js/child process#.D8.A7.D9.84.D8.AE.D8.A7.D8.B5.D9.8A.D8.A9 subprocess.stdin|subprocess.stdin]]). | ||
* المجرى process.stdout و process.stderr. | * المجرى [[Node.js/process#process.stdout|process.stdout]] و [[Node.js/process#process.stderr|process.stderr]]. | ||
بعض هذه الأمثلة هي فعليًا مجاري من النوع Duplex والتي تنفّذ الواجهة Writable. | بعض هذه الأمثلة هي فعليًا مجاري من النوع [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Duplex|Duplex]] والتي تنفّذ الواجهة [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Writable|Writable]]. | ||
كل المجاري التي من النوع Writable تنفّذ الواجهة المعرّفة بالصنف stream.Writable. | كل المجاري التي من النوع [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Writable|Writable]] تنفّذ الواجهة المعرّفة بالصنف <code>stream.Writable</code>. | ||
بينما قد تختلف نسخ المجاري التي من النوع Writable بطرق متنوعة، كل المجاري Writable تتبع نمط الاستخدام الأساسي كما هو موضّح في المثال أدناه:<syntaxhighlight lang="javascript"> | بينما قد تختلف نسخ المجاري التي من النوع [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Writable|Writable]] بطرق متنوعة، كل المجاري <code>Writable</code> تتبع نمط الاستخدام الأساسي كما هو موضّح في المثال أدناه:<syntaxhighlight lang="javascript"> | ||
const myStream = getWritableStreamSomehow(); | const myStream = getWritableStreamSomehow(); | ||
myStream.write('some data'); | myStream.write('some data'); | ||
سطر 131: | سطر 132: | ||
</syntaxhighlight> | </syntaxhighlight> | ||
==== الصنف stream.Writable ==== | ==== الصنف <code>stream.Writable</code> ==== | ||
أضيف في الإصدار: 0.9.4. | أضيف في الإصدار: 0.9.4. | ||
===== الحدث 'close' ===== | ===== الحدث <code>'close'</code> ===== | ||
يُطلَق الحدث 'close' عندما يكون المجرى أو أحد موارده الأساسية (واصف الملفات مثلًا) قد أُغلق. يشير هذا الحدث أنّه لن تُطلق المزيد من الأحداث، ولن تحصل المزيد من العمليات المتعلقة بالمجرى. | يُطلَق الحدث <code>'close'</code> عندما يكون المجرى أو أحد موارده الأساسية (واصف الملفات مثلًا) قد أُغلق. يشير هذا الحدث أنّه لن تُطلق المزيد من الأحداث، ولن تحصل المزيد من العمليات المتعلقة بالمجرى. | ||
لا تُطلق كل المجاري Writable الحدث 'close'. | لا تُطلق كل المجاري <code>Writable</code> الحدث <code>'close'</code>. | ||
===== الحدث 'drain' ===== | ===== الحدث <code>'drain'</code> ===== | ||
أُضيف في الإصدار:0.9.4. | أُضيف في الإصدار:0.9.4. | ||
إذا أعاد التابع stream.write(chunk) القيمة | إذا أعاد التابع [[ميسنبت\سمبيتسئم|stream.write(chunk)]] القيمة <code>false</code>، فسوف يُطلَق الحدث <code>'drain'</code> عندما يكون من المناسب استئناف كتابة البيانات على المجرى.<syntaxhighlight lang="javascript"> | ||
كتابة البيانات إلى المجرى القابل للكتابة المزوّد لمليون مرّة | //كتابة البيانات إلى المجرى القابل للكتابة المزوّد لمليون مرّة | ||
كن منتبهًا للضغط العائد | //كن منتبهًا للضغط العائد | ||
سطر 156: | سطر 157: | ||
if (i === 0) { | if (i === 0) { | ||
آخر مرّة | //آخر مرّة | ||
writer.write(data, encoding, callback); | writer.write(data, encoding, callback); | ||
سطر 163: | سطر 164: | ||
انظر إذا كان ينبغي لنا الإستمرار أو الإنتظار// | //انظر إذا كان ينبغي لنا الإستمرار أو الإنتظار | ||
لا تمرر دالة رد النداء لأنّنا لم ننته بعد | //لا تمرر دالة رد النداء لأنّنا لم ننته بعد | ||
ok = writer.write(data, encoding); | ok = writer.write(data, encoding); | ||
سطر 171: | سطر 172: | ||
if (i > 0) { | if (i > 0) { | ||
وجب التوقف باكرًا// | //وجب التوقف باكرًا | ||
اكتب بعض المزيد حالما تنضب | //اكتب بعض المزيد حالما تنضب | ||
writer.once('drain', write); | writer.once('drain', write); | ||
سطر 181: | سطر 182: | ||
</syntaxhighlight> | </syntaxhighlight> | ||
===== الحدث 'error' ===== | ===== الحدث <code>'error'</code> ===== | ||
أضيف في الإصدار: 0.9.4. | أضيف في الإصدار: 0.9.4. | ||
* <Error> | * [[JavaScript/Error|<Error>]] | ||
يُطلق الحدث 'error' إذا حصل خطأ أثناء الكتابة على أو إرسال البيانات إلى المجرى. سوف يُمرَّر إلى دالة رد النداء المنصتة الوسيط Error فقط عند استدعائها. | يُطلق الحدث <code>'error'</code> إذا حصل خطأ أثناء الكتابة على أو إرسال البيانات إلى المجرى. سوف يُمرَّر إلى دالة رد النداء المنصتة الوسيط <code>Error</code> فقط عند استدعائها. | ||
لن يُغلق المجرى عندما يُطلق الحدث 'error'. | لن يُغلق المجرى عندما يُطلق الحدث <code>'error'</code>. | ||
===== الحدث 'finish' ===== | ===== الحدث <code>'finish'</code> ===== | ||
أضيف في الإصدار: 0.9.4. | أضيف في الإصدار: 0.9.4. | ||
سوف يُطلق الحدث 'finish' بعد استدعاء التابع | سوف يُطلق الحدث <code>'finish'</code> بعد استدعاء التابع <nowiki/>[[Node.js/stream#writable.end.28.5Bchunk.5D.5B.2C encoding.5D.5B.2C callback.5D.E2.80.8E.29.E2.80.8E|stream.end()]]، وبعد أن دُفعت كل البيانات للنظام الأساسي.<syntaxhighlight lang="javascript"> | ||
const writer = getWritableStreamSomehow(); | const writer = getWritableStreamSomehow(); | ||
for (let i = 0; i < 100; i++) { | for (let i = 0; i < 100; i++) { | ||
سطر 202: | سطر 203: | ||
</syntaxhighlight> | </syntaxhighlight> | ||
===== الحدث 'pipe' ===== | ===== الحدث <code>'pipe'</code> ===== | ||
أُضيف في الإصدار: 0.9.4. | أُضيف في الإصدار: 0.9.4. | ||
* | * <code>src</code>: [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Readable|<stream.Readable>]]<nowiki/>مجرى القراءة والذي يُوصل إلى هذا المجرى القابل للكتابة. | ||
يٌطلَق الحدث 'pipe' عندما يُستدعى التابع stream.pipe() على مجرًى قابلٍ للقراءة مُضيفًا المجرى القابل للكتابة إلى مجموعة وجهاته.<syntaxhighlight lang="javascript"> | يٌطلَق الحدث <code>'pipe'</code> عندما يُستدعى التابع [[Node.js/stream#readable.pipe.28destination.5B.2C options.5D.E2.80.8E.29.E2.80.8E|stream.pipe()]] على مجرًى قابلٍ للقراءة مُضيفًا المجرى القابل للكتابة إلى مجموعة وجهاته.<syntaxhighlight lang="javascript"> | ||
const writer = getWritableStreamSomehow(); | const writer = getWritableStreamSomehow(); | ||
const reader = getReadableStreamSomehow(); | const reader = getReadableStreamSomehow(); | ||
سطر 216: | سطر 217: | ||
</syntaxhighlight> | </syntaxhighlight> | ||
===== الحدث 'unpipe' ===== | ===== الحدث <code>'unpipe'</code> ===== | ||
أضيف في الإصدار: 0.9.4. | أضيف في الإصدار: 0.9.4. | ||
* | * <code>src</code>: [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Readable|<stream.Readable>]] مجرى المصدر الذي لغى الاتصال بهذا المجرى القابل للكاتبة. | ||
يُطلَق الحدث 'unpipe' عندما يُستدعى التابع stream.unpipe() على مجرًى قابلٍ | يُطلَق الحدث <code>'unpipe'</code> عندما يُستدعى التابع [[Node.js/stream#readable.unpipe.28.5Bdestination.5D.E2.80.8E.29.E2.80.8E|stream.unpipe()]] على مجرًى [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Readable|قابلٍ للقراءة]]، مزيلًا المجرى [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Writable|القابل لكتابة]] من مجموعة وجهاته. | ||
يُطلق هذا الحدث أيضًا في الحالة التي يطلق فيها المجرى القابل للكاتبة خطأً عندما يُوصل به مجرى قابل للقراءة. <syntaxhighlight lang="javascript"> | يُطلق هذا الحدث أيضًا في الحالة التي يطلق فيها المجرى [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Writable|القابل للكاتبة]] خطأً عندما يُوصل به مجرى قابل للقراءة. <syntaxhighlight lang="javascript"> | ||
const writer = getWritableStreamSomehow(); | const writer = getWritableStreamSomehow(); | ||
const reader = getReadableStreamSomehow(); | const reader = getReadableStreamSomehow(); | ||
سطر 233: | سطر 234: | ||
</syntaxhighlight> | </syntaxhighlight> | ||
===== writable.cork() ===== | ===== <code>writable.cork()</code> ===== | ||
أضيف في الإصدار: 0.11.2. | أضيف في الإصدار: 0.11.2. | ||
يجبر التابع writable.cork() كل البيانات المكتوبة على أن تُخزّن مؤقتًا في الذاكرة. ستُدفع البيانات المخزّنة عندما يُستدعى أحد التابعين stream.uncork() أو stream.end(). | يجبر التابع <code>writable.cork()</code> كل البيانات المكتوبة على أن تُخزّن مؤقتًا في الذاكرة. ستُدفع البيانات المخزّنة عندما يُستدعى أحد التابعين [[Node.js/stream#writable.uncork.28.29.E2.80.8E|stream.uncork()]] أو [[Node.js/stream#writable.end.28.5Bchunk.5D.5B.2C encoding.5D.5B.2C callback.5D.E2.80.8E.29.E2.80.8E|stream.end()]]. | ||
المقصد الأساسي من التابع writable.cork() هو تجنب حالة يحدث فيها كتابة العديد من القطع الصغيرة من البيانات إلى المجرى دون جلب نسخةٍ منها إلى المخزن المؤقت الداخلي، إذ سيكون لذلك تأثير عكسي على الأداء. في مثل هذه الحالات، يمكن أن تنجز التطبيقات التي تنّفذ التابع writable._writev() عملية الكتابة المخزّنة مؤقتًا بأسلوب أكثر مثاليةً. | المقصد الأساسي من التابع <code>writable.cork()</code> هو تجنب حالة يحدث فيها كتابة العديد من القطع الصغيرة من البيانات إلى المجرى دون جلب نسخةٍ منها إلى المخزن المؤقت الداخلي، إذ سيكون لذلك تأثير عكسي على الأداء. في مثل هذه الحالات، يمكن أن تنجز التطبيقات التي تنّفذ التابع <code>writable._writev()</code> عملية الكتابة المخزّنة مؤقتًا بأسلوب أكثر مثاليةً. | ||
انظر أيضًا: التابع writable.uncork(). | انظر أيضًا: التابع [[Node.js/stream#writable.uncork.28.29.E2.80.8E|writable.uncork()]]. | ||
===== writable.destroy([error]) ===== | ===== <code>writable.destroy([error])</code> ===== | ||
أُضيف في الإصدار: 8.0.0 | أُضيف في الإصدار: 8.0.0 | ||
* | * <code>error</code>: [[JavaScript/Error|<Error>]] | ||
* القيمة المُعادة: <this> | * القيمة المُعادة: [[سلبسشقثبسئي|<this>]] | ||
يهدم التابع المجرى، ويطلق الأحداث 'error' و 'close' المُمررة. بعد هذا الاستدعاء، تكون المجاري القابلة للكتابة قد انتهت والاستدعاءات اللاحقة للتابع write() والتابع end() سوف تُفضي إلى خطأ ERR_STREAM_DESTROYED. لا ينبغي أن يعيد المنفّذون تعريف هذا التابع، ولكن يمكنهم أن يستعملوا التابع writable._destroy() بدلًا عن ذلك. | يهدم التابع المجرى، ويطلق الأحداث <code>'error'</code> و <code>'close'</code> المُمررة. بعد هذا الاستدعاء، تكون المجاري القابلة للكتابة قد انتهت والاستدعاءات اللاحقة للتابع <code>write()</code> والتابع <code>end()</code> سوف تُفضي إلى خطأ <code>ERR_STREAM_DESTROYED</code>. لا ينبغي أن يعيد المنفّذون تعريف هذا التابع، ولكن يمكنهم أن يستعملوا التابع [[Node.js/stream#writable. destroy.28err.2C callback.29.E2.80.8E|writable._destroy()]] بدلًا عن ذلك. | ||
===== writable.end([chunk][, encoding][, callback]) ===== | ===== <code>writable.end([chunk][, encoding][, callback])</code> ===== | ||
سجل التغييرات | سجل التغييرات | ||
{| class="wikitable mw-collapsible" | {| class="wikitable sortable mw-collapsible" | ||
!الإصدار | !الإصدار | ||
!التغييرات | !التغييرات | ||
! | ! rowspan="4" | | ||
|- | |- | ||
|10.0.0 | |10.0.0 | ||
|يعيد هذا التابع الآن مرجعًا إلى مجرًى من النوع writable. | |يعيد هذا التابع الآن مرجعًا إلى مجرًى من النوع <code>writable</code>. | ||
|- | |- | ||
|8.0.0 | |8.0.0 | ||
|يمكن الآن أن يكون الوسيط chunk نسخة من النوع Uint8Array. | |يمكن الآن أن يكون الوسيط <code>chunk</code> نسخة من النوع <code>Uint8Array</code>. | ||
|- | |- | ||
|0.9.4 | |0.9.4 | ||
|أضيف هذا التابع في الإصدار 0.9.4 | |أضيف هذا التابع في الإصدار 0.9.4 | ||
|} | |} | ||
* chunk: <string> | <Buffer> | <Uint8Array> | <any> بيانات اختيارية يراد كتابتها. من أجل المجاري التي لا تعمل في وضع الكائن، يجب أن يكون الوسيط chunk سلسلةً نصيةً (string) أو كائنًا من النوع Buffer أو النوع Uint8Array. من أجل المجاري في نمط الكائن، يمكن أن تكون chunk أي قيمة غير null. | * <code>chunk</code>: [[JavaScript/String|<string>]] | [[Node.js/buffer#.D8.A7.D9.84.D8.B5.D9.86.D9.81 Buffer|<Buffer>]] | [[سيب\ئسبيب|<Uint8Array>]] | [[يبليءئبل|<any>]] بيانات اختيارية يراد كتابتها. من أجل المجاري التي لا تعمل في وضع الكائن، يجب أن يكون الوسيط <code>chunk</code> سلسلةً نصيةً (string) أو كائنًا من النوع <code>Buffer</code> أو النوع <code>Uint8Array</code>. من أجل المجاري في نمط الكائن، يمكن أن تكون <code>chunk</code> أي قيمة غير <code>null</code>. | ||
* encoding: <string> التشفير، إذا كانت | * <code>encoding</code>: [[JavaScript/String|<string>]] التشفير، إذا كانت <code>chunk</code> سلسلة نصية. | ||
* <Function> : | * [[JavaScript/Function|<Function>]] :<code>callback</code> دالة رد نداء اختيارية عندما يُنهى المجرى. | ||
* القيمة المعادة: <this> | * القيمة المعادة: [[قفسافيلب|<this>]] | ||
يشير استدعاء التابع writable.end() أنّه لا مزيد من البيانات يراد كتابتها على المجرى القابل للكتابة (Writable). سيسمح الوسيطان الاختياريان chunk و encoding أن تُكتب قطعة إضافية واحدة أخيرة من البيانات مباشرةً قبل إغلاق المجرى. إذا زُودت، سترفق دالة رد نداء اختيارية كمنصت للحدث 'finish'. | يشير استدعاء التابع <code>writable.end()</code> أنّه لا مزيد من البيانات يراد كتابتها على المجرى [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Writable|القابل للكتابة (Writable)]]. سيسمح الوسيطان الاختياريان <code>chunk</code> و <code>encoding</code> أن تُكتب قطعة إضافية واحدة أخيرة من البيانات مباشرةً قبل إغلاق المجرى. إذا زُودت، سترفق دالة رد نداء <code>callback</code> اختيارية كمنصت للحدث [[Node.js/stream#.D8.A7.D9.84.D8.AD.D8.AF.D8.AB .27finish.27|'finish']]. | ||
سيثير استدعاء التابع stream.write() بعد استدعاء stream.end() خطأً.<syntaxhighlight lang="javascript"> | سيثير استدعاء التابع [[Node.js/stream#writable.write.28chunk.5B.2C encoding.5D.5B.2C callback.5D.E2.80.8E.29.E2.80.8E|stream.write()]] بعد استدعاء [[Node.js/stream#writable.end.28.5Bchunk.5D.5B.2C encoding.5D.5B.2C callback.5D.E2.80.8E.29.E2.80.8E|stream.end()]] خطأً.<syntaxhighlight lang="javascript"> | ||
// 'world!' ومن ثم أنهِ ب ' hello,' اكتب | |||
const fs = require('fs'); | const fs = require('fs'); | ||
سطر 286: | سطر 280: | ||
file.end('world!'); | file.end('world!'); | ||
كتابة المزيد ليست مسموحة الآن | //كتابة المزيد ليست مسموحة الآن | ||
</syntaxhighlight> | </syntaxhighlight> | ||
===== writable.setDefaultEncoding(encoding) ===== | ===== <code>writable.setDefaultEncoding(encoding)</code> ===== | ||
سجل التغييرات | سجل التغييرات | ||
سطر 296: | سطر 290: | ||
!الإصدار | !الإصدار | ||
!التغييرات | !التغييرات | ||
! | ! rowspan="4" | | ||
|- | |- | ||
|6.1.0 | |6.1.0 | ||
|يعيد هذا التابع الآن مرجعًا إلى مجرًى من النوع writable. | |يعيد هذا التابع الآن مرجعًا إلى مجرًى من النوع <code>writable</code>. | ||
|- | |- | ||
|0.11.15 | |0.11.15 | ||
|أضيف في 0.11.15 | |أضيف في 0.11.15 | ||
|- | |- | ||
| | | colspan="2" | | ||
| | |||
|} | |} | ||
* string> :encoding | * [[JavaScript/String|<string>]] :encoding | ||
* القيمة المعادة: <this> | * القيمة المعادة: [[sldkfj;|<this>]] | ||
يضبط التابع writable.setDefaultEncoding() الترميزَ الافتراضي (encoding) للمجرى القابل للكتابة (Writable). | يضبط التابع <code>writable.setDefaultEncoding()</code> الترميزَ الافتراضي (<code>encoding</code>) للمجرى [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Writable|القابل للكتابة (Writable)]]. | ||
===== writable.uncork() ===== | ===== <code>writable.uncork()</code> ===== | ||
أُضيف في الإصدار: 0.11.2. | أُضيف في الإصدار: 0.11.2. | ||
يفرِّغ التابع writable.uncork() كل البيانات التي خُزِّنت منذ أن استُدعي stream.cork(). | يفرِّغ التابع <code>writable.uncork()</code> كل البيانات التي خُزِّنت منذ أن استُدعي [[Node.js/stream#writable.cork.28.29.E2.80.8E|stream.cork()]]. | ||
عند استخدام التابع writable.cork() والتابع writable. | عند استخدام التابع [[Node.js/stream#writable.cork.28.29.E2.80.8E|writable.cork()]] والتابع <code>()writable.uncork</code> لإدارة التخزين المؤقت أثناء الكتابة على مجرًى، يُنصح بأن يؤخر استدعاء <code>writable.uncork()</code> باستخدام <code>process.nextTick()</code>. يسمح القيام بذلك بتجميع كل استدعاءات التابع <code>writable.write()</code> التي تحصل خلال طورٍ واحدٍ لحلقة أحداث Node.js مُعطاة.<syntaxhighlight lang="javascript"> | ||
stream.cork(); | stream.cork(); | ||
stream.write('some '); | stream.write('some '); | ||
stream.write('data '); | stream.write('data '); | ||
process.nextTick(() => stream.uncork()); | process.nextTick(() => stream.uncork()); | ||
</syntaxhighlight>إذا استُدعي التابع writable.cork() عدة مرات لمجرىً ما، فيجب أن يُستدعى التابع writable.uncork() نفس عدد الاستدعاءات لتفريغ البيانات المخزنة.<syntaxhighlight lang="javascript"> | </syntaxhighlight>إذا استُدعي التابع [[Node.js/stream#writable.cork.28.29.E2.80.8E|writable.cork()]] عدة مرات لمجرىً ما، فيجب أن يُستدعى التابع <code>writable.uncork()</code> نفس عدد الاستدعاءات لتفريغ البيانات المخزنة.<syntaxhighlight lang="javascript"> | ||
stream.cork(); | stream.cork(); | ||
stream.write('some '); | stream.write('some '); | ||
سطر 337: | سطر 323: | ||
//مرّة ثانية uncork()لن تُفرّغ البيانات حتى يُستدعى التابع | |||
stream.uncork(); | stream.uncork(); | ||
}); | }); | ||
</syntaxhighlight>انظر أيضًا: التابع writable.cork(). | </syntaxhighlight>انظر أيضًا: التابع [[Node.js/stream#writable.cork.28.29.E2.80.8E|writable.cork()]]. | ||
===== writable.writableHighWaterMark ===== | ===== <code>writable.writableHighWaterMark</code> ===== | ||
أُضيفت في الإصدار: 9.3.0. | أُضيفت في الإصدار: 9.3.0. | ||
* <number> | * [[Node.js/stream#writable.cork.28.29.E2.80.8E|<number>]] | ||
يعيد القيمة highWaterMark التي مُرِّرت عند إنشاء المجرى القابل للكتابة. | يعيد القيمة <code>highWaterMark</code> التي مُرِّرت عند إنشاء المجرى القابل للكتابة (<code>Writable</code>). | ||
===== writable.writableLength ===== | ===== <code>writable.writableLength</code> ===== | ||
أُضيفت في الإصدار: 9.4.0. | أُضيفت في الإصدار: 9.4.0. | ||
تحوي هذه الخاصية عدد البايتات (أو الكائنات) الجاهزة للكتابة والموجودة في الطابور. توفر هذه القيمة بيانات داخلية متعلقة بحالة القيمة highWaterMark. | تحوي هذه الخاصية عدد البايتات (أو الكائنات) الجاهزة للكتابة والموجودة في الطابور. توفر هذه القيمة بيانات داخلية متعلقة بحالة القيمة <code>highWaterMark</code>. | ||
===== <code>writable.write(chunk[, encoding][, callback])</code> ===== | |||
سجل التغييرات | سجل التغييرات | ||
{| class="wikitable" | {| class="wikitable mw-collapsible" | ||
!الإصدار | !الإصدار | ||
!التغييرات | !التغييرات | ||
! | ! rowspan="4" | | ||
|- | |- | ||
|8.0.0 | |8.0.0 | ||
|يمكن الآن أن يكون الوسيط نسخة من النوع Uint8Array. | |يمكن الآن أن يكون الوسيط نسخة من النوع Uint8Array. | ||
|- | |- | ||
|6.0.0 | |6.0.0 | ||
|سوف يعتبر دائمًا تمريرالقيمة null كمعامل chunk غير صالح الآن، حتى في نمط الكائن. | |سوف يعتبر دائمًا تمريرالقيمة null كمعامل chunk غير صالح الآن، حتى في نمط الكائن. | ||
|- | |- | ||
|0.9.4 | |0.9.4 | ||
|أُضيف هذا التابع. | |أُضيف هذا التابع. | ||
|} | |} | ||
* | * <code>chunk</code>: [[JavaScript/String|<string>]] | [[Node.js/buffer#.D8.A7.D9.84.D8.B5.D9.86.D9.81 Buffer|<Buffer>]] | [[سب\ئلئسلشئس|<Uint8Array>]] | [[\سبيس\بي|<any>]] بيانات اختيارية للكتابة. من أجل المجاري التي لا تعمل في نمط الكائن، يجب أن تكون <code>chunk</code> سلسلة نصية أو <code>Buffer</code> أو <code>Uint8Array</code>. من أجل مجاري نمط الكائن، يمكن أن تكون chunk أي نوع من أنواع بيانات JavaScript غير <code>null</code>. | ||
* string> :encoding> الترميز، إذا كان الوسيط chunk سلسلةً نصيةً. | * [[JavaScript/String|<string>]] :<code>encoding</code> الترميز، إذا كان الوسيط <code>chunk</code> سلسلةً نصيةً. | ||
* callback: <Function> دالة رد النداء المراد استدعاؤها عندما تُدفع قطعة البيانات. | * <code>callback</code>: [[JavaScript/Function|<Function>]] دالة رد النداء المراد استدعاؤها عندما تُدفع قطعة البيانات. | ||
* القيمة المُعادة: <boolean> قيمة منطقية تكون false إذا كان المجرى يريد للشيفرة المستدعية أن تنتظر انطلاق الحدث 'drain' قبل الاستمرار بكتابة المزيد من البيانات؛ وإلّا تكون true. | * القيمة المُعادة: [[JavaScript/Boolean|<boolean>]] قيمة منطقية تكون <code>false</code> إذا كان المجرى يريد للشيفرة المستدعية أن تنتظر انطلاق الحدث <code>'drain'</code> قبل الاستمرار بكتابة المزيد من البيانات؛ وإلّا تكون <code>true</code>. | ||
يكتب التابع writable.write() بعض البيانات إلى المجرى. ويستدعي رد النداء (callback) المُزوّد حالما تكون البيانات المكتوبة قد عولجت بشكل كامل. إذا حصل خطأ، ربما يُستدعى callback أو لا مع تمرير الخطأ كوسيطه الأول. للكشف عن أخطاء الكتابة بشكل موثوق، أضف مُنصتًا إلى الحدث 'error'. | يكتب التابع <code>writable.write()</code> بعض البيانات إلى المجرى. ويستدعي رد النداء (<code>callback</code>) المُزوّد حالما تكون البيانات المكتوبة قد عولجت بشكل كامل. إذا حصل خطأ، ربما يُستدعى <code>callback</code> أو لا مع تمرير الخطأ كوسيطه الأول. للكشف عن أخطاء الكتابة بشكل موثوق، أضف مُنصتًا إلى الحدث <code>'error'</code>. | ||
القيمة المعادة هي true إذا كانت ذاكرة التخزين الداخلية أقل من القيمة highWaterMark المُهيئة عندما أُنشئ المجرى بعد إقرار chunk. إذا أُعيد | القيمة المعادة هي <code>true</code> إذا كانت ذاكرة التخزين الداخلية أقل من القيمة <code>highWaterMark</code> المُهيئة عندما أُنشئ المجرى بعد إقرار <code>chunk</code>. إذا أُعيد <code>false</code>، ينبغي أن تتوقف المحاولات الإضافية لكتابة البيانات على المجرى حتى يُطلق الحدث [[Node.js/stream#.D8.A7.D9.84.D8.AD.D8.AF.D8.AB .27drain.27|'drain']]. | ||
طالما أن المجرى لم يفرغ، فإن استدعاء write() سوف يخزّن chunk مؤقتًا، ويعيد خطأ. حالما تُستهلك كل القطع الحالية المخزنة مؤقتًا (قُبلَت للتسليم من قبل نظام التشغيل)، سوف يُطلق الحدث 'drain'. حين يعيد write() القيمة | طالما أن المجرى لم يفرغ، فإن استدعاء <code>write()</code> سوف يخزّن <code>chunk</code> مؤقتًا، ويعيد خطأ. حالما تُستهلك كل القطع الحالية المخزنة مؤقتًا (قُبلَت للتسليم من قبل نظام التشغيل)، سوف يُطلق الحدث <code>'drain'</code>. حين يعيد <code>write()</code> القيمة <code>false</code>، فمن المستحسن ألّا تكتب المزيد من القطع حتى يُطلق الحدث <code>'drain'</code>. بينما يُسمح باستدعاء <code>write()</code> على مجرًى لم يفرغ بعد، فسوف تخزن Node.js مؤقتًا كل القطع المكتوبة حتى يحصل استخدام ذاكرة أعظمي،عند هذه النقطة ستتوقف دون قيود. حتى قبل حصول التوقف، سيسبب الاستخدام المرتفع للذاكرة أداءًا ضعيفًا لجامع القمامة (منظّف الذاكرة) وارتفاع مجموعة الذاكرة المُقِيمة (RSS) [والذي لن يعاد استرجاعه للنظام بشكل طبيعي، حتى بعد عدم الحاجة للذاكرة]. بما أن المقابس TCP قد لا تنضب أبدًا إذا لم يقرأ الند (peer) البعيد البيانات، ربما تقود الكتابة على مقبس لا يُصرّف لحصول ثغرة قابلة للاستغلال (exploitable vulnerability). | ||
كتابة البيانات على مجرى لا ينفد هو معضلة خصوصًا مع مجاري التحويل ( Transform)، لأنّ مجاري Transform تتوقف افتراضيًا حتى تنقل تُتصل بمجرى آخر أو يضاف معالج للحدث 'data' أو الحدث 'readable'. | كتابة البيانات على مجرى لا ينفد هو معضلة خصوصًا مع مجاري التحويل ( [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81: stream.Transform|Transform]])، لأنّ مجاري <code>Transform</code> تتوقف افتراضيًا حتى تنقل تُتصل بمجرى آخر أو يضاف معالج للحدث <code>'data'</code> أو الحدث <code>'readable'</code>. | ||
اذا أمكن توليد البيانات التي ستُكتب أو جلبها عند الطلب، فمن المستحسن تغليفها في نفس المجرى ثم تحويلها إلى المجرى القابل للقراءة عبر استخدام التابع stream.pipe(). | اذا أمكن توليد البيانات التي ستُكتب أو جلبها عند الطلب، فمن المستحسن تغليفها في نفس المجرى ثم تحويلها إلى المجرى [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Readable|القابل للقراءة]] عبر استخدام التابع [[Node.js/stream#readable.pipe.28destination.5B.2C options.5D.E2.80.8E.29.E2.80.8E|stream.pipe()]]. | ||
ولكن إذا كان رُجّج استدعاء write() | ولكن إذا كان رُجّج استدعاء <code>write()</code>، فمن الممكن مراعاة الضغط العائد وتجنب مشاكل الذاكرة باستخدام الحدث [[Node.js/stream#.D8.A7.D9.84.D8.AD.D8.AF.D8.AB .27drain.27|'drain']]:<syntaxhighlight lang="javascript"> | ||
function write(data, cb) { | function write(data, cb) { | ||
if (!stream.write(data)) { | if (!stream.write(data)) { | ||
سطر 403: | سطر 381: | ||
//قبل القيام بأي كتابة أخرى cb الانتظار حتى يُستدعى | |||
write('hello', () => { | write('hello', () => { | ||
سطر 409: | سطر 387: | ||
}); | }); | ||
</syntaxhighlight>سوف يهمل المجرى Writable في نمط الكائن الوسيط encoding دائمًا. | </syntaxhighlight>سوف يهمل المجرى <code>Writable</code> في نمط الكائن الوسيط <code>encoding</code> دائمًا. | ||
=== المجاري القابلة للقراءة === | === المجاري القابلة للقراءة === | ||
المجاري القابلة للقراءة هي فكرة مجرّدة عن مصدرٍ تُستهلك منه البيانات. | المجاري القابلة للقراءة هي فكرة مجرّدة عن مصدرٍ تُستهلك منه البيانات. | ||
أمثلة عن المجاري المقروءة هي: | أمثلة عن المجاري المقروءة (<code>Readable</code>) هي: | ||
* استجابة لطلبيات HTTP من طرف العميل. | * [[Node.js/http#.D8.A7.D9.84.D8.B5.D9.86.D9.81 http.IncomingMessage|استجابة لطلبيات HTTP من طرف العميل]]. | ||
* طلبيات HTTP من طرف الخادم. | * [[Node.js/http#.D8.A7.D9.84.D8.B5.D9.86.D9.81 http.IncomingMessage|طلبيات HTTP من طرف الخادم.]] | ||
* مجاري الوحدة fs القابلة للقراءة. | * [[Node.js/fs#.D8.A7.D9.84.D8.B5.D9.86.D9.81 fs.ReadStream.E2.80.8E|مجاري الوحدة fs القابلة للقراءة]]. | ||
* مجاري الوحدة zlib. | * [[Node.js/zlib|مجاري الوحدة zlib]]. | ||
* مجاري الوحدة crypto. | * [[Node.js#.D8.A7.D9.84.D9.88.D8.AD.D8.AF.D8.A9 Crypto|مجاري الوحدة crypto]]. | ||
* مقابس TCP. | * [[Node.js/net#.D8.A7.D9.84.D8.B5.D9.86.D9.81 net.Socket|مقابس TCP]]. | ||
* المجرى stdout والمجرى stderr للعملية الابن. | * [[Node.js/child process#.D8.A7.D9.84.D8.AE.D8.A7.D8.B5.D9.8A.D8.A9 subprocess.stdout|المجرى stdout والمجرى stderr للعملية الابن.]] | ||
* المجرى Process.stdin للعملية الحالية. | * المجرى [[Node.js/process#process.stdin|Process.stdin]] للعملية الحالية. | ||
كل المجاري | كل المجاري <nowiki/>[[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Readable|القابلة للقراءة (Readable)]] تطبّق الواجهة المعرّفة من قبل الصنف <code>stream.Readable</code>. | ||
==== نمطا القراءة ==== | ==== نمطا القراءة ==== | ||
تعمل المجاري القابلة للقراءة بفاعلية في أحد النمطين التاليين: نمط التدفق (flowing mode) ونمط التوقف المؤقتً (paused mode). هذه الأنماط منفصلة عن نمط الكائن. يمكن أن يكون المجرى المقروء في نمط الكائن أو لا، بغض النظر عن إذا كان في نمط التدفق أو نمط التوقف المؤقت. | تعمل المجاري [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Readable|القابلة للقراءة (<code>Readable</code>)]] بفاعلية في أحد النمطين التاليين: نمط التدفق (flowing mode) ونمط التوقف المؤقتً (paused mode). هذه الأنماط منفصلة عن [[Node.js/stream#.D9.86.D9.85.D8.B7 .D8.A7.D9.84.D9.83.D8.A7.D8.A6.D9.86|نمط الكائن]]. يمكن أن يكون المجرى المقروء في نمط الكائن أو لا، بغض النظر عن إذا كان في نمط التدفق أو نمط التوقف المؤقت. | ||
* في نمط التدفق، تُقرأ البيانات من النظام الرئيسي بشكل تلقائي وتُقدّم للتطبيق بأسرع ما يمكن باستخدام الأحداث عبر الواجهة EventEmitter. | * في نمط التدفق، تُقرأ البيانات من النظام الرئيسي بشكل تلقائي وتُقدّم للتطبيق بأسرع ما يمكن باستخدام الأحداث عبر الواجهة [[Node.js/events#.D8.A7.D9.84.D8.B5.D9.86.D9.81: EventEmitter|EventEmitter]]. | ||
* في نمط التوقف المؤقت، يجب أن يُستدعى stream.read() بشكل صريح لقراءة قطع البيانات من المجرى. | * في نمط التوقف المؤقت، يجب أن يُستدعى stream.read() بشكل صريح لقراءة قطع البيانات من المجرى. | ||
تبدأ كل | تبدأ كل [[Node.js/stream#.D9.86.D9.85.D8.B7 .D8.A7.D9.84.D9.83.D8.A7.D8.A6.D9.86|المجاري القابلة للقراءة]] في نمط التوقف المؤقت ولكن يمكن تبديلها إلى نمط التدفق بإحدى الطرق التالية: | ||
* إضافة معالج للحدث 'data'. | * إضافة معالج للحدث [[Node.js/stream#.D8.A7.D9.84.D8.AD.D8.AF.D8.AB:.27data.27|'data']]. | ||
* استدعاء التابع stream.resume(). | * استدعاء التابع [[Node.js/stream#readable.resume.28.29.E2.80.8E|stream.resume()]]. | ||
* استدعاء التابع stream.pipe() لإرسال البيانات إلى المجرى القابل للكتابة (Writable). | * استدعاء التابع [[Node.js/stream#readable.pipe.28destination.5B.2C options.5D.E2.80.8E.29.E2.80.8E|stream.pipe()]] لإرسال البيانات إلى المجرى [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Writable|القابل للكتابة (Writable)]]. | ||
يمكن | يمكن نقل <code>Readable</code> بالعكس إلى نمط التوقف باستخدام أحد التالي: | ||
* إذا لم يكن هناك أنبوب متصل بالوجهات، فيمكن ذلك عبر استدعاء التابع stream.pause(). | * إذا لم يكن هناك أنبوب متصل بالوجهات، فيمكن ذلك عبر استدعاء التابع [[Node.js/stream#readable.pause.28.29.E2.80.8E|stream.pause()]]. | ||
* إذا كان هناك أنبوب متصل بالوجهات، فيمكن ذلك عبر إزالة كل أنابيب الوجهات. قد تُزال العديد من أنابيب الوجهات باستدعاء التابع stream.unpipe(). | * إذا كان هناك أنبوب متصل بالوجهات، فيمكن ذلك عبر إزالة كل أنابيب الوجهات. قد تُزال العديد من أنابيب الوجهات باستدعاء التابع [[Node.js/stream#readable.unpipe.28.5Bdestination.5D.E2.80.8E.29.E2.80.8E|stream.unpipe()]]. | ||
المبدأ الأساسي الذي يجب تذكره دومًا هو أنّ المجرى القابل للقراءة (Readable) لن يولّد بيانات حتى تتوافر آلية إمّا لاستهلاك تلك البيانات أو تجاهلها. إذا كانت آلية الاستهلاك معطلة أو مُستبعدة، سوف يحاول Readable إيقاف توليد البيانات. | المبدأ الأساسي الذي يجب تذكره دومًا هو أنّ المجرى [https://wiki.hsoub.com/Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81_stream.Readable القابل للقراءة (<code>Readable</code>)] لن يولّد بيانات حتى تتوافر آلية إمّا لاستهلاك تلك البيانات أو تجاهلها. إذا كانت آلية الاستهلاك معطلة أو مُستبعدة، سوف يحاول <code>Readable</code> إيقاف توليد البيانات. | ||
من أجل التوافقية مع الإصدارات السابقة، لن توقف ازالة معالج حدث 'data' المجرى تلقائيًا. أيضًا، إذا كان هنالك ممرات للوجهات، فحين ذلك لن يضمَن استدعاء stream.pause() أن يبقى المجرى متوقفًا حالما تفرغ هذه الوجهات وتسأل عن المزيد من البيانات. | من أجل التوافقية مع الإصدارات السابقة، لن توقف ازالة معالج حدث [[Node.js/stream#.D8.A7.D9.84.D8.AD.D8.AF.D8.AB:.27data.27|'data']] المجرى تلقائيًا. أيضًا، إذا كان هنالك ممرات للوجهات، فحين ذلك لن يضمَن استدعاء [[Node.js/stream#readable.pause.28.29.E2.80.8E|stream.pause()]] أن يبقى المجرى متوقفًا حالما تفرغ هذه الوجهات وتسأل عن المزيد من البيانات. | ||
إذا قُلب Readable إلى نمط التدفق ولم يتوفر مستهلكون لمعالجة البيانات، ستضيع تلك البيانات. يمكن أن يحصل هذا على سبيل المثال، عندما يُستدعى تابع readable.resume() دون منصت مرفق بحدث 'data'، أو عندما يُزال معالج حدث 'data' من المجرى. | إذا قُلب [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Readable|Readable]] إلى نمط التدفق ولم يتوفر مستهلكون لمعالجة البيانات، ستضيع تلك البيانات. يمكن أن يحصل هذا على سبيل المثال، عندما يُستدعى تابع <code>readable.resume()</code> دون منصت مرفق بحدث <code>'data'</code>، أو عندما يُزال معالج حدث <code>'data'</code> من المجرى. | ||
إضافة معالج حدث 'readable' يجعل توقف تدفق المجرى تلقائيًا، وتُتستهلك البيانات بواسطة readable.read(). إذا أُزيل معالج حدث 'readable'، حينذاك سيبدأ المجرى بالتدفق مُجدّدًا إذا كان هناك معالج حدث 'data' موجود. | إضافة معالج حدث [[Node.js/stream#.D8.A7.D9.84.D8.AD.D8.AF.D8.AB: .27readable.27|'readable']] يجعل توقف تدفق المجرى تلقائيًا، وتُتستهلك البيانات بواسطة [[Node.js/stream#readable.read.28.5Bsize.5D.E2.80.8E.29.E2.80.8E|readable.read()]]. إذا أُزيل معالج حدث [[Node.js/stream#.D8.A7.D9.84.D8.AD.D8.AF.D8.AB: .27readable.27|'readable']]، حينذاك سيبدأ المجرى بالتدفق مُجدّدًا إذا كان هناك معالج حدث [[Node.js/stream#.D8.A7.D9.84.D8.AD.D8.AF.D8.AB:.27data.27|'data']] موجود. | ||
==== الحالات الثلاث ==== | ==== الحالات الثلاث ==== | ||
هناك "وضعا" تشغيل للمجرى القابل للقراءة (Readable)، هما تلخيص مبسّط لإدارة حالات داخلية كثر تعقيدًا والتي تحصل خلال تطبيق مجرى قابل للقراءة (Readable). | هناك "وضعا" تشغيل للمجرى القابل للقراءة (<code>Readable</code>)، هما تلخيص مبسّط لإدارة حالات داخلية كثر تعقيدًا والتي تحصل خلال تطبيق مجرى قابل للقراءة (<code>Readable</code>). | ||
على وجه التحديد وفي أي نقطة معطاة من الزمن يأخذ كل مجرى القابل للقراءة (Readable) إحدى الحالات الثلاث التالية: | على وجه التحديد وفي أي نقطة معطاة من الزمن يأخذ كل مجرى القابل للقراءة (<code>Readable</code>) إحدى الحالات الثلاث التالية: | ||
* readable.readableFlowing === null | * <code>readable.readableFlowing === null</code> | ||
* readable.readableFlowing === false | * <code>readable.readableFlowing === false</code> | ||
* readable.readableFlowing === true | * <code>readable.readableFlowing === true</code> | ||
عندما تكون readable.readableFlowing هي null ، لن تكون هناك آلية مُقدمة لاستهلاك بيانات المجرى. لذلك، لن يولد المجرى بيانات. بينما في هذه الحالة، سيبدل إرفاق مُنصت إلى الحدث 'data' أو استدعاء التابع readable.pipe() أو استدعاء التابع readable.resume() قيمة الخاصية readable.readableFlowing | عندما تكون <code>readable.readableFlowing</code> هي <code>null</code> ، لن تكون هناك آلية مُقدمة لاستهلاك بيانات المجرى. لذلك، لن يولد المجرى بيانات. بينما في هذه الحالة، سيبدل إرفاق مُنصت إلى الحدث <code>'data'</code> أو استدعاء التابع <code>readable.pipe()</code> أو استدعاء التابع <code>readable.resume()</code> قيمة الخاصية <code>readable.readableFlowing</code> | ||
إلى القيمة | إلى القيمة <code>true</code>، متسببًا ببدء <code>Readable</code> بإطلاق الأحداث بفاعلية حين توّلد البيانات. | ||
سوف يتسبب استدعاء readable.pause() أو readable.unpipe() أو استقبال الضغط العائد بضبط قيمة الخاصية readable.readableFlowing إلى | سوف يتسبب استدعاء <code>readable.pause()</code> أو <code>readable.unpipe()</code> أو استقبال الضغط العائد بضبط قيمة الخاصية <code>readable.readableFlowing</code> إلى <code>false</code>، قاطعًا تدفق الأحداث مؤقتًا ولكن غير قاطعٍ لتدفق البيانات. بينما لن يقلب إرفاق منصت بالحدث <code>'data'</code> قيمة الخاصية <code>readable.readableFlowing</code> إلى <code>true</code> في هذه الحالة.<syntaxhighlight lang="javascript"> | ||
const { PassThrough, Writable } = require('stream'); | const { PassThrough, Writable } = require('stream'); | ||
const pass = new PassThrough(); | const pass = new PassThrough(); | ||
سطر 469: | سطر 447: | ||
pass.write('ok'); | pass.write('ok'); | ||
لن تطلق الحدث | // 'data'لن تطلق الحدث | ||
pass.resume(); | pass.resume(); | ||
يجب أن تُستدعى لجعل المجرى يطلق الحدث | //'data' يجب أن تُستدعى لجعل المجرى يطلق الحدث | ||
</syntaxhighlight>بينما تكون قيمة readable.readableFlowing هي | </syntaxhighlight>بينما تكون قيمة <code>readable.readableFlowing</code> هي <code>false</code>، قد تتراكم البيانات داخل الذاكرة المؤقتة الداخلية للمجرى. | ||
=== اختر أسلوب واجهة برمجية وحيد === | ==== اختر أسلوب واجهة برمجية وحيد ==== | ||
تطورت الواجهات البرمجية (API) للمجرى القابل للقراءة (Readable) عبر عدة إصدارات Node.js وقدّمت عدّة توابع لاستهلاك بيانات المجرى. بشكل عام، ينبغي أن يختار المطورون واحدة من توابع استهلاك البيانات ولا ينبغي استخدام عدة توابع لاستهلاك البيانات من مجرى واحد. بشكل خاص، قد يقود استخدام مزيج من on('data') أو on('readable') أو pipe() أو توابع تكرارية غير متزامنة إلى سلوك غير متوقع. | تطورت الواجهات البرمجية (API) للمجرى القابل للقراءة (<code>Readable</code>) عبر عدة إصدارات Node.js وقدّمت عدّة توابع لاستهلاك بيانات المجرى. بشكل عام، ينبغي أن يختار المطورون واحدة من توابع استهلاك البيانات ولا ينبغي استخدام عدة توابع لاستهلاك البيانات من مجرى واحد. بشكل خاص، قد يقود استخدام مزيج من <code>on('data')</code> أو <code>on('readable')</code> أو <code>pipe()</code> أو توابع تكرارية غير متزامنة إلى سلوك غير متوقع. | ||
ينصح باستخدام التابع readable.pipe() لأغلب المستخدمين بما أنه نُفّذ لتقديم أسهل طريقة لاستهلاك بيانات المجرى. | ينصح باستخدام التابع <code>readable.pipe()</code> لأغلب المستخدمين بما أنه نُفّذ لتقديم أسهل طريقة لاستهلاك بيانات المجرى. | ||
يمكن أن يستخدم المطورون الذين يطلبون المزيد من التحكم الدقيق على نقل وتوليد البيانات الصنف EventEmitter و readable.on('readable')/readable.read() أو الواجهات readable.pause()/readable.resume(). | يمكن أن يستخدم المطورون الذين يطلبون المزيد من التحكم الدقيق على نقل وتوليد البيانات الصنف [[Node.js/events#.D8.A7.D9.84.D8.B5.D9.86.D9.81: EventEmitter|EventEmitter]] و <code>readable.on('readable')</code>/<code>readable.read()</code> أو الواجهات <code>readable.pause()</code>/<code>readable.resume()</code>. | ||
=== الصنف stream.Readable === | ==== الصنف <code>stream.Readable</code> ==== | ||
أُضيف في الإصدار: 0.9.4. | أُضيف في الإصدار: 0.9.4. | ||
==== الحدث:'close' ==== | ===== الحدث:<code>'close'</code> ===== | ||
أُضيف في الإصدار: 0.9.4. | أُضيف في الإصدار: 0.9.4. | ||
يُطلق الحدث'close' عندما يكون المجرى أو أحد موارده الأساسية (واصف الملف مثلًا) قد أُغلق. يشير الحدث أنّه لن تُطلق مزيد من الأحداث ولن يحصل المزيد من العمليات المتعلقة بالمجرى. | يُطلق الحدث<code>'close'</code> عندما يكون المجرى أو أحد موارده الأساسية (واصف الملف مثلًا) قد أُغلق. يشير الحدث أنّه لن تُطلق مزيد من الأحداث ولن يحصل المزيد من العمليات المتعلقة بالمجرى. | ||
لا تُطلق كل المجاري القابلة للقراءة الحدث 'close'. | لا تُطلق كل المجاري القابلة للقراءة الحدث <code>'close'</code>. | ||
==== الحدث:'data' ==== | ===== الحدث:<code>'data'</code> ===== | ||
أُضيف في الإصدار: 0.9.4. | أُضيف في الإصدار: 0.9.4. | ||
* <Buffer> | <string> | <any> : | * [[Node.js/buffer#.D8.A7.D9.84.D8.B5.D9.86.D9.81 Buffer|<Buffer>]] | [[JavaScript/String|<string>]] | [[شيخبتطش\حخبتح|<any>]] :<code>chunk</code> قطع البيانات. من أجل المجاري التي لا تعمل في نمط الكائن، ستكون القطعة إمّا سلسلة نصية أو كائنًا من النوع Buffer. من أجل المجاري في نمط الكائن، يمكن أن تكون القطعة أي نوع من أنواع البيانات باستثناء null. | ||
سيُطلق الحدث 'data' كلما تخلى المجرى عن حيازة قطعة من البيانات إلى المستهلك. قد يحصل هذا كلّما بُدِّل المجرى في نمط الكائن باستدعاء readable.pipe() أو readable.resume() أو بإرفاق رد نداء منصت للحدث 'data'. سوف يُطلق الحدث 'data' أيضًا كلّما استُدعي التابع readable.read() وكانت قطعة من البيانات متوفرة لتُعاد. | سيُطلق الحدث <code>'data'</code> كلما تخلى المجرى عن حيازة قطعة من البيانات إلى المستهلك. قد يحصل هذا كلّما بُدِّل المجرى في نمط الكائن باستدعاء <code>readable.pipe()</code> أو <code>readable.resume()</code> أو بإرفاق رد نداء منصت للحدث <code>'data'</code>. سوف يُطلق الحدث <code>'data'</code> أيضًا كلّما استُدعي التابع <code>readable.read()</code> وكانت قطعة من البيانات متوفرة لتُعاد. | ||
إرفاق مُنصت بالحدث 'data' لمجرىً لم يُوقَف بشكل صريح سوف يقلب المجرى إلى نمط التدفق. ستُمرر البيانات بعدئذٍ حالما تكون متوفرة. | إرفاق مُنصت بالحدث <code>'data'</code> لمجرىً لم يُوقَف بشكل صريح سوف يقلب المجرى إلى نمط التدفق. ستُمرر البيانات بعدئذٍ حالما تكون متوفرة. | ||
سوف يمرر ردُ نداء المُنصت قطعةَ البيانات كسلسلة نصية إذا كان التشفير الافتراضي مُحدّدًا باستخدام التابع readable.setEncoding() | سوف يمرر ردُ نداء المُنصت قطعةَ البيانات كسلسلة نصية إذا كان التشفير الافتراضي مُحدّدًا باستخدام التابع <code>readable.setEncoding()</code>، وإلّا ستمرر البيانات ضمن كائن من النوع <code>Buffer</code>.<syntaxhighlight lang="javascript"> | ||
const readable = getReadableStreamSomehow(); | const readable = getReadableStreamSomehow(); | ||
readable.on('data', (chunk) => { | readable.on('data', (chunk) => { | ||
سطر 508: | سطر 486: | ||
</syntaxhighlight> | </syntaxhighlight> | ||
==== الحدث: 'end' ==== | ===== الحدث: <code>'end'</code> ===== | ||
أُضيف في الإصدار: 0.9.4. | أُضيف في الإصدار: 0.9.4. | ||
يُطلق الحدث 'end' عندما لا يكون هناك المزيد من البيانات لتُقرَأ من المجرى. | يُطلق الحدث <code>'end'</code> عندما لا يكون هناك المزيد من البيانات لتُقرَأ من المجرى. | ||
لن يُطلق الحدث 'end' مالم تُستهلك البيانات بالكامل. يمكن أن يُحقَّق ذلك بقلب المجرى إلى نمط التدفق، أو باستدعاء stream.read() مرارًا حتى يتم استهلاك كل البيانات.<syntaxhighlight lang="javascript"> | '''لن يُطلق''' الحدث <code>'end'</code> مالم تُستهلك البيانات بالكامل. يمكن أن يُحقَّق ذلك بقلب المجرى إلى نمط التدفق، أو باستدعاء [[Node.js/stream#readable.read.28.5Bsize.5D.E2.80.8E.29.E2.80.8E|stream.read()]] مرارًا حتى يتم استهلاك كل البيانات.<syntaxhighlight lang="javascript"> | ||
const readable = getReadableStreamSomehow(); | const readable = getReadableStreamSomehow(); | ||
readable.on('data', (chunk) => { | readable.on('data', (chunk) => { | ||
سطر 523: | سطر 501: | ||
</syntaxhighlight> | </syntaxhighlight> | ||
==== الحدث: 'error' ==== | ===== الحدث: <code>'error'</code> ===== | ||
أُضيف في الإصدار: 0.9.4. | أُضيف في الإصدار: 0.9.4. | ||
* <Error> | * [[JavaScript/Error|<Error>]] | ||
ربما يُطلق الحدث 'error' من قبل إجراء مجرى قابل للقراءة (Readable) في أي وقت. عادةً، قد يحصل ذلك إذا كان المجرى الأساسي غير قادر على توليد بيانات بسبب فشل داخلي أساسي، أو عندما يحاول إجراء المجرى دفع قطعة بيانات غير صالحة. | ربما يُطلق الحدث <code>'error'</code> من قبل إجراء مجرى قابل للقراءة (<code>Readable</code>) في أي وقت. عادةً، قد يحصل ذلك إذا كان المجرى الأساسي غير قادر على توليد بيانات بسبب فشل داخلي أساسي، أو عندما يحاول إجراء المجرى دفع قطعة بيانات غير صالحة. | ||
سيُمرَّر إلى دالة رد نداء المُنصت كائنٌ من النوع Error فقط. | سيُمرَّر إلى دالة رد نداء المُنصت كائنٌ من النوع <code>Error</code> فقط. | ||
==== الحدث: 'readable' ==== | ===== الحدث: <code>'readable'</code> ===== | ||
سجل التغييرات | سجل التغييرات | ||
{| class="wikitable" | {| class="wikitable" | ||
!الإصدار | !الإصدار | ||
!التغييرات | !التغييرات | ||
! | ! rowspan="4" | | ||
|- | |- | ||
|10.0.0 | |10.0.0 | ||
|يُطلق الحدث 'readable' دائمًا في النبضة التالية بعد أن يُستدعى .push() | |يُطلق الحدث <code>'readable'</code> دائمًا في النبضة التالية بعد أن يُستدعى <code>.push()</code> | ||
|- | |- | ||
|10.0.0 | |10.0.0 | ||
|يتطلب استخدام 'readable' استدعاءَ .read() | |يتطلب استخدام <code>'readable'</code> استدعاءَ <code>.read()</code> | ||
|- | |- | ||
|0.9.4 | |0.9.4 | ||
|أُضيف في الإصدار: 0.9.4. | |أُضيف في الإصدار: 0.9.4. | ||
|} | |} | ||
يُطلق التابع 'readable' عندما يكون هناك بيانات متوفرة للقراءة من المجرى. في بعض الحالات، سوف يسبب ربط مُنصت بالحدث 'readable' أن تُقرَأ بعض كمية البيانات إلى ذاكرة مؤقتة داخلية.<syntaxhighlight lang="javascript"> | يُطلق التابع <code>'readable'</code> عندما يكون هناك بيانات متوفرة للقراءة من المجرى. في بعض الحالات، سوف يسبب ربط مُنصت بالحدث <code>'readable'</code> أن تُقرَأ بعض كمية البيانات إلى ذاكرة مؤقتة داخلية.<syntaxhighlight lang="javascript"> | ||
const readable = getReadableStreamSomehow(); | const readable = getReadableStreamSomehow(); | ||
readable.on('readable', function() { | readable.on('readable', function() { | ||
يوجد بعض البيانات لتُقرأ الآن | //يوجد بعض البيانات لتُقرأ الآن | ||
let data; | let data; | ||
سطر 565: | سطر 536: | ||
} | } | ||
}); | }); | ||
</syntaxhighlight>سوف يُطلق الحدث 'readable' أيضًا حالما يتم التوصل إلى نهاية بيانات المجرى ولكن قبل أن يُطلق الحدث 'end'. | </syntaxhighlight>سوف يُطلق الحدث <code>'readable'</code> أيضًا حالما يتم التوصل إلى نهاية بيانات المجرى ولكن قبل أن يُطلق الحدث <code>'end'</code>. | ||
فعليًا، يشير الحدث 'readable' أن المجرى يملك معلومات جديدة وهي: إمّا بيانات جديدة متوفرة أو تم الوصول إلى نهاية المجرى. في الحالة الأولى، سوف يعيد stream.read() البيانات المتوفرة. في الحالة الثانية، سوف يعيد stream.read() القيمة null . على سبيل المثال، في المثال التالي، foo.txt هو ملف فارغ:<syntaxhighlight lang="javascript"> | فعليًا، يشير الحدث <code>'readable'</code> أن المجرى يملك معلومات جديدة وهي: إمّا بيانات جديدة متوفرة أو تم الوصول إلى نهاية المجرى. في الحالة الأولى، سوف يعيد [[Node.js/stream#readable.read.28.5Bsize.5D.E2.80.8E.29.E2.80.8E|stream.read()]] البيانات المتوفرة. في الحالة الثانية، سوف يعيد [[Node.js/stream#readable.read.28.5Bsize.5D.E2.80.8E.29.E2.80.8E|stream.read()]] القيمة <code>null</code> . على سبيل المثال، في المثال التالي، <code>foo.txt</code> هو ملف فارغ:<syntaxhighlight lang="javascript"> | ||
const fs = require('fs'); | const fs = require('fs'); | ||
const rr = fs.createReadStream('foo.txt'); | const rr = fs.createReadStream('foo.txt'); | ||
سطر 582: | سطر 553: | ||
end | end | ||
</syntaxhighlight>بشكل عام، فإنّ آليات readable.pipe() والحدث 'data' هي أسهل للفهم من الحدث 'readable'. ولكن، معالجة 'readable' قد تنتج زيادة بمعدل الانتاجية . | </syntaxhighlight>بشكل عام، فإنّ آليات <code>readable.pipe()</code> والحدث <code>'data'</code> هي أسهل للفهم من الحدث <code>'readable'</code>. ولكن، معالجة <code>'readable'</code> قد تنتج زيادة بمعدل الانتاجية . | ||
إذا استُخدم كلا الحدثين 'readable' و 'data' في نفس الوقت، فإن الحدث 'readable' يأخذ الأولوية في التحكم بالتدفق؛ أي أنّ الحدث 'data' سوف يُطلق فقط إذا استُدعي stream.read(). قد تصبح قيمة الخاصية | إذا استُخدم كلا الحدثين <code>'readable'</code> و [[Node.js/stream#.D8.A7.D9.84.D8.AD.D8.AF.D8.AB:.27data.27|'data']] في نفس الوقت، فإن الحدث <code>'readable'</code> يأخذ الأولوية في التحكم بالتدفق؛ أي أنّ الحدث <code>'data'</code> سوف يُطلق فقط إذا استُدعي [[Node.js/stream#readable.read.28.5Bsize.5D.E2.80.8E.29.E2.80.8E|stream.read()]]. قد تصبح قيمة الخاصية <code>readableFlowing</code> هي <code>false</code> . إذا كان هناك مُنصتات للحدث <code>'data'</code> عندما يُزال الحدث <code>'readable'</code>، سيبدأ المجرى بالتدفق؛ أي أن الأحداث <code>'data'</code> سوف تُطلق دون استدعاء <code>.resume()</code> | ||
==== readable.destroy([error]) ==== | ===== <code>readable.destroy([error])</code> ===== | ||
أُضيف في الإصدار: 8.0.0. | أُضيف في الإصدار: 8.0.0. | ||
* Error> | * [[JavaScript/Error|<Error>]] :<code>error</code> الخطأ الحاصل الذي سيُمرَّر عند تحميل الحدث 'error'. | ||
* القيمة المُعادة: <this> | * القيمة المُعادة: [[ss\fdvsf|<this>]] | ||
يهدم التابع المجرى ويطلق الحدث 'error' و 'close'. بعد هذا الاستدعاء، سوف يحرر المجرى القابل للقراءة أيّة موارد داخلية وسوف تُتجاهل الاستدعاءات اللاحقة للتابع push() . لا ينبغي على المنفذين إعادة تعريف هذا التابع، ولكن يمكنهم تطبيق readable._destroy() بدلًا عنه. | يهدم التابع المجرى ويطلق الحدث <code>'error'</code> و <code>'close'</code>. بعد هذا الاستدعاء، سوف يحرر المجرى القابل للقراءة أيّة موارد داخلية وسوف تُتجاهل الاستدعاءات اللاحقة للتابع <code>push()</code> . لا ينبغي على المنفذين إعادة تعريف هذا التابع، ولكن يمكنهم تطبيق [[Node.js/stream#readable. destroy.28err.2C callback.29.E2.80.8E|readable._destroy()]] بدلًا عنه. | ||
==== readable.isPaused() ==== | ===== <code>readable.isPaused()</code> ===== | ||
أُضيف في الإصدار: 0.11.14. | أُضيف في الإصدار: 0.11.14. | ||
* القيمة المُعادة: <boolean> | * القيمة المُعادة: [[JavaScript/Boolean|<boolean>]] | ||
يعيد التابع readable.isPaused() حالة التشغيل الحالية للمجرى Readable . يُستخدم هذا التابع بالمقام الأول من قبل آليات ترتكز على التابع readable.pipe(). في معظم الحالات الطبيعية، لا يوجد حاجة لاستخدام هذا التابع بشكل مباشر.<syntaxhighlight lang="javascript"> | يعيد التابع <code>readable.isPaused()</code> حالة التشغيل الحالية للمجرى <code>Readable</code> . يُستخدم هذا التابع بالمقام الأول من قبل آليات ترتكز على التابع <code>readable.pipe()</code>. في معظم الحالات الطبيعية، لا يوجد حاجة لاستخدام هذا التابع بشكل مباشر.<syntaxhighlight lang="javascript"> | ||
const readable = new stream.Readable(); | const readable = new stream.Readable(); | ||
سطر 606: | سطر 577: | ||
</syntaxhighlight> | </syntaxhighlight> | ||
==== readable.pause() ==== | ===== <code>readable.pause()</code> ===== | ||
أضيف في الإصدار: 0.9.4. | أضيف في الإصدار: 0.9.4. | ||
القيمة المُعادة: <this> | القيمة المُعادة: [[dsfzvdgdd|<this>]] | ||
سوف يسبب التابع readable.pause() وقف إطلاق أحداث 'data' في نمط التدفق، مع إخراج المجرى من نمط التدفق. أية بيانات تصبح متوفرة ستبقى في الذاكرة المؤقتة الداخلية.<syntaxhighlight lang="javascript"> | سوف يسبب التابع <code>readable.pause()</code> وقف إطلاق أحداث [[Node.js/stream#.D8.A7.D9.84.D8.AD.D8.AF.D8.AB:.27data.27|'data']] في نمط التدفق، مع إخراج المجرى من نمط التدفق. أية بيانات تصبح متوفرة ستبقى في الذاكرة المؤقتة الداخلية.<syntaxhighlight lang="javascript"> | ||
const readable = getReadableStreamSomehow(); | const readable = getReadableStreamSomehow(); | ||
readable.on('data', (chunk) => { | readable.on('data', (chunk) => { | ||
سطر 623: | سطر 594: | ||
}); | }); | ||
</syntaxhighlight>لا يملك التابع readable.pause() أي تأثير إذا كان هناك مُنصت للحدث 'readable'. | </syntaxhighlight>لا يملك التابع <code>readable.pause()</code> أي تأثير إذا كان هناك مُنصت للحدث <code>'readable'</code>. | ||
==== readable.pipe(destination[, options]) ==== | ===== <code>readable.pipe(destination[, options])</code> ===== | ||
أُضيف في الإصدار: 0.9.4. | أُضيف في الإصدار: 0.9.4. | ||
* <stream.Writable> :destination الوجهة التي ستُكتَب فيها البيانات. | * [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Writable|<stream.Writable>]] :<code>destination</code> الوجهة التي ستُكتَب فيها البيانات. | ||
* Object> :options> خيارات النقل. | * [[JavaScript/Object|<Object>]] :<code>options</code> خيارات النقل. | ||
** boolean> : | ** [[JavaScript/Boolean|<boolean>]] :<code>end</code> إنهاء الكاتب عندما ينتهي القارئ. '''القيمة الافتراضية:''' <code>true</code>. | ||
* القيمة المُعادة: <stream.Writable> تعاد الوجهة destination سامحةً بسلسلة من الأنابيب والممرات إذا كانت المجاري من النوع المزدوج (Duplex) أو مجاري تحويل (Transform). | * القيمة المُعادة: [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Writable|<stream.Writable>]] تعاد الوجهة destination سامحةً بسلسلة من الأنابيب والممرات إذا كانت المجاري من النوع [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Duplex|المزدوج (Duplex)]] أو مجاري [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81: stream.Transform|تحويل (Transform)]]. | ||
يربط التابع readable.pipe() مجرًى قابلًا للكتابة مع مجرًى قابلٍ للقراءة مسببًا قلبه تلقائيّا إلى نمط التدفق ودفع جميع بياناته إلى المجرى القابل للكتابة المرتبط به. أي كأن هذا التابع يضع أنبوبًا بين المجريين لتتدفق البيانات من أحدهما إلى الآخر عبره. سيضبط تدفق البيانات تلقائيًا لذلك لن يُغمر مجرى Writable الهدف من قبل مجرى قابل للقراءة(Readable) أسرع منه. | يربط التابع <code>readable.pipe()</code> مجرًى [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Writable|قابلًا للكتابة]] مع مجرًى قابلٍ للقراءة (<code>Readable</code>) مسببًا قلبه تلقائيّا إلى نمط التدفق ودفع جميع بياناته إلى المجرى [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Writable|القابل للكتابة]] المرتبط به. أي كأن هذا التابع يضع أنبوبًا بين المجريين لتتدفق البيانات من أحدهما إلى الآخر عبره. سيضبط تدفق البيانات تلقائيًا لذلك لن يُغمر مجرى <code>Writable</code> الهدف من قبل مجرى قابل للقراءة(<code>Readable</code>) أسرع منه. | ||
ينقل المثال التالي كل البيانات من المجرى readable إلى ملف اسمه file.txt:<syntaxhighlight lang="javascript"> | ينقل المثال التالي كل البيانات من المجرى <code>readable</code> إلى ملف اسمه <code>file.txt</code>:<syntaxhighlight lang="javascript"> | ||
const fs = require('fs'); | const fs = require('fs'); | ||
const readable = getReadableStreamSomehow(); | const readable = getReadableStreamSomehow(); | ||
سطر 640: | سطر 611: | ||
تذهب كل البيانات من المجرى القابل للقراءة إلى | //'file.txt' تذهب كل البيانات من المجرى القابل للقراءة إلى | ||
readable.pipe(writable); | readable.pipe(writable); | ||
</syntaxhighlight>من الممكن ربط عدة مجاري قابلة للكتابة مع مجرى واحد قابل للقراءة. | </syntaxhighlight>من الممكن ربط عدة مجاري قابلة للكتابة (<code>Writable</code>) مع مجرى واحد قابل للقراءة (<code>Readable</code>). | ||
يعيد التابع readable.pipe() مرجعًا إلى مجرى الوجهة جاعلًا من الممكن إقامة سلسلة من المجاري المتصلة ببعضها بأنابيب:<syntaxhighlight lang="javascript"> | يعيد التابع <code>readable.pipe()</code> مرجعًا إلى مجرى الوجهة جاعلًا من الممكن إقامة سلسلة من المجاري المتصلة ببعضها بأنابيب:<syntaxhighlight lang="javascript"> | ||
const fs = require('fs'); | const fs = require('fs'); | ||
const r = fs.createReadStream('file.txt'); | const r = fs.createReadStream('file.txt'); | ||
سطر 653: | سطر 624: | ||
r.pipe(z).pipe(w); | r.pipe(z).pipe(w); | ||
</syntaxhighlight>بشكل افتراضي، يُستدعى stream.end() على مجرى الوجهة القابل للكتابة (Writable) عندما يطلق مجرى Readable المصدر الحدث 'end'. لذلك لن تكون الوجهة قابلةً للكتابة بعد الآن. لتعطيل هذا السلوك الافتراضي، يمكن أن تمرير القيمة false إلى الخيار end مسبّبًا بقاء الوجهة مفتوحة:<syntaxhighlight lang="javascript"> | </syntaxhighlight>بشكل افتراضي، يُستدعى [[Node.js/stream#writable.end.28.5Bchunk.5D.5B.2C encoding.5D.5B.2C callback.5D.E2.80.8E.29.E2.80.8E|stream.end()]] على مجرى الوجهة القابل للكتابة (<code>Writable</code>) عندما يطلق مجرى <code>Readable</code> المصدر الحدث [[Node.js/stream#.D8.A7.D9.84.D8.AD.D8.AF.D8.AB: .27end.27|'end']]. لذلك لن تكون الوجهة قابلةً للكتابة بعد الآن. لتعطيل هذا السلوك الافتراضي، يمكن أن تمرير القيمة <code>false</code> إلى الخيار <code>end</code> مسبّبًا بقاء الوجهة مفتوحة:<syntaxhighlight lang="javascript"> | ||
reader.pipe(writer, { end: false }); | reader.pipe(writer, { end: false }); | ||
reader.on('end', () => { | reader.on('end', () => { | ||
writer.end('Goodbye\n'); | writer.end('Goodbye\n'); | ||
}); | }); | ||
</syntaxhighlight>أحد التحذيرات المهمة أنّه إذا أطلق المجرى Readable خطأً أثناء المعالجة، لن يُغلق مجرى الوجهة Writable تلقائيًا، لذا من الضروري في هذه الحالة إغلاق كل مجرى يدويًا لمنع التسريب في الذاكرة. | </syntaxhighlight>أحد التحذيرات المهمة أنّه إذا أطلق المجرى <code>Readable</code> خطأً أثناء المعالجة، لن يُغلق مجرى الوجهة <code>Writable</code> تلقائيًا، لذا من الضروري في هذه الحالة إغلاق كل مجرى يدويًا لمنع التسريب في الذاكرة. | ||
لن يغلق المجريان process.stderr و process.stdout القابلان للكتابة (Writable) أبدًا حتى تنتهي عملية Node.js بغض النظر عن الخيارات المحدّدة. | لن يغلق المجريان [[Node.js/process#process.stderr|process.stderr]] و [[Node.js/process#process.stdout|process.stdout]] القابلان للكتابة (<code>Writable</code>) أبدًا حتى تنتهي عملية Node.js بغض النظر عن الخيارات المحدّدة. | ||
==== readable.read([size]) ==== | ===== <code>readable.read([size])</code> ===== | ||
أضيف في الإصدار: 0.9.4. | أضيف في الإصدار: 0.9.4. | ||
* <number> :size وسيط اختياري يحدد كمية البيانات للقراءة. | * [[JavaScript/Number|<number>]] :<code>size</code> وسيط اختياري يحدد كمية البيانات للقراءة. | ||
* القيمة المعادة: <string> | <Buffer> | <null> | <any> | * القيمة المعادة: [[JavaScript/String|<string>]] | [[Node.js/buffer#.D8.A7.D9.84.D8.B5.D9.86.D9.81 Buffer|<Buffer>]] | [[JavaScript/null|<null>]] | [[sgzsrfgsdxfzsd|<any>]] | ||
يسحب التابع readable.read() بعض البيانات من الذاكرة المؤقتة الداخلية ويعيدها. إذا لم يكن هناك بيانات متوافرة للقراءة، يعيد التابع القيمة null. بشكل افتراضي، ستعاد البيانات في كائن من النوع Buffer إلّا إذا حُدد الترميز باستخدام التابع readable.setEncoding() أو إذا كان المجرى يشتغل في نمط الكائن. | يسحب التابع <code>readable.read()</code> بعض البيانات من الذاكرة المؤقتة الداخلية ويعيدها. إذا لم يكن هناك بيانات متوافرة للقراءة، يعيد التابع القيمة <code>null</code>. بشكل افتراضي، ستعاد البيانات في كائن من النوع <code>Buffer</code> إلّا إذا حُدد الترميز باستخدام التابع <code>readable.setEncoding()</code> أو إذا كان المجرى يشتغل في نمط الكائن. | ||
يحدد الوسيط الاختياري size عدد البايتات المحدد للقراءة. إذا لم يكن هنالك بايتات بالحجم size متوافرة للقراءة، ستُعاد القيمة null إلّا إذا كان المجرى قد انتهى. في هذه الحالة، ستُعاد كل البيانات المتبقية في ذاكرة التخزين المؤقت الداخلية. | يحدد الوسيط الاختياري <code>size</code> عدد البايتات المحدد للقراءة. إذا لم يكن هنالك بايتات بالحجم <code>size</code> متوافرة للقراءة، ستُعاد القيمة <code>null</code> إلّا إذا كان المجرى قد انتهى. في هذه الحالة، ستُعاد كل البيانات المتبقية في ذاكرة التخزين المؤقت الداخلية. | ||
إذا لم يكن الوسيط size محدّدًا، ستُعاد كل البيانات المحتواة في ذاكرة التخزين المؤقت الداخلية. | إذا لم يكن الوسيط <code>size</code> محدّدًا، ستُعاد كل البيانات المحتواة في ذاكرة التخزين المؤقت الداخلية. | ||
ينبغي أن يُستدعى التابع readable.read() على مجاري Readable المُشغّلة في نمط التوقف المؤقت فقط، في نمط التدفق، يُستدعى readable.read() تلقائيًا حتى فراغ ذاكرة التخزين المؤقت الداخلية بالكامل.<syntaxhighlight lang="javascript"> | ينبغي أن يُستدعى التابع <code>readable.read()</code> على مجاري <code>Readable</code> المُشغّلة في نمط التوقف المؤقت فقط، في نمط التدفق، يُستدعى <code>readable.read()</code> تلقائيًا حتى فراغ ذاكرة التخزين المؤقت الداخلية بالكامل.<syntaxhighlight lang="javascript"> | ||
const readable = getReadableStreamSomehow(); | const readable = getReadableStreamSomehow(); | ||
readable.on('readable', () => { | readable.on('readable', () => { | ||
سطر 681: | سطر 652: | ||
}) | }) | ||
</syntaxhighlight>سيعيد المجرى Readable في نمط الكائن عنصرًا وحيدًا دائمًا من استدعاء readable.read(size) بغض النظر عن عن قيمة الوسيط size. | </syntaxhighlight>سيعيد المجرى <code>Readable</code> في نمط الكائن عنصرًا وحيدًا دائمًا من استدعاء [[Node.js/stream#readable.read.28.5Bsize.5D.E2.80.8E.29.E2.80.8E|readable.read(size)]] بغض النظر عن عن قيمة الوسيط <code>size</code>. | ||
إذا أعاد التابع readable.read() قطعة من البيانات، سيُطلق أيضًا الحدث 'data'. | إذا أعاد التابع <code>readable.read()</code> قطعة من البيانات، سيُطلق أيضًا الحدث <code>'data'</code>. | ||
استدعاء stream.read([size]) بعد أن يكون الحدث 'end' قد أُطلق سيعيد null. لن تُطلَق أي أخطاء وقت التشغيل (runtime). | استدعاء [[Node.js/stream#readable.read.28.5Bsize.5D.E2.80.8E.29.E2.80.8E|stream.read([size])]] بعد أن يكون الحدث [[Node.js/stream#.D8.A7.D9.84.D8.AD.D8.AF.D8.AB: .27end.27|'end']] قد أُطلق سيعيد <code>null</code>. لن تُطلَق أي أخطاء وقت التشغيل (runtime). | ||
==== readable.readableHighWaterMark ==== | ===== <code>readable.readableHighWaterMark</code> ===== | ||
أضيفت في الإصدار: 9.3.0. | أضيفت في الإصدار: 9.3.0. | ||
* القيمة المعادة: <number> | * القيمة المعادة: [[JavaScript/Number|<number>]] | ||
تعيد قيمة الخاصية highWaterMark المُمرّرة عند إنشاء مجرى Readable هذا. | تعيد قيمة الخاصية <code>highWaterMark</code> المُمرّرة عند إنشاء مجرى <code>Readable</code> هذا. | ||
===== readable.readableLength ===== | ===== <code>readable.readableLength</code> ===== | ||
أُضيفت في الإصدار:9.4.0. | أُضيفت في الإصدار:9.4.0. | ||
* القيمة المعادة:<number> | * القيمة المعادة:[[JavaScript/Number|<number>]] | ||
تحوي هذه الخاصية عدد البايتات (أو الكائنات) الجاهزة للقراءة في الطابور. تقدم هذه القيمة البيانات الداخلية المخزنة بما يتوافق مع حالة highWaterMark. | تحوي هذه الخاصية عدد البايتات (أو الكائنات) الجاهزة للقراءة في الطابور. تقدم هذه القيمة البيانات الداخلية المخزنة بما يتوافق مع حالة <code>highWaterMark</code>. | ||
==== readable.resume() ==== | ===== <code>readable.resume()</code> ===== | ||
سجل التغييرات | سجل التغييرات | ||
سطر 703: | سطر 674: | ||
!الإصدار | !الإصدار | ||
!التغييرات | !التغييرات | ||
! | ! rowspan="3" | | ||
|- | |- | ||
|10.0.0 | |10.0.0 | ||
|لا يملك resume() تأثير إذا كان هناك استماع للحدث 'readable'. | |لا يملك <code>resume()</code> تأثير إذا كان هناك استماع للحدث <code>'readable'</code>. | ||
|- | |- | ||
|0.9.4 | |0.9.4 | ||
|أُضيف في الإصدار 0.9.4. | |أُضيف في الإصدار 0.9.4. | ||
|} | |} | ||
* القيمة المعادة: <this> | * القيمة المعادة: [[ئبءرئسيبلئس|<this>]] | ||
يسبب التابع readable.resume() إيقاف صريح لمجرى Readable لاستئناف إطلاق أحداث 'data'، قالبًا المجرى إلى نمط التدفق. | يسبب التابع <code>readable.resume()</code> إيقاف صريح لمجرى <code>Readable</code> لاستئناف إطلاق أحداث [[Node.js/stream#.D8.A7.D9.84.D8.AD.D8.AF.D8.AB:.27data.27|'data']]، قالبًا المجرى إلى نمط التدفق. | ||
يمكن أن يُستخدم التابع readable.resume() لاستهلاك كامل للبيانات من المجرى دون معالجة أي منها فعليًا.<syntaxhighlight lang="javascript"> | يمكن أن يُستخدم التابع <code>readable.resume()</code> لاستهلاك كامل للبيانات من المجرى دون معالجة أي منها فعليًا.<syntaxhighlight lang="javascript"> | ||
getReadableStreamSomehow() | getReadableStreamSomehow() | ||
.resume() | .resume() | ||
سطر 731: | سطر 692: | ||
}); | }); | ||
</syntaxhighlight>لا يملك التابع readable.resume() تأثير إذا كان هناك مُنصت للحدث 'readable'. | </syntaxhighlight>لا يملك التابع <code>readable.resume()</code> تأثير إذا كان هناك مُنصت للحدث <code>'readable'</code>. | ||
==== readable.setEncoding(encoding) ==== | ===== <code>readable.setEncoding(encoding)</code> ===== | ||
أُضيف في الإصدار: 0.9.4. | أُضيف في الإصدار: 0.9.4. | ||
* <string> :encoding الترميز المراد استعماله. | * [[JavaScript/String|<string>]] :<code>encoding</code> الترميز المراد استعماله. | ||
* القيمة المعادة: <this> | * القيمة المعادة: [[باءالءؤاء|<this>]] | ||
يضبط التابع readable.setEncoding() ترميز المحارف للبيانات المقروءة من المجرى Readable. | يضبط التابع <code>readable.setEncoding()</code> ترميز المحارف للبيانات المقروءة من المجرى <code>Readable</code>. | ||
بشكل افتراضي، لا يُخصص أي ترميز وستعاد بيانات المجرى في كائنات من النوع | بشكل افتراضي، لا يُخصص أي ترميز وستعاد بيانات المجرى في كائنات من النوع <code>Buffer</code>، ضبط الترميز يتسبب بأن تُعاد بيانات المجرى كسلسلة نصية مرمّزة بالترميز المحدد بدلًا من إعادتها في كائنات من النوع <code>Buffer</code>. على سبيل المثال، سيسسب استدعاء <code>readable.setEncoding('utf8')</code> بأن تُفسر بيانات الخرج كبيانات مرمزة بالترميز UTF-8، وتُمرر كسلاسل نصية. سوف يسبب استدعاء <code>readable.setEncoding('hex')</code> أن تُرمّز البيانات بشكل سلسلة نصية ستة عشرية. | ||
سيعالج المجرى Readable المحارف متعددة البايتات المُستلمة خلال المجرى بصورة صحيحة وإلّا قد تصبح مرمّزة بشكل غير صحيح إذا سُحبت من المجرى ببساطة في كائنات من النوع Buffer.<syntaxhighlight lang="javascript"> | سيعالج المجرى <code>Readable</code> المحارف متعددة البايتات المُستلمة خلال المجرى بصورة صحيحة وإلّا قد تصبح مرمّزة بشكل غير صحيح إذا سُحبت من المجرى ببساطة في كائنات من النوع <code>Buffer</code>.<syntaxhighlight lang="javascript"> | ||
const readable = getReadableStreamSomehow(); | const readable = getReadableStreamSomehow(); | ||
readable.setEncoding('utf8'); | readable.setEncoding('utf8'); | ||
سطر 748: | سطر 709: | ||
console.log('got %d characters of string data', chunk.length); | console.log('got %d characters of string data', chunk.length); | ||
}); | }); | ||
</syntaxhighlight> | </syntaxhighlight> | ||
===== <code>readable.unpipe([destination])</code> ===== | |||
أضيف في الإصدار:0.9.4. | أضيف في الإصدار:0.9.4. | ||
* <stream.Writable> : | * [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Writable|<stream.Writable>]] :<code>destination</code> مجرى محدد اختياريًا | ||
* القيمة المُعادة: <this> | * القيمة المُعادة: [[يسالاسثقلسيءبق|<this>]] | ||
يفصل التابع readable.unpipe() مجرى Writable المربوط مؤخرًا باستخدام التابع | يفصل التابع <code>readable.unpipe()</code> مجرى <code>Writable</code> المربوط مؤخرًا باستخدام التابع <nowiki/>[[Node.js/stream#readable.pipe.28destination.5B.2C options.5D.E2.80.8E.29.E2.80.8E|stream.pipe()]]. | ||
إذا لم يُحدد مجرى الوجهة (destination)، ستفصل حينذاك كل الأنابيب(الممرات). | إذا لم يُحدد مجرى الوجهة (<code>destination</code>)، ستفصل حينذاك كل الأنابيب(الممرات). | ||
إذا كان destination محددًا، ولكن لا يوجد أنابيب(قنوات) مُنصّبة عليه، لن يفعل التابع حينذاك شيئًا.<syntaxhighlight lang="javascript"> | إذا كان <code>destination</code> محددًا، ولكن لا يوجد أنابيب(قنوات) مُنصّبة عليه، لن يفعل التابع حينذاك شيئًا.<syntaxhighlight lang="javascript"> | ||
const fs = require('fs'); | const fs = require('fs'); | ||
const readable = getReadableStreamSomehow(); | const readable = getReadableStreamSomehow(); | ||
سطر 764: | سطر 726: | ||
تذهب كل البيانات من المجرى القابل للقراءة | // 'file.txt' إلى readable تذهب كل البيانات من المجرى القابل للقراءة | ||
ولكن فقط للثانية الأولى | //ولكن فقط للثانية الأولى | ||
readable.pipe(writable); | readable.pipe(writable); | ||
سطر 777: | سطر 739: | ||
</syntaxhighlight> | </syntaxhighlight> | ||
==== readable.unshift(chunk) ==== | ===== <code>readable.unshift(chunk)</code> ===== | ||
سجل التغييرات | سجل التغييرات | ||
سطر 783: | سطر 745: | ||
!الإصدار | !الإصدار | ||
!التغييرات | !التغييرات | ||
! | ! rowspan="4" | | ||
|- | |- | ||
|8.0.0 | |8.0.0 | ||
|يمكن أن يكون الآن الوسيط | |يمكن أن يكون الآن الوسيط <code>chunk</code> نسخةَ <code>Uint8Array</code>. | ||
|- | |- | ||
|0.9.11 | |0.9.11 | ||
|أُضيف في الإصدار: 0.9.11. | |أُضيف في الإصدار: 0.9.11. | ||
|- | |- | ||
| | | colspan="2" | | ||
| | |||
|} | |} | ||
* Buffer> | <Uint8Array> | <string> | <any> | * [[Node.js/buffer#.D8.A7.D9.84.D8.B5.D9.86.D9.81 Buffer|<Buffer>]] | [[تسيابخنباتهخاثي|<Uint8Array>]] | [[JavaScript/String|<string>]] | [[dfszgszgf|<any>]] :chunk قطع البيانات المراد منعها من القفز (الإزاحة) فوق الطابور. من أجل المجاري غير المشغّلة في نمط الكائن، يجب أن تكون <code>chunk</code> سلسلة نصية أو كائنًا من النوع <code>Buffer</code> أو النوع <code>Uint8Array</code>. من أجل المجاري في نمط الكائن، يمكن أن يقبل <code>chunk</code> أي نوع من أنواع بيانات JavaScript باستثناء <code>null</code>. | ||
يدفع التابع readable.unshift() قطع البيانات خلفًا إلى ذاكرة التخزين المؤقتة الداخلية. هذا مفيد في حالات معيّنة تُقرَأ فيها بيانات محدَّدة لا حاجة لقراءتها من المجرى، لذلك يُمكن أن تُزاح هذه البيانات لتُمرر أخرى إلى طرف آخر. | يدفع التابع <code>readable.unshift()</code> قطع البيانات خلفًا إلى ذاكرة التخزين المؤقتة الداخلية. هذا مفيد في حالات معيّنة تُقرَأ فيها بيانات محدَّدة لا حاجة لقراءتها من المجرى، لذلك يُمكن أن تُزاح هذه البيانات لتُمرر أخرى إلى طرف آخر. | ||
هذا مفيد في حالات معينة حيث يُستهلك المجرى بشيفرات تتطلب "عدم استهلاك" لبعض كمية البيانات المسحوبة بشكل مثالي من المصدر، لذلك يمكن أن تُمرر البيانات إلى بعض الأطراف الأخرى. | هذا مفيد في حالات معينة حيث يُستهلك المجرى بشيفرات تتطلب "عدم استهلاك" لبعض كمية البيانات المسحوبة بشكل مثالي من المصدر، لذلك يمكن أن تُمرر البيانات إلى بعض الأطراف الأخرى. | ||
لايمكن أن يُستدعى التابع stream.unshift(chunk) بعد أن أُطلق الحدث 'end' أو سوف يُرمى خطأ وقت التشغيل. | لايمكن أن يُستدعى التابع <code>stream.unshift(chunk)</code> بعد أن أُطلق الحدث [[Node.js/stream#.D8.A7.D9.84.D8.AD.D8.AF.D8.AB: .27end.27|'end']] أو سوف يُرمى خطأ وقت التشغيل. | ||
ينبغي على المطورين المستخدمين للتابع stream.unshift() التبديل إلى استعمال مجرى التحويل (Transform) بدلًا عن المجرى Readable. انظر مقطع الواجهات البرمجية لمنفذي المجاري للمزيد من المعلومات.<syntaxhighlight lang="javascript"> | ينبغي على المطورين المستخدمين للتابع <code>stream.unshift()</code> التبديل إلى استعمال مجرى [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81: stream.Transform|التحويل (Transform)]] بدلًا عن المجرى Readable. انظر مقطع [[Node.js/stream#.D8.A7.D9.84.D9.88.D8.A7.D8.AC.D9.87.D8.A7.D8.AA .D8.A7.D9.84.D8.A8.D8.B1.D9.85.D8.AC.D9.8A.D8.A9 .D9.84.D9.85.D9.86.D9.81.D8.B0.D9.8A .D8.A7.D9.84.D9.85.D8.AC.D8.A7.D8.B1.D9.8A|الواجهات البرمجية لمنفذي المجاري]] للمزيد من المعلومات.<syntaxhighlight lang="javascript"> | ||
// \n\nانتزاع العنوان المحدّد ب | |||
//إذا حصلت على الكثير unshift()استخدم | |||
استدعِ دالة رد النداء مع (الخطأ، المجرى، العنوان | //(error, header, stream) (استدعِ دالة رد النداء مع (الخطأ، المجرى، العنوان | ||
const { StringDecoder } = require('string_decoder'); | const { StringDecoder } = require('string_decoder'); | ||
سطر 831: | سطر 785: | ||
const buf = Buffer.from(remaining, 'utf8'); | const buf = Buffer.from(remaining, 'utf8'); | ||
stream.removeListener('error', callback); | stream.removeListener('error', callback); | ||
// | |||
// unshift() قبل استدعاء 'readable' أزل مُنصت | |||
stream.removeListener('readable', onReadable); | stream.removeListener('readable', onReadable); | ||
سطر 842: | سطر 795: | ||
callback(null, header, stream); | callback(null, header, stream); | ||
} else { | } else { | ||
//لا يزال يقرأ العنوان | //لا يزال يقرأ العنوان | ||
header += str; | header += str; | ||
سطر 850: | سطر 803: | ||
} | } | ||
</syntaxhighlight>خلافًا للتابع stream.push(chunk)، لن ينهي التابع stream.unshift(chunk) عملية القراءة بإعادة تصفير حالة القراءة الداخلية للمجرى. يمكن أن يسبب هذا نتائج غير متوقعة إذا استُدعي readable.unshift() أثناء القراءة (أي من داخل تنفيذ stream._read() على مجرى مخصص). | </syntaxhighlight>خلافًا للتابع [[Node.js/stream#readable.push.28chunk.5B.2C encoding.5D.E2.80.8E.29.E2.80.8E|stream.push(chunk)]]، لن ينهي التابع <code>stream.unshift(chunk)</code> عملية القراءة بإعادة تصفير حالة القراءة الداخلية للمجرى. يمكن أن يسبب هذا نتائج غير متوقعة إذا استُدعي <code>readable.unshift()</code> أثناء القراءة (أي من داخل تنفيذ [[Node.js/stream#readable. read.28size.29.E2.80.8E|stream._read()]] على مجرى مخصص). | ||
اتباع استدعاء التابع readable.unshift() بالتابع stream.push(<nowiki>''</nowiki>) مباشرةً سوف يعيد تصفير حالة القراءة بشكل مناسب، ولكن الأفضل ببساطة هو تجنب استدعاء readable.unshift() أثناء عملية تنفيذ القراءة. | اتباع استدعاء التابع <code>readable.unshift()</code> بالتابع [[Node.js/stream#readable.push.28chunk.5B.2C encoding.5D.E2.80.8E.29.E2.80.8E|stream.push(<nowiki>''</nowiki>)]] مباشرةً سوف يعيد تصفير حالة القراءة بشكل مناسب، ولكن الأفضل ببساطة هو تجنب استدعاء <code>readable.unshift()</code> أثناء عملية تنفيذ القراءة. | ||
==== readable.wrap(stream) ==== | ===== <code>readable.wrap(stream)</code> ===== | ||
أضيف في الإصدار: 0.9.4. | أضيف في الإصدار: 0.9.4. | ||
* <Stream> :stream مجرى قابل للقراءة من "نمط قديم" (old style). | * [[Node.js/stream|<Stream>]] :<code>stream</code> مجرى قابل للقراءة من "نمط قديم" (old style). | ||
* القيمة المُعادة: <this> | * القيمة المُعادة: [[بلش\بي|<this>]] | ||
قبل الإصدار Node.js 0.10، لم تطبّق المجاري كامل الواجهة البرمجية للوحدة stream كما هي معرّفة الآن. (انظر التوافق مع إصدارات Node.js الأقدم للمزيد من المعلومات.) | قبل الإصدار Node.js 0.10، لم تطبّق المجاري كامل الواجهة البرمجية للوحدة <code>stream</code> كما هي معرّفة الآن. (انظر [[Node.js/stream#.D8.A7.D9.84.D8.AA.D9.88.D8.A7.D9.81.D9.82 .D9.85.D8.B9 .D8.A5.D8.B5.D8.AF.D8.A7.D8.B1.D8.A7.D8.AA Node.js .D8.A7.D9.84.D8.A3.D9.82.D8.AF.D9.85|التوافق مع إصدارات Node.js الأقدم]] للمزيد من المعلومات.) | ||
عند استخدام مكتبات Node.js أقدم والتي تطلق الأحداث 'data' وفيها التابع stream.pause() الذي هو إرشادي فقط، يمكن أن يُستخدم التابع readable.wrap() لإنشاء مجرًى قابل للقراءة يستخدم المجرى القديم كمصدر لبياناته. | عند استخدام مكتبات Node.js أقدم والتي تطلق الأحداث [[Node.js/stream#.D8.A7.D9.84.D8.AD.D8.AF.D8.AB:.27data.27|'data']] وفيها التابع [[Node.js/stream#readable.pause.28.29.E2.80.8E|stream.pause()]] الذي هو إرشادي فقط، يمكن أن يُستخدم التابع <code>readable.wrap()</code> لإنشاء مجرًى [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Readable|قابل للقراءة]] يستخدم المجرى القديم كمصدر لبياناته. | ||
سيكون من النادر الحاجة إلى استخدام readable.wrap() ولكن التابع قُدّم كملائمة للتفاعل مع تطبيقات Node.js أقدم.<syntaxhighlight lang="javascript"> | سيكون من النادر الحاجة إلى استخدام <code>readable.wrap()</code> ولكن التابع قُدّم كملائمة للتفاعل مع تطبيقات Node.js أقدم.<syntaxhighlight lang="javascript"> | ||
const { OldReader } = require('./old-api-module.js'); | const { OldReader } = require('./old-api-module.js'); | ||
const { Readable } = require('stream'); | const { Readable } = require('stream'); | ||
سطر 873: | سطر 826: | ||
</syntaxhighlight> | </syntaxhighlight> | ||
==== readable[Symbol.asyncIterator]() ==== | ===== <code>readable[Symbol.asyncIterator]()</code> ===== | ||
أضيف في الإصدار: 10.0.0. | أضيف في الإصدار: 10.0.0. | ||
الاستقرار: 1- تجريبي | الاستقرار: 1- تجريبي | ||
القيمة المعادة: <AsyncIterator> لاستهلاك كامل المجرى.<syntaxhighlight lang="javascript"> | القيمة المعادة: [https://github.com/tc39/proposal-async-iteration <AsyncIterator>] لاستهلاك كامل المجرى.<syntaxhighlight lang="javascript"> | ||
const fs = require('fs'); | const fs = require('fs'); | ||
سطر 891: | سطر 844: | ||
print(fs.createReadStream('file')).catch(console.log); | print(fs.createReadStream('file')).catch(console.log); | ||
</syntaxhighlight>إذا انتهت الحلقة التكرارية ب break أو | </syntaxhighlight>إذا انتهت الحلقة التكرارية ب <code>break</code> أو <code>throw</code>، فسوف يُدمّر المجرى. بعبارة أخرى، سوف يستهلك التكرار عبر المجرى جميع بياناته ويقرأ المجرى كاملًا. سيُقرَأ المجرى بقطع بيانات ذات حجم مساوٍ لقيمة الخيار <code>highWaterMark</code>. في مثال الشيفرة في الأعلى، ستكون البيانات في قطعة وحيدة إذا حوى الملف على أقل من 64 كيلو بت من البيانات لأنه قيمة الخيار <code>highWaterMark</code> لم تُقدّم للتابع [[Node.js/fs#fs.createReadStream.28path.5B.2C options.5D.29.E2.80.8E|fs.createReadStream()]]. | ||
== المجاري المزدوجة (Duplex) ومجاري التحويل (Transform) == | == المجاري المزدوجة (Duplex) ومجاري التحويل (Transform) == | ||
=== الصنف stream.Duplex === | === الصنف <code>stream.Duplex</code> === | ||
سجل التغييرات | سجل التغييرات | ||
سطر 901: | سطر 854: | ||
!الإصدار | !الإصدار | ||
!التغييرات | !التغييرات | ||
! | ! rowspan="4" | | ||
|- | |- | ||
|6.8.0 | |6.8.0 | ||
|ستعيد الآن نُسخ Duplex القيمة true عند فحص instanceof stream.Writable | |ستعيد الآن نُسخ <code>Duplex</code> القيمة <code>true</code> عند فحص <code>instanceof stream.Writable</code> | ||
|- | |- | ||
|0.9.4 | |0.9.4 | ||
|أضيف في 0.9.4. | |أضيف في 0.9.4. | ||
|- | |- | ||
| | | colspan="2" | | ||
| | |||
|} | |} | ||
المجاري المزدوجة (Duplex) هي المجاري التي تطبّق كلا واجهتي المجرى القابل للقراءة (Readable) والمجرى القابل للكتابة (Writable). | المجاري المزدوجة (<code>Duplex</code>) هي المجاري التي تطبّق كلا واجهتي المجرى [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Readable|القابل للقراءة (Readable)]] والمجرى [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Writable|القابل للكتابة (Writable)]]. | ||
تتضمن أمثلة مجاري Duplex: | تتضمن أمثلة مجاري Duplex: | ||
* مقابس TCP. | * [[Node.js/net#.D8.A7.D9.84.D8.B5.D9.86.D9.81 net.Socket|مقابس TCP]]. | ||
* مجاري الوحدة zlib. | * [[Node.js/zlib|مجاري الوحدة zlib]]. | ||
* مجاري الوحدة crypto. | * [[Node.js/crypto|مجاري الوحدة crypto]]. | ||
=== الصنف: stream.Transform === | === الصنف: <code>stream.Transform</code> === | ||
أُضيف في الإصدار: 0.9.4. | أُضيف في الإصدار: 0.9.4. | ||
مجاري التحويل (Transform streams) هي مجاري من النوع المزدوج (Duplex) حيث أنَّ الخرج مرتبط بطريقة ما مع الدخل. مثل كل مجاري | مجاري التحويل (Transform streams) هي مجاري من النوع [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Duplex|المزدوج (Duplex)]] حيث أنَّ الخرج مرتبط بطريقة ما مع الدخل. مثل كل مجاري [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Duplex|Duplex]]، تطبّق مجاري <code>Transform</code> كلا واجهتي المجرى [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Readable|القابل للقراءة (Readable)]] والمجرى [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Writable|القابل للكتابة (Writable)]]. | ||
من الأمثلة عن المجاري Transform: | من الأمثلة عن المجاري <code>Transform</code>: | ||
* مجاري الوحدة zlib. | * [[Node.js/zlib|مجاري الوحدة zlib.]] | ||
* مجاري الوحدة crypto. | * [[Node.js/crypto|مجاري الوحدة crypto.]] | ||
=== transform.destroy([error]) === | ==== <code>transform.destroy([error])</code> ==== | ||
أُضيف في الإصدار: 8.0.0. | أُضيف في الإصدار: 8.0.0. | ||
* Error> | * [[JavaScript/Error|<Error>]] :error | ||
يدّمر المجرى، ويطلق الحدث 'error'. بعد هذا الاستدعاء، سيحرر مجرى التحويل أي مصادر داخلية، ينبغي على المستخدمين ألّا يعيدو تعريف هذا التابع، ولكن يمكنهم تطبيق readable._destroy() بدلًا عنه. التنفيذ الافتراضي لتابع _destroy() من أجل المجرى Transform يطلق أيضًا الحدث 'close'. | يدّمر المجرى، ويطلق الحدث <code>'error'</code>. بعد هذا الاستدعاء، سيحرر مجرى التحويل أي مصادر داخلية، ينبغي على المستخدمين ألّا يعيدو تعريف هذا التابع، ولكن يمكنهم تطبيق [[Node.js/stream#readable. destroy.28err.2C callback.29.E2.80.8E|readable._destroy()]] بدلًا عنه. التنفيذ الافتراضي لتابع <code>_destroy()</code> من أجل المجرى <code>Transform</code> يطلق أيضًا الحدث <code>'close'</code>. | ||
=== stream.finished(stream, callback) === | === <code>stream.finished(stream, callback)</code> === | ||
أُضيف في الإصدار: 10.0.0. | أُضيف في الإصدار: 10.0.0. | ||
* Stream> : | * [[Node.js/stream|<Stream>]] :<code>stream</code> مجرى قابل للقراءة و/أو قابل للكتابة. | ||
* Function> :callback> دالة رد نداء والتي تأخذ وسيط خطأ اختياري. | * [[JavaScript/Function|<Function>]] :<code>callback</code> دالة رد نداء والتي تأخذ وسيط خطأ اختياري. | ||
دالة تستعمل للحصول على إشعارات عندما لا يعود المجرى قابلًا للقراءة، أو قابلًا للكتابة أو واجه خطأً ما، أو جرى إغلاقه بشكل سابق لأوانه.<syntaxhighlight lang="javascript"> | دالة تستعمل للحصول على إشعارات عندما لا يعود المجرى قابلًا للقراءة، أو قابلًا للكتابة أو واجه خطأً ما، أو جرى إغلاقه بشكل سابق لأوانه.<syntaxhighlight lang="javascript"> | ||
const { finished } = require('stream'); | const { finished } = require('stream'); | ||
سطر 959: | سطر 904: | ||
rs.resume(); //تفريغ المجرى | rs.resume(); //تفريغ المجرى | ||
</syntaxhighlight>هذا التابع مفيد خصوصًا في حالات معالجة الخطأ الذي يُدمّر فيه المجرى بشكل سابق لأوانه (مثل طلب HTTP مُجهض) ولم يُطلق الحدث 'end' أو 'finish'. | </syntaxhighlight>هذا التابع مفيد خصوصًا في حالات معالجة الخطأ الذي يُدمّر فيه المجرى بشكل سابق لأوانه (مثل طلب HTTP مُجهض) ولم يُطلق الحدث <code>'end'</code> أو <code>'finish'</code>. | ||
الواجهة finished البرمجية هي promisify'able قابلة للتعامل مع الوعود (promise) كذلك.<syntaxhighlight lang="javascript"> | الواجهة <code>finished</code> البرمجية هي promisify'able قابلة للتعامل مع الوعود (promise) كذلك.<syntaxhighlight lang="javascript"> | ||
const finished = util.promisify(stream.finished); | const finished = util.promisify(stream.finished); | ||
سطر 976: | سطر 921: | ||
</syntaxhighlight> | </syntaxhighlight> | ||
=== | === <code>stream.pipeline(...streams[, callback])</code> === | ||
أُضيف في الإصدار: 10.0.0. | أُضيف في الإصدار: 10.0.0. | ||
* Stream> | * [[Node.js/stream|<Stream>]] : <code>...streams</code> مجرييان أو أكثر للربط بينهما. | ||
* <Function> :callback دالة رد نداء تأخذ وسيط خطأ اختياري. | * [[JavaScript/Function|<Function>]] :<code>callback</code> دالة رد نداء تأخذ وسيط خطأ اختياري. | ||
تابع نموذجي للنقل (الربط) بين المجاري موجهةً الأخطاء ومنظفةً بشكل ملائم و مُزوِدةً بدالة رد نداء عندما ينتهي خط النقل.<syntaxhighlight lang="javascript"> | تابع نموذجي للنقل (الربط) بين المجاري موجهةً الأخطاء ومنظفةً بشكل ملائم و مُزوِدةً بدالة رد نداء عندما ينتهي خط النقل.<syntaxhighlight lang="javascript"> | ||
const { pipeline } = require('stream'); | const { pipeline } = require('stream'); | ||
سطر 987: | سطر 932: | ||
//البرمجية لربط سلسلة مجاري معًا بسهولة والحصول على إشعارات pipeline استخدم واجهة | |||
عندما يتم التوصيل والنقل بشكل كامل | //عندما يتم التوصيل والنقل بشكل كامل | ||
خط النقل إلى | // :ضخمًا بكفاءة tar يحتمل ملف gzip خط النقل إلى | ||
سطر 1٬006: | سطر 951: | ||
} | } | ||
); | ); | ||
</syntaxhighlight>الواجهة البرمجية pipeline قابلة للتعامل مع الوعود (promise) أيضًا:<syntaxhighlight lang="javascript"> | </syntaxhighlight>الواجهة البرمجية <code>pipeline</code> قابلة للتعامل مع الوعود (promise) أيضًا:<syntaxhighlight lang="javascript"> | ||
const pipeline = util.promisify(stream.pipeline); | const pipeline = util.promisify(stream.pipeline); | ||
سطر 1٬022: | سطر 967: | ||
== الواجهات البرمجية لمنفذي المجاري == | == الواجهات البرمجية لمنفذي المجاري == | ||
صُممت الواجهات البرمجية للوحدة stream لتجعل بالإمكان تنفيذ المجاري بسهولة باستعمال وحدة الوراثة النموذجية في JavaScript. | صُممت الواجهات البرمجية للوحدة <code>stream</code> لتجعل بالإمكان تنفيذ المجاري بسهولة باستعمال وحدة الوراثة النموذجية في JavaScript. | ||
أولًا، سيعرّف مطوّر المجرى صنف جافا سكريبت جديد والذي سيكون توسعة لأحد أصناف المجاري الأساسية الأربعة | أولًا، سيعرّف مطوّر المجرى صنف جافا سكريبت جديد والذي سيكون توسعة لأحد أصناف المجاري الأساسية الأربعة | ||
(stream.Writable أو stream.Readable أو stream.Duplex أو stream.Transform) ضامنًا أنها تستدعي باني صنف الأب المناسب:<syntaxhighlight lang="javascript"> | (<code>stream.Writable</code> أو <code>stream.Readable</code> أو <code>stream.Duplex</code> أو <code>stream.Transform</code>) ضامنًا أنها تستدعي باني صنف الأب المناسب:<syntaxhighlight lang="javascript"> | ||
const { Writable } = require('stream'); | const { Writable } = require('stream'); | ||
سطر 1٬042: | سطر 987: | ||
!الصنف | !الصنف | ||
!توابع للتطبيق | !توابع للتطبيق | ||
! | ! rowspan="4" | | ||
|- | |- | ||
|قراءة فقط | |قراءة فقط | ||
|Readable | |[[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Readable|Readable]] | ||
|_read | |<code>[[Node.js/stream#readable. read.28size.29.E2.80.8E|_read]]</code> | ||
|- | |- | ||
|كتابة فقط | |كتابة فقط | ||
|Writable | |[[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Writable|Writable]] | ||
| | |<code>[[Node.js/stream#writable. write.28chunk.2C encoding.2C callback.29.E2.80.8E|_write]]</code>, <code>[[Node.js/stream#writable. writev.28chunks.2C callback.29.E2.80.8E|_writev]]</code>, <code>[[Node.js/stream#writable. final.28callback.29.E2.80.8E.E2.80.8E|_final]]</code> | ||
| | |||
|- | |- | ||
|قراءة وكتابة | |قراءة وكتابة | ||
|Duplex | |[[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Duplex|Duplex]] | ||
| | |<code>[[Node.js/stream#readable. read.28size.29.E2.80.8E|_read]]</code>, <code>[[Node.js/stream#writable. write.28chunk.2C encoding.2C callback.29.E2.80.8E|_write]]</code>, <code>[[Node.js/stream#writable. writev.28chunks.2C callback.29.E2.80.8E|_writev]]</code>, <code>[[Node.js/stream#writable. final.28callback.29.E2.80.8E.E2.80.8E|_final]]</code> | ||
| | |||
|} | |} | ||
لا ينبغي أبدًا أن تستدعي شيفرة التنفيذ للمجرى التوابع "العامة" (public) للمجرى والتي ضُمّنت للاستخدام من قبل المستهلكين (كما هو موصوف في مقطع الواجهات البرمجية | لا ينبغي أبدًا أن تستدعي شيفرة التنفيذ للمجرى التوابع "العامة" (public) للمجرى والتي ضُمّنت للاستخدام من قبل المستهلكين (كما هو موصوف في مقطع [[Node.js/stream#.D8.A7.D9.84.D9.88.D8.A7.D8.AC.D9.87.D8.A7.D8.AA .D8.A7.D9.84.D8.A8.D8.B1.D9.85.D8.AC.D9.8A.D8.A9 .D9.84.D9.85.D9.86.D9.81.D8.B0.D9.8A .D8.A7.D9.84.D9.85.D8.AC.D8.A7.D8.B1.D9.8A|الواجهات البرمجية لمستهلكي المجرى]]). فعل ذلك قد يقود إلى تأثيرات جانبية معاكسة في شيفرة التطبيق المستهلك للمجرى. | ||
=== البناء المبسّط (Simplified Construction) === | === البناء المبسّط (Simplified Construction) === | ||
أُضيف في الإصدار: 1.2.0. | أُضيف في الإصدار: 1.2.0. | ||
من أجل العديد من الحالات البسيطة، من الممكن انشاء مجرىً دون الإعتماد على الوراثة، يمكن أن يُنجز هذا بانشاء مباشر لنسخ من الكائنات stream.Writable أو stream.Readable أو stream.Duplex أو stream.Transform وتمرير التوابع المناسبة كخيارات للباني.<syntaxhighlight lang="javascript"> | من أجل العديد من الحالات البسيطة، من الممكن انشاء مجرىً دون الإعتماد على الوراثة، يمكن أن يُنجز هذا بانشاء مباشر لنسخ من الكائنات <code>stream.Writable</code> أو <code>stream.Readable</code> أو <code>stream.Duplex</code> أو <code>stream.Transform</code> وتمرير التوابع المناسبة كخيارات للباني.<syntaxhighlight lang="javascript"> | ||
const { Writable } = require('stream'); | const { Writable } = require('stream'); | ||
سطر 1٬074: | سطر 1٬016: | ||
</syntaxhighlight> | </syntaxhighlight> | ||
== تنفيذ مجرى قابل للكتابة == | === تنفيذ مجرى قابل للكتابة === | ||
وُسِع الصنف stream.Writable لينفذ المجرى Writable. | وُسِع الصنف <code>stream.Writable</code> لينفذ المجرى [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Writable|Writable]]. | ||
يجب أن تستدعي مجاري Writable المخصصة الباني new stream.Writable([options]) وتنفّذ التابع | يجب أن تستدعي مجاري Writable المخصصة الباني <code>new stream.Writable([options])</code> وتنفّذ التابع | ||
writable._write(). ويمُكن أن يُنفّذ التابع writable._writev() أيضًا. | <code>writable._write()</code>. ويمُكن أن يُنفّذ التابع <code>writable._writev()</code> أيضًا. | ||
==== new stream.Writable([options]) ==== | ==== new stream.Writable([options]) ==== | ||
{| class="wikitable" | {| class="wikitable mw-collapsible" | ||
!الإصدار | !الإصدار | ||
!التغييرات | !التغييرات | ||
! | ! rowspan="2" | | ||
|- | |- | ||
|10.0.0 | |10.0.0 | ||
|أضيف الخيار emitClose ليحدد فيما إذا أطلق 'close' عند الهدم. | |أضيف الخيار <code>emitClose</code> ليحدد فيما إذا أطلق <code>'close'</code> عند الهدم. | ||
|- | |- | ||
| | | colspan="3" | | ||
| | |||
|} | |} | ||
* | * [[JavaScript/Object|<Object>]] <code>options</code> | ||
** <number> :highWaterMark مستوى الذاكرة المؤقتة عندما يبدأ stream.write() بإعادة false. القيمة الإفتراضية: | ** [[JavaScript/Number|<number>]] :<code>highWaterMark</code> مستوى الذاكرة المؤقتة عندما يبدأ stream.write() بإعادة <code>false</code>. '''القيمة الإفتراضية:''' <code>16384</code> (16kb) أو <code>16</code> من أجل المجاري <code>objectMode</code>. | ||
** <boolean> :decodeStrings قيمة منطقية تحدد إذا كان يراد ترميز السلاسل النصية ككائنات من النوع Buffer أم لا قبل تمريرها إلى stream._write()، مستخدمًا الترميز المحدد في استدعاء stream.write(). القيمة الإفتراضية: true. | ** [[JavaScript/Boolean|<boolean>]] :<code>decodeStrings</code> قيمة منطقية تحدد إذا كان يراد ترميز السلاسل النصية ككائنات من النوع <code>Buffer</code> أم لا قبل تمريرها إلى [[Node.js/stream#writable. write.28chunk.2C encoding.2C callback.29.E2.80.8E|stream._write()]]، مستخدمًا الترميز المحدد في استدعاء [[Node.js/stream#writable. write.28chunk.2C encoding.2C callback.29.E2.80.8E|stream.write()]]. '''القيمة الإفتراضية:''' <code>true</code>. | ||
** <string> :defaultEncoding الترميز الافتراضي الذي يُستخدم عندما لا يحدد ترميز كوسيط في stream.write(). القيمة الإفتراضية: 'utf8'. | ** [[JavaScript/String|<string>]] :<code>defaultEncoding</code> الترميز الافتراضي الذي يُستخدم عندما لا يحدد ترميز كوسيط في [[Node.js/stream#writable. write.28chunk.2C encoding.2C callback.29.E2.80.8E|stream.write()]]. '''القيمة الإفتراضية:''' <code>'utf8'</code>. | ||
** <boolean> :objectMode قيمة منطقية تحدد فيما إذا كانت stream.write(anyObj) عملية صالحة أم لا. عند ضبطها، يصبح من الممكن كتابة قيم من أي نوع من أنواع JavaScript باستثناء السلاسل النصية، أو Buffer أو Uint8Array إذا دُعمت من قبل منفّذ المجرى. القيمة الإفتراضية: false. | ** [[JavaScript/Boolean|<boolean>]] :<code>objectMode</code> قيمة منطقية تحدد فيما إذا كانت [[Node.js/stream#writable.write.28chunk.5B.2C encoding.5D.5B.2C callback.5D.E2.80.8E.29.E2.80.8E|stream.write(anyObj)]] عملية صالحة أم لا. عند ضبطها، يصبح من الممكن كتابة قيم من أي نوع من أنواع JavaScript باستثناء السلاسل النصية، أو <code>Buffer</code> أو <code>Uint8Array</code> إذا دُعمت من قبل منفّذ المجرى. '''القيمة الإفتراضية:''' <code>false</code>. | ||
** <boolean> :emitClose قيمة منطقية تحدد فيما إذا كان ينبغي أن يطلقَ المجرى الحدثَ 'close' بعد أن يُدمّر أو لا. القيمة الإفتراضية: true. | ** [[JavaScript/Boolean|<boolean>]] :<code>emitClose</code> قيمة منطقية تحدد فيما إذا كان ينبغي أن يطلقَ المجرى الحدثَ <code>'close'</code> بعد أن يُدمّر أو لا. '''القيمة الإفتراضية:''' <code>true</code>. | ||
** <Function> :write تنفيذ للتابع stream._write(). | ** [[JavaScript/Function|<Function>]] :<code>write</code> تنفيذ للتابع [[Node.js/stream#writable. write.28chunk.2C encoding.2C callback.29.E2.80.8E|stream._write()]]. | ||
** <Function> :writev تنفيذ للتابع stream._writev() . | ** [[JavaScript/Function|<Function>]] :<code>writev</code> تنفيذ للتابع [[Node.js/stream#writable. writev.28chunks.2C callback.29.E2.80.8E|stream._writev()]] . | ||
** <Function> :destroy تنفيذ للتابع stream._destroy(). | ** [[JavaScript/Function|<Function>]] :<code>destroy</code> تنفيذ للتابع [[Node.js/stream#writable. destroy.28err.2C callback.29.E2.80.8E|stream._destroy()]]. | ||
** <Function> :final تنفيذ للتابع stream._final(). | ** [[JavaScript/Function|<Function>]] :<code>final</code> تنفيذ للتابع [[Node.js/stream#writable. final.28callback.29.E2.80.8E.E2.80.8E|stream._final()]]. | ||
<syntaxhighlight lang="javascript"> | <syntaxhighlight lang="javascript"> | ||
const { Writable } = require('stream'); | const { Writable } = require('stream'); | ||
سطر 1٬119: | سطر 1٬050: | ||
constructor(options) { | constructor(options) { | ||
//stream.Writable() استدعِ باني | |||
super(options); | super(options); | ||
سطر 1٬150: | سطر 1٬081: | ||
</syntaxhighlight> | </syntaxhighlight> | ||
=== writable._write(chunk, encoding, callback) === | ==== <code>writable._write(chunk, encoding, callback)</code> ==== | ||
* <Buffer> | <string> | <any> :chunk قطعة البيانات المراد كتابتها. سوف تكون دائمًا من النوع buffer إلا إذا كان الخيار decodeStrings مضبوطًا إلى القيمة false أو أن المجرى يعمل في نمط الكائن. | * [[Node.js/buffer#.D8.A7.D9.84.D8.B5.D9.86.D9.81 Buffer|<Buffer>]] | [[JavaScript/String|<string>]] | <any> :<code>chunk</code> قطعة البيانات المراد كتابتها. سوف تكون '''دائمًا''' من النوع buffer إلا إذا كان الخيار <code>decodeStrings</code> مضبوطًا إلى القيمة <code>false</code> أو أن المجرى يعمل في نمط الكائن. | ||
* <string> :encoding إذا كانت القطعة هي سلسلة نصية فإن encoding سيكون هو ترميز المحارف لهذه السلسلة. إذا كانت القطعة chunck من النوع | * [[JavaScript/String|<string>]] :<code>encoding</code> إذا كانت القطعة هي سلسلة نصية فإن <code>encoding</code> سيكون هو ترميز المحارف لهذه السلسلة. إذا كانت القطعة chunck من النوع <code>Buffer</code>، أو أن المجرى يعمل في نمط الكائن، سيُتجاهل الوسيط <code>encoding</code>. | ||
* <Function> :callback يستدعي هذه الدالة (مع وسيط خطأ اختياري) عندما تنتهي معالجة القطع المعطاة. | * [[JavaScript/Function|<Function>]] :<code>callback</code> يستدعي هذه الدالة (مع وسيط خطأ اختياري) عندما تنتهي معالجة القطع المعطاة. | ||
يجب أن توفر كل مجاري Writable التابع writable._write() لإرسال البيانات إلى المصادر الأساسية. | يجب أن توفر كل مجاري <code>Writable</code> التابع [[Node.js/stream#writable. write.28chunk.2C encoding.2C callback.29.E2.80.8E|writable._write()]] لإرسال البيانات إلى المصادر الأساسية. | ||
تقدّم مجاري التحويل (Transform) تنفيذها الخاص للتابع | تقدّم مجاري التحويل (Transform) تنفيذها الخاص للتابع <nowiki/>[[Node.js/stream#writable. write.28chunk.2C encoding.2C callback.29.E2.80.8E|writable._write()]]. | ||
يجب ألّا تُستدعَى هذه الدالة من قبل الشيفرة التطبيق مباشرةً. يجب أن تُنفّذ باستخدام صنف ابن، وتُستدعى من قبل توابع الصنف Writable الداخلية فقط. | يجب ألّا تُستدعَى هذه الدالة من قبل الشيفرة التطبيق مباشرةً. يجب أن تُنفّذ باستخدام صنف ابن، وتُستدعى من قبل توابع الصنف <code>Writable</code> الداخلية فقط. | ||
يجب أن تستدعَى الدالة callback لتشير إمّا إلى انتهاء الكتابة بنجاح أو فشلها مع خطأ. يجب أن يكون الوسيط الأول المُمرر إلى callback هو كائن Error إذا فشل الإستدعاء أو null إذا نجحت الكتابة. | يجب أن تستدعَى الدالة <code>callback</code> لتشير إمّا إلى انتهاء الكتابة بنجاح أو فشلها مع خطأ. يجب أن يكون الوسيط الأول المُمرر إلى <code>callback</code> هو كائن <code>Error</code> إذا فشل الإستدعاء أو <code>null</code> إذا نجحت الكتابة. | ||
كل الاستدعاءات للتابع writable.write() التي تحصل بين وقت استدعاء writable._write() واستدعاء الدالة callback سوف تؤدي إلى تخزين البيانات المكتوبة مؤقتًا. عندما تستدعى الدالة callback، قد يطلق المجرى الحدث 'drain'. إذا كان تطبيق المجرى قادرًا على معالجة قطع متعددة من البيانات مرّة واحدة، فينبغي أن يُنفَّذ التابع writable._writev(). | كل الاستدعاءات للتابع <code>writable.write()</code> التي تحصل بين وقت استدعاء <code>writable._write()</code> واستدعاء الدالة <code>callback</code> سوف تؤدي إلى تخزين البيانات المكتوبة مؤقتًا. عندما تستدعى الدالة callback، قد يطلق المجرى الحدث [[Node.js/stream#.D8.A7.D9.84.D8.AD.D8.AF.D8.AB .27drain.27|'drain']]. إذا كان تطبيق المجرى قادرًا على معالجة قطع متعددة من البيانات مرّة واحدة، فينبغي أن يُنفَّذ التابع <code>writable._writev()</code>. | ||
إذا كانت الخاصية decodeStrings مضبوطة بشكل صريح إلى false في خيارات الباني. عند ذلك ستبقى chunk نفس الكائن الذي يمُرر إلى التابع | إذا كانت الخاصية <code>decodeStrings</code> مضبوطة بشكل صريح إلى <code>false</code> في خيارات الباني. عند ذلك ستبقى <code>chunk</code> نفس الكائن الذي يمُرر إلى التابع <code>.write()</code> وقد تكون سلسلة نصية بدلًا من <code>Buffer</code>. هذا من أجل دعم التطبيقات التي تمتلك معالجة مثالية لبعض ترميزات البيانات النصية. في هذه الحالة، سوف يحدد الوسيط <code>encoding</code> ترميز محارف السلسلة النصية. وإلّا، سوف يُهمل الوسيط <code>encoding</code> بشكل آمن. | ||
سيكون التابع writable._write() مسبوقًا بالرمز _ لأنه داخلي للصنف الذي عرّفه، وينبغي ألّا يُستدعى مطلقًا بشكل مباشر من قبل برامج المستخدم. | سيكون التابع <code>writable._write()</code> مسبوقًا بالرمز _ لأنه داخلي للصنف الذي عرّفه، وينبغي ألّا يُستدعى مطلقًا بشكل مباشر من قبل برامج المستخدم. | ||
=== writable._writev(chunks, callback) === | ==== <code>writable._writev(chunks, callback)</code> ==== | ||
* <Object[]> :chunks القطع التي ستُكتب. كل قطعة لها الشكل التالي: { chunk: ..., encoding: ... | * [[JavaScript/Object|<Object[]>]] :<code>chunks</code> القطع التي ستُكتب. كل قطعة لها الشكل التالي: <code>{ chunk: ..., encoding: ... }</code>. | ||
* <Function> :callback دالة رد نداء (مع وسيط خطأ بشكل اختياري) لتُستدعى عند انتهاء المعالجة للقطع المُزوّدة. | * [[JavaScript/Function|<Function>]] :<code>callback</code> دالة رد نداء (مع وسيط خطأ بشكل اختياري) لتُستدعى عند انتهاء المعالجة للقطع المُزوّدة. | ||
يجب ألّا تُستدعى هذه الدالة من قبل شيفرة التطبيق مباشرةً، يجب أن تُنفّذ من قبل صنف ابن، وتُستدعى من قبل توابع الصنف Writable الداخلية فقط. | يجب ألّا تُستدعى هذه الدالة من قبل شيفرة التطبيق مباشرةً، يجب أن تُنفّذ من قبل صنف ابن، وتُستدعى من قبل توابع الصنف <code>Writable</code> الداخلية فقط. | ||
يمكن أن يُنفّذ التابع writable._writev() بالإضافة إلى writable._write() في تطبيقات المجاري والتي تمتلك القدرة على معالجة قطع متعددة من البيانات في المرة الواحدة. إذا نُفّذت، سوف يُستدعى التابع مع كل قطع البيانات المخزّنة حاليًا في طابور الكتابة. | يمكن أن يُنفّذ التابع <code>writable._writev()</code> بالإضافة إلى <code>writable._write()</code> في تطبيقات المجاري والتي تمتلك القدرة على معالجة قطع متعددة من البيانات في المرة الواحدة. إذا نُفّذت، سوف يُستدعى التابع مع كل قطع البيانات المخزّنة حاليًا في طابور الكتابة. | ||
يُسبق التابع writable._writev() بالرمز _ لأنّه داخلي للصنف الذي عرّفه، وينبغي ألّا يُستدعى مطلقًا من قبل من قبل برامج المستخدم. | يُسبق التابع <code>writable._writev()</code> بالرمز _ لأنّه داخلي للصنف الذي عرّفه، وينبغي ألّا يُستدعى مطلقًا من قبل من قبل برامج المستخدم. | ||
== writable._destroy(err, callback) == | ==== <code>writable._destroy(err, callback)</code> ==== | ||
أضيف في الإصدار: 8.0.0. | أضيف في الإصدار: 8.0.0. | ||
* <Error> :err خطأ محتمل. | * [[JavaScript/Error|<Error>]] :<code>err</code> خطأ محتمل. | ||
* <Function> :callback دالة رد نداء والتي تأخذ وسيط خطأ اختياري. | * [[JavaScript/Function|<Function>]] :<code>callback</code> دالة رد نداء والتي تأخذ وسيط خطأ اختياري. | ||
يُستدعى التابع | يُستدعى التابع <code>_destroy()</code> من قبل التابع [[Node.js/stream#writable.destroy.28.5Berror.5D.E2.80.8E.29.E2.80.8E|writable.destroy()]]. يمكن أن يُعاد تعريفه من قبل صنف ابن ولكن يجب ألّا يُستدعى بشكل مباشر. | ||
== writable._final(callback) == | ==== <code>writable._final(callback)</code> ==== | ||
أضيف في الإصدار: 8.0.0. | أضيف في الإصدار: 8.0.0. | ||
<Function> :callback يستدعي هذه الدالة (مع وسيط خطأ اختياري) عند انتهاء كتابة أي بيانات متبقية. | [[JavaScript/Function|<Function>]] :<code>callback</code> يستدعي هذه الدالة (مع وسيط خطأ اختياري) عند انتهاء كتابة أي بيانات متبقية. | ||
يجب ألّا يُستدعى التابع _final() بشكل مباشر. يمكن تنفيذه من قبل صنف ابن؛ وإذا كان ذلك، سوف يُستدعى مع توابع الصنف Writable الداخلية فقط. | يجب ألّا يُستدعى التابع <code>_final()</code> بشكل مباشر. يمكن تنفيذه من قبل صنف ابن؛ وإذا كان ذلك، سوف يُستدعى مع توابع الصنف <code>Writable</code> الداخلية فقط. | ||
سوف تُستدعى هذه الدالة الاختيارية قبل أن يغلق المجرى، مؤخرةً الحدث 'finish' حتى تستدعى الدالة callback. هذا مفيد لإغلاق المصادر أو كتابة البيانات المخزّنة مؤقتًا قبل أن ينتهي المجرى. | سوف تُستدعى هذه الدالة الاختيارية قبل أن يغلق المجرى، مؤخرةً الحدث <code>'finish'</code> حتى تستدعى الدالة <code>callback</code>. هذا مفيد لإغلاق المصادر أو كتابة البيانات المخزّنة مؤقتًا قبل أن ينتهي المجرى. | ||
== أخطاء أثناء الكتابة == | ==== أخطاء أثناء الكتابة ==== | ||
يُنصح بأن يُكتب تقرير بالأخطاء الحاصلة أثناء معالجة التابعين writable._write() و writable._writev() عن طريق استدعاء دالة رد النداء وتمرير الخطأ كأول وسيط. هذا سوف يسبب انطلاق الحدث 'error' من قِبَل المجرى Writable. رمي خطأ أثناء writable._write() يمكن أن ينتج سلوك غير متوقّع ومتضارب بالاعتماد على كيفية استعمال المجرى. يضمن استخدام دالة رد النداء معالجة أخطاء متناسقة ومتوقعة. | يُنصح بأن يُكتب تقرير بالأخطاء الحاصلة أثناء معالجة التابعين <code>writable._write()</code> و <code>writable._writev()</code> عن طريق استدعاء دالة رد النداء وتمرير الخطأ كأول وسيط. هذا سوف يسبب انطلاق الحدث <code>'error'</code> من قِبَل المجرى <code>Writable</code>. رمي خطأ <code>Error</code> أثناء <code>writable._write()</code> يمكن أن ينتج سلوك غير متوقّع ومتضارب بالاعتماد على كيفية استعمال المجرى. يضمن استخدام دالة رد النداء معالجة أخطاء متناسقة ومتوقعة. | ||
إذا كان المجرى Readable موصولًا بأنبوب مع المجرى Writable وأطلق Writable خطأً في هذه الأثناء، فسوف ينفصل المجرى Readable.<syntaxhighlight lang="javascript"> | إذا كان المجرى <code>Readable</code> موصولًا بأنبوب مع المجرى <code>Writable</code> وأطلق <code>Writable</code> خطأً في هذه الأثناء، فسوف ينفصل المجرى <code>Readable</code>.<syntaxhighlight lang="javascript"> | ||
const { Writable } = require('stream'); | const { Writable } = require('stream'); | ||
سطر 1٬211: | سطر 1٬142: | ||
</syntaxhighlight> | </syntaxhighlight> | ||
== مثالٌ عن مجرى قابل للكتابة (Writable) == | ==== مثالٌ عن مجرى قابل للكتابة (Writable) ==== | ||
يشرح التالي تنفيذ مجرى مخصص من النوع Writable وبسيط إلى حد ما (عديم الجدوى إلى حدٍ ما). بينما تكون نسخة المجرى Writable المحدد ليست ذات فائدة محددة حقيقية. يشرح المثال كل العناصر المطلوبة لنسخة مجرى Writable مخصصة:<syntaxhighlight lang="javascript"> | يشرح التالي تنفيذ مجرى مخصص من النوع <code>Writable</code> وبسيط إلى حد ما (عديم الجدوى إلى حدٍ ما). بينما تكون نسخة المجرى <code>Writable</code> المحدد ليست ذات فائدة محددة حقيقية. يشرح المثال كل العناصر المطلوبة لنسخة مجرى [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Writable|Writable]] مخصصة:<syntaxhighlight lang="javascript"> | ||
const { Writable } = require('stream'); | const { Writable } = require('stream'); | ||
سطر 1٬231: | سطر 1٬162: | ||
</syntaxhighlight> | </syntaxhighlight> | ||
== ترميز الذواكر المؤقتة في المجرى القابل للكتابة (Writable) == | ==== فك ترميز الذواكر المؤقتة في المجرى القابل للكتابة (Writable) ==== | ||
ترميز الذواكر المؤقتة هو مهمة شائعة، على سبيل المثال، عند استخدام محوّلات دخلها هو سلسلة نصية. فهي ليست عملية بديهية عند استخدام ترميز محارف متعدد البايتات، مثل UTF-8. يعرض المثال التالي كيفية ترميز سلاسل متعددة البايتات باستخدام StringDecoder و Writable.<syntaxhighlight lang="javascript"> | فك ترميز الذواكر المؤقتة هو مهمة شائعة، على سبيل المثال، عند استخدام محوّلات دخلها هو سلسلة نصية. فهي ليست عملية بديهية عند استخدام ترميز محارف متعدد البايتات، مثل UTF-8. يعرض المثال التالي كيفية فك ترميز سلاسل متعددة البايتات باستخدام <code>StringDecoder</code> و [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Writable|Writable]].<syntaxhighlight lang="javascript"> | ||
const { Writable } = require('stream'); | const { Writable } = require('stream'); | ||
const { StringDecoder } = require('string_decoder'); | const { StringDecoder } = require('string_decoder'); | ||
سطر 1٬262: | سطر 1٬193: | ||
w.end(euro[1]); | w.end(euro[1]); | ||
console.log(w.data); // عملة | console.log(w.data); //€ :عملة | ||
</syntaxhighlight> | </syntaxhighlight> | ||
== تنفيذ مجرى قابل للقراءة (Readable) == | == تنفيذ مجرى قابل للقراءة (Readable) == | ||
يُوسّع الصنف stream.Readable لينفّذ مجرى قابل للقراءة (Readable). | يُوسّع الصنف <code>stream.Readable</code> لينفّذ مجرى [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Readable|قابل للقراءة (Readable)]]. | ||
يجب أن تستدعي مجاري Readable المخصصة الباني stream.Readable([options]) وتنفّذ التابع readable._read(). | يجب أن تستدعي مجاري <code>Readable</code> المخصصة الباني <code>stream.Readable([options])</code> وتنفّذ التابع <code>readable._read()</code>. | ||
== new stream.Readable([options]) == | === <code>new stream.Readable([options])</code> === | ||
* <Object> :options | * <Object> :<code>options</code> | ||
** | ** <nowiki/>[[JavaScript/Number|<number>]] :<code>highWaterMark</code> [[Node.js/stream#.D8.A7.D8.AE.D8.AA.D9.84.D8.A7.D9.81 highWaterMark .D8.A8.D8.B9.D8.AF .D8.A7.D8.B3.D8.AA.D8.AF.D8.B9.D8.A7.D8.A1 readable.setEncoding.28.29.E2.80.8E|العدد الأعظمي من البايتات]] لتُخزّن في الذاكرة المؤقتة الداخلية قبل ابطال القراءة من المصادر الأساسية. القيمة الإفتراضية: <code>16384</code> (16kb) أو <code>16</code> لأجل مجاري <code>objectMode</code>. | ||
** <string> :encoding إذا كان محدّدًا،عندئذ سوف يُفك ترميز الذاكرة إلى سلاسل نصية باستخدام الترميز المحدد. القيمة الإفتراضية: null. | ** [[JavaScript/String|<string>]] :encoding إذا كان محدّدًا،عندئذ سوف يُفك ترميز الذاكرة إلى سلاسل نصية باستخدام الترميز المحدد. '''القيمة الإفتراضية:''' <code>null</code>. | ||
** <boolean> :objectMode قيمة منطقية تحدد فيما إذا كان ينبغي أن يتصرف هذا المجرى كمجرى من الكائنات. يعني أن يعيد stream.read(n) قيمة وحيدة بدلًا من Buffer بالحجم n. القيمة الإفتراضية: false. | ** [[JavaScript/Boolean|<boolean>]] :<code>objectMode</code> قيمة منطقية تحدد فيما إذا كان ينبغي أن يتصرف هذا المجرى كمجرى من الكائنات. يعني أن يعيد [[Node.js/stream#readable.read.28.5Bsize.5D.E2.80.8E.29.E2.80.8E|stream.read(n)]] قيمة وحيدة بدلًا من <code>Buffer</code> بالحجم <code>n</code>. '''القيمة الإفتراضية:''' <code>false</code>. | ||
** <Function> :read تنفيذ للتابع stream._read(). | ** [[JavaScript/Function|<Function>]] :<code>read</code> تنفيذ للتابع [[Node.js/stream#readable. read.28size.29.E2.80.8E|stream._read()]]. | ||
** <Function> :destroy تنفيذ للتابع stream._destroy(). | ** [[JavaScript/Function|<Function>]] :<code>destroy</code> تنفيذ للتابع [[Node.js/stream#readable. destroy.28err.2C callback.29.E2.80.8E|stream._destroy()]]. | ||
<syntaxhighlight lang="javascript"> | <syntaxhighlight lang="javascript"> | ||
const { Readable } = require('stream'); | const { Readable } = require('stream'); | ||
سطر 1٬311: | سطر 1٬242: | ||
</syntaxhighlight> | </syntaxhighlight> | ||
== readable._read(size) == | === <code>readable._read(size)</code> === | ||
سجل التغييرات | سجل التغييرات | ||
سطر 1٬317: | سطر 1٬248: | ||
!الإصدار | !الإصدار | ||
!التغييرات | !التغييرات | ||
! | ! rowspan="4" | | ||
|- | |- | ||
|10.0.0 | |10.0.0 | ||
|استدعاء | |استدعاء <code>_read()</code> لمرّة واحدة في كل جزء صغير من النبضة (microtick). | ||
|- | |- | ||
|0.9.4 | |0.9.4 | ||
|أُضيف في الإصدار: 0.9.4. | |أُضيف في الإصدار: 0.9.4. | ||
|- | |- | ||
| | | colspan="2" | | ||
| | |||
|} | |} | ||
* <number> :size عدد البايتات المراد قراءتها بشكل غير متزامن. | * [[JavaScript/Number|<number>]] :<code>size</code> عدد البايتات المراد قراءتها بشكل غير متزامن. | ||
يجب ألّا تُستدعى هذه الدالة من قبل شيفرة التطبيق مباشرةً. يجب أن تنفّذ من قبل أصناف ابن. وتُستدعى من قبل توابع الصنف Readable الداخلية فقط. | يجب ألّا تُستدعى هذه الدالة من قبل شيفرة التطبيق مباشرةً. يجب أن تنفّذ من قبل أصناف ابن. وتُستدعى من قبل توابع الصنف <code>Readable</code> الداخلية فقط. | ||
كل تطبيقات المجاري القابلة للقراءة (Readable) يجب أن توفر تنفيذًا للتابع readable._read() لجلب البيانات من المصادر الأساسية. | كل تطبيقات المجاري القابلة للقراءة (<code>Readable</code>) يجب أن توفر تنفيذًا للتابع <code>readable._read()</code> لجلب البيانات من المصادر الأساسية. | ||
عندما يُستدعى readable._read()، إذا كانت البيانات متوافرة من المصدر، ينبغي أن يبدأ التطبيق بدفع البيانات إلى داخل طابور القراءة باستخدام التابع this.push(dataChunk) . ينبغي أن يكمل | عندما يُستدعى <code>readable._read()</code>، إذا كانت البيانات متوافرة من المصدر، ينبغي أن يبدأ التطبيق بدفع البيانات إلى داخل طابور القراءة باستخدام التابع [[Node.js/stream#readable.push.28chunk.5B.2C encoding.5D.E2.80.8E.29.E2.80.8E|this.push(dataChunk)]]. ينبغي أن يكمل <code>_read()</code> القراءة من المصادر ودفع البيانات حتى يعيد <code>readable.push()</code> القيمة <code>false</code>. فقط عندما يُستدعى <code>_read()</code> مجدّدًا بعد أن توقف، ينبغي أن يستأنف دفع بيانات إضافية فوق الطابور. | ||
حالما استُدعي التابع readable._read()، لن يُستدعى مجدّدًا حتى يُستدعى التابع readable.push(). يُضمن أن يُستدعى readable._read() لمرّة واحدة فقط خلال التنفيذ المتزامن؛ أي -6 ^10 من النبضة (microtick). | حالما استُدعي التابع <code>readable._read()</code>، لن يُستدعى مجدّدًا حتى يُستدعى التابع [[Node.js/stream#readable.push.28chunk.5B.2C encoding.5D.E2.80.8E.29.E2.80.8E|readable.push()]]. يُضمن أن يُستدعى <code>readable._read()</code> لمرّة واحدة فقط خلال التنفيذ المتزامن؛ أي -6 ^10 من النبضة (microtick). | ||
الوسيط size هو إرشادي. من أجل تطبيقات تكون فيها القراءة هي عملية وحيدة تعيد بيانات، يمكن أن تستخدم الوسيط size لتحديد كمية البيانات المراد جلبها. قد تتجاهل تطبيقات أخرى هذا الوسيط وتقدّم البيانات ببساطة عندما تصبح متوافرة. لا يوجد حاجة للإنتظار حتى تتوافر بايتات بالحجم size قبل استدعاء stream.push(chunk). | الوسيط <code>size</code> هو إرشادي. من أجل تطبيقات تكون فيها القراءة هي عملية وحيدة تعيد بيانات، يمكن أن تستخدم الوسيط <code>size</code> لتحديد كمية البيانات المراد جلبها. قد تتجاهل تطبيقات أخرى هذا الوسيط وتقدّم البيانات ببساطة عندما تصبح متوافرة. لا يوجد حاجة للإنتظار حتى تتوافر بايتات بالحجم <code>size</code> قبل استدعاء [[Node.js/stream#readable.push.28chunk.5B.2C encoding.5D.E2.80.8E.29.E2.80.8E|stream.push(chunk)]]. | ||
يُسبق التابع readable._read() بالرمز _ لأنّه داخلي للصنف الذي عرّفه، ولا ينبغي أبدًا أن يُستدعى بشكل مباشر من قبل برامج المستخدم. | يُسبق التابع <code>readable._read()</code> بالرمز _ لأنّه داخلي للصنف الذي عرّفه، ولا ينبغي أبدًا أن يُستدعى بشكل مباشر من قبل برامج المستخدم. | ||
== readable._destroy(err, callback) == | === <code>readable._destroy(err, callback)</code> === | ||
أُضيف في الإصدار: 8.0.0 | أُضيف في الإصدار: 8.0.0 | ||
* <Error> :err خطأ مُحتمل. | * [[JavaScript/Error|<Error>]] :<code>err</code> خطأ مُحتمل. | ||
* <Function> :callback دالة رد نداء والتي تأخذ وسيط خطأ اختياري. | * [[JavaScript/Function|<Function>]] :<code>callback</code> دالة رد نداء والتي تأخذ وسيط خطأ اختياري. | ||
يُستدعى تابع _destroy() من قبل readable.destroy(). يمكن أن يُعاد تعريفه من قبل صنف ابن ولكن يجب ألّا يُستدعى بشكل مباشر. | يُستدعى تابع <code>_destroy()</code> من قبل [[Node.js/stream#readable.destroy.28.5Berror.5D.E2.80.8E.29.E2.80.8E|readable.destroy()]]. يمكن أن يُعاد تعريفه من قبل صنف ابن ولكن يجب ألّا يُستدعى بشكل مباشر. | ||
== readable.push(chunk[, encoding]) == | === <code>readable.push(chunk[, encoding])</code> === | ||
سجل التغييرات | سجل التغييرات | ||
سطر 1٬360: | سطر 1٬283: | ||
!الإصدار | !الإصدار | ||
!التغييرات | !التغييرات | ||
! | ! rowspan="2" | | ||
|- | |- | ||
|8.0.0 | |8.0.0 | ||
|يمكن الآن أن يكون الوسيط chunk نسخة من Uint8Array. | |يمكن الآن أن يكون الوسيط <code>chunk</code> نسخة من <code>Uint8Array</code>. | ||
|- | |- | ||
| | | colspan="3" | | ||
| | |||
|} | |} | ||
* <Buffer> | <Uint8Array> | <string> | <null> | <any> :chunk قطعة من البيانات للدفع إلى طابور القراءة. من أجل المجاري التي لا تعمل في نمط الكائن، يجب أن تكون chunk هي سلسلة نصية أو Buffer أو Uint8Array. من أجل المجاري التي تعمل في نمط الكائن، يمكن أن تكون chunk أي نوع من أنواع بيانات JavaScript. | * [[Node.js/buffer#.D8.A7.D9.84.D8.B5.D9.86.D9.81 Buffer|<Buffer>]] | [https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Uint8Array <Uint8Array>] | [[JavaScript/String|<string>]] | [[JavaScript/null|<null>]] | [[svgzsdfgzsdf|<any>]] :<code>chunk</code> قطعة من البيانات للدفع إلى طابور القراءة. من أجل المجاري التي لا تعمل في نمط الكائن، يجب أن تكون <code>chunk</code> هي سلسلة نصية أو <code>Buffer</code> أو <code>Uint8Array</code>. من أجل المجاري التي تعمل في نمط الكائن، يمكن أن تكون <code>chunk</code> أي نوع من أنواع بيانات JavaScript. | ||
* string> :encoding> ترميز قطع السلاسل النصية. يجب أن يكون ترميز Buffer صالحًا، مثل 'utf8' أو 'ascii'. | * [[JavaScript/String|<string>]] :<code>encoding</code> ترميز قطع السلاسل النصية. يجب أن يكون ترميز <code>Buffer</code> صالحًا، مثل <code>'utf8'</code> أو <code>'ascii'</code>. | ||
* القيمة المُعادة: <boolean> تكون القيمة true إذا كان هناك قطع اضافية من البيانات قد يستمر دفعها، وإلّا فستكون false. | * القيمة المُعادة: [[JavaScript/Boolean|<boolean>]] تكون القيمة <code>true</code> إذا كان هناك قطع اضافية من البيانات قد يستمر دفعها، وإلّا فستكون <code>false</code>. | ||
عندما تكون chunk هي من النوع Buffer أو Uint8Array أو | عندما تكون <code>chunk</code> هي من النوع <code>Buffer</code> أو <code>Uint8Array</code> أو <code>string</code>، ستضاف البيانات التي تحتويها إلى الطابور الداخلي لمستخدمي المجاري لتصبح متوافرة للقراءة والاستهلاك من قبل مستخدمي المجرى. تمرير القيمة <code>null</code> إلى الخيار <code>chunk</code> يشير إلى نهاية المجرى (EOF)، أي لا يوجد بعدها المزيد من البيانات لتكتب. | ||
عندما يعمل Readable في نمط التوقف المؤقت (paused mode)، يمكن قراءة البيانات المضافة مع التابع readable.push() باستدعاء التابع readable.read() عندما يُطلق الحدث 'readable'. | عندما يعمل <code>Readable</code> في نمط التوقف المؤقت (paused mode)، يمكن قراءة البيانات المضافة مع التابع <code>readable.push()</code> باستدعاء التابع [[Node.js/stream#readable.read.28.5Bsize.5D.E2.80.8E.29.E2.80.8E|readable.read()]] عندما يُطلق الحدث [[Node.js/stream#.D8.A7.D9.84.D8.AD.D8.AF.D8.AB: .27readable.27|'readable']]. | ||
عندما يعمل Readable في نمط التدفق، ستُستلم كل البيانات المضافة مع readable.push() بمجرد إطلاق الحدث 'data'. | عندما يعمل <code>Readable</code> في نمط التدفق، ستُستلم كل البيانات المضافة مع <code>readable.push()</code> بمجرد إطلاق الحدث <code>'data'</code>. | ||
يُصمم التابع readable.push() ليكون مرنًا قدر الإمكان. على سبيل المثال، عند تغليف مصدر منخفض المستوى (lower-level source) والذي يقدّم بعض أشكال آليات التوقف أو الإستئناف، ورد نداء البيانات، يمكن أن يُغلَّف المصدر منخفض المستوى من قبل نسخة Readable مخصصة:<syntaxhighlight lang="javascript"> | يُصمم التابع <code>readable.push()</code> ليكون مرنًا قدر الإمكان. على سبيل المثال، عند تغليف مصدر منخفض المستوى (lower-level source) والذي يقدّم بعض أشكال آليات التوقف أو الإستئناف، ورد نداء البيانات، يمكن أن يُغلَّف المصدر منخفض المستوى من قبل نسخة <code>Readable</code> مخصصة:<syntaxhighlight lang="javascript"> | ||
//readStop() و readStart() المصدر هو كائن مع التابعين | |||
//والذي يحصل على استدعاء عندما يمتلك بيانات `ondata` ومع العضو | |||
//والذي يحصل على استدعاء عندما تنتهي البيانات `onend` ومع العضو | |||
سطر 1٬400: | سطر 1٬312: | ||
كل مرّة توجد بيانات ، ادفعها إلى داخل الذاكرة المؤقتة الداخلية | //كل مرّة توجد بيانات ، ادفعها إلى داخل الذاكرة المؤقتة الداخلية | ||
this._source.ondata = (chunk) => { | this._source.ondata = (chunk) => { | ||
// عند ذلك أوقف القراءة من المصدر،false القيمة push()إذا أعاد التابع | |||
if (!this.push(chunk)) | if (!this.push(chunk)) | ||
سطر 1٬411: | سطر 1٬323: | ||
}; | }; | ||
عندما ينتهي المصدر، ادفع قطعة اشارات النهاية | //`null`عندما ينتهي المصدر، ادفع قطعة اشارات النهاية | ||
this._source.onend = () => { | this._source.onend = () => { | ||
سطر 1٬422: | سطر 1٬334: | ||
// عندما يريد المجرى سحب المزيد من البيانات إليه _read سوف يُستدعى | |||
يُتجاهل وسيط الحجم الموصى به في هذه الحالة | //يُتجاهل وسيط الحجم الموصى به في هذه الحالة | ||
_read(size) { | _read(size) { | ||
سطر 1٬430: | سطر 1٬342: | ||
} | } | ||
</syntaxhighlight>يُعَدّ التابع eadable.push() للاستدعاء فقط من قبل مُنفّذات المجرى | </syntaxhighlight>يُعَدّ التابع <code>eadable.push()</code> للاستدعاء فقط من قبل مُنفّذات المجرى <code>Readable</code>، وفقط من داخل التابع <code>readable._read()</code>. | ||
من أجل المجاري التي لا تعمل في نمط الكائن، إذا كان معامل chunk الخاص بالتابع | من أجل المجاري التي لا تعمل في نمط الكائن، إذا كان معامل <code>chunk</code> الخاص بالتابع <code>readable.push()</code> هو <code>undefined</code>، سيُعامل كسلسلة نصية فارغة أو ذاكرة تخزين مؤقت فارغة. انظر [[Node.js/stream#readable.push.28.27.27.29.E2.80.8E|readable.push(<nowiki>''</nowiki>)]] للمزيد من المعلومات. | ||
== أخطاء أثناء القراءة == | === أخطاء أثناء القراءة === | ||
يوصى بأن تُطلَق الأخطاء الحاصلة خلال معالجة التابع readable._read() باستخدام الحدث 'error' بدلًا من أن تُرمى فقط. قد ينتج رمي خطأٍ من داخل readable._read() إلى سلوك غير متوقع و متضارب بالإعتماد على إذا كان المجرى يعمل في نمط التدفق أو النمط المتوقف. يضمن استخدام الحدث 'error' معالجة أخطاء متوقعة ومتناسقة.<syntaxhighlight lang="javascript"> | يوصى بأن تُطلَق الأخطاء الحاصلة خلال معالجة التابع <code>readable._read()</code> باستخدام الحدث <code>'error'</code> بدلًا من أن تُرمى فقط. قد ينتج رمي خطأٍ من داخل <code>readable._read()</code> إلى سلوك غير متوقع و متضارب بالإعتماد على إذا كان المجرى يعمل في نمط التدفق أو النمط المتوقف. يضمن استخدام الحدث <code>'error'</code> معالجة أخطاء متوقعة ومتناسقة.<syntaxhighlight lang="javascript"> | ||
const { Readable } = require('stream'); | const { Readable } = require('stream'); | ||
سطر 1٬449: | سطر 1٬361: | ||
</syntaxhighlight> | </syntaxhighlight> | ||
== مثالٌ عن مجرى عدٍّ (Counting Stream) == | === مثالٌ عن مجرى عدٍّ (Counting Stream) === | ||
التالي هو مثال أساسي لمجرى قابل للقراءة (Readable) والذي يطلق الأعداد من 1 إلى 1,000,000 بترتيب تصاعدي ومن ثم ينتهي.<syntaxhighlight lang="javascript"> | التالي هو مثال أساسي لمجرى قابل للقراءة (<code>Readable</code>) والذي يطلق الأعداد من 1 إلى 1,000,000 بترتيب تصاعدي ومن ثم ينتهي.<syntaxhighlight lang="javascript"> | ||
const { Readable } = require('stream'); | const { Readable } = require('stream'); | ||
سطر 1٬474: | سطر 1٬386: | ||
</syntaxhighlight> | </syntaxhighlight> | ||
== | == تنفيذ مجرى مزدوج (Duplex) == | ||
المجرى المزدوج (Duplex) هو الذي ينفِّذ كلا الواجهتين Readable و | المجرى [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Duplex|المزدوج (Duplex)]] هو الذي ينفِّذ كلا الواجهتين [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Readable|Readable]] و [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Writable|Writable]]، مثل اتصال المقبس TCP. | ||
لأنّ JavaScript لا تدعم الوراثة المتعددة، وُسِّع الصنف stream.Duplex لينفّذ المجرى Duplex (كمقابل لتوسعة الصنفين stream.Readable و stream.Writable). | لأنّ JavaScript لا تدعم الوراثة المتعددة، وُسِّع الصنف <code>stream.Duplex</code> لينفّذ المجرى [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Duplex|Duplex]] (كمقابل لتوسعة الصنفين <code>stream.Readable</code> و <code>stream.Writable</code>). | ||
يرث stream.Duplex من الصنف stream.Readable بشكل نموذجي (prototypically) ومن الصنف stream.Writable بشكل طفيلي (parasitically)، ولكن سوف يعمل instanceof بشكل صحيح لأجل كلا الصنفين الأساسيين بسبب إعادة تعريف Symbol.hasInstance على stream.Writable. | يرث <code>stream.Duplex</code> من الصنف <code>stream.Readable</code> بشكل نموذجي (prototypically) ومن الصنف <code>stream.Writable</code> بشكل طفيلي (parasitically)، ولكن سوف يعمل <code>instanceof</code> بشكل صحيح لأجل كلا الصنفين الأساسيين بسبب إعادة تعريف [https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Symbol/hasInstance Symbol.hasInstance] على <code>stream.Writable</code>. | ||
يجب أن تستدعي مجاري Duplex المخصصة الباني new stream.Duplex([options]) وتنفّذ كلا التابعين readable._read() و writable._write(). | يجب أن تستدعي مجاري <code>Duplex</code> المخصصة الباني <code>new stream.Duplex([options])</code> وتنفّذ كلا التابعين <code>readable._read()</code> و <code>writable._write()</code>. | ||
== new stream.Duplex(options) == | === <code>new stream.Duplex(options)</code> === | ||
سجل التغييرات | سجل التغييرات | ||
سطر 1٬489: | سطر 1٬401: | ||
!الإصدار | !الإصدار | ||
!التغييرات | !التغييرات | ||
! | ! rowspan="2" | | ||
|- | |- | ||
|8.4.0 | |8.4.0 | ||
|دُعم الآن الخياران readableHighWaterMark و writableHighWaterMark | |دُعم الآن الخياران <code>readableHighWaterMark</code> و <code>writableHighWaterMark</code> | ||
|- | |- | ||
| | | colspan="3" | | ||
| | |||
|} | |} | ||
* <Object> :options يمرر إلى كلا البانيين Writable و Readable. ويملك أيضًا الحقول الآتية: | * [[JavaScript/Object|<Object>]] :<code>options</code> يمرر إلى كلا البانيين <code>Writable</code> و <code>Readable</code>. ويملك أيضًا الحقول الآتية: | ||
** <boolean> :allowHalfOpen قيمة منطقية إذا كانت | ** [[JavaScript/Boolean|<boolean>]] :<code>allowHalfOpen</code> قيمة منطقية إذا كانت <code>false</code>،فسوف ينهي المجرى تلقائيًا الطرف القابل للكتابة عندما ينتهي الطرف القابل للقراءة. '''القيمة الافتراضية:''' <code>true</code>. | ||
** <boolean> :readableObjectMode يضبط هذا الخيار الوضع objectMode للطرف القابل للقراءة في المجرى. ليس له تأثير إذا كانت objectMode هي true. القيمة الافتراضية: false. | ** [[JavaScript/Boolean|<boolean>]] :<code>readableObjectMode</code> يضبط هذا الخيار الوضع <code>objectMode</code> للطرف القابل للقراءة في المجرى. ليس له تأثير إذا كانت <code>objectMode</code> هي <code>true</code>. '''القيمة الافتراضية:''' <code>false</code>. | ||
** <boolean> :writableObjectMode يضبط هذا الخيار الوضع objectMode للطرف القابل للكتابة في المجرى. ليس له تأثير إذا كانت objectMode هي true. القيمة الافتراضية: false. | ** [[JavaScript/Boolean|<boolean>]] :<code>writableObjectMode</code> يضبط هذا الخيار الوضع <code>objectMode</code> للطرف القابل للكتابة في المجرى. ليس له تأثير إذا كانت <code>objectMode</code> هي <code>true</code>. '''القيمة الافتراضية:''' <code>false</code>. | ||
** <number> :readableHighWaterMark يضبط القيمة highWaterMark لأجل الطرف القابل للقراءة من المجرى. لا يملك تأثيرًا إذا كانت القيمة highWaterMark معطاة. | ** [[JavaScript/Number|<number>]] :<code>readableHighWaterMark</code> يضبط القيمة <code>highWaterMark</code> لأجل الطرف القابل للقراءة من المجرى. لا يملك تأثيرًا إذا كانت القيمة <code>highWaterMark</code> معطاة. | ||
<number> :writableHighWaterMark يضبط القيمة highWaterMark لأجل الطرف القابل للكتابة من المجرى. لا يملك تأثيرًا إذا كانت القيمة highWaterMark معطاة.<syntaxhighlight lang="javascript"> | ** [[JavaScript/Number|<number>]] :<code>writableHighWaterMark</code> يضبط القيمة <code>highWaterMark</code> لأجل الطرف القابل للكتابة من المجرى. لا يملك تأثيرًا إذا كانت القيمة <code>highWaterMark</code> معطاة. | ||
<syntaxhighlight lang="javascript"> | |||
const { Duplex } = require('stream'); | const { Duplex } = require('stream'); | ||
سطر 1٬548: | سطر 1٬450: | ||
</syntaxhighlight> | </syntaxhighlight> | ||
== مثال على مجرى مزدوج (Duplex) == | === مثال على مجرى مزدوج (Duplex) === | ||
يشرح التالي مثال بسيط عن مجرى من النوع المزدوج (Duplex) والذي يغلّف كائن مصدر افتراضي منخفض المستوى إلى أي بيانات يمكن أن تُكتب وأن تُقرأ، ولو باستخدام واجهات برمجية غير متوافقة مع مجاري Node.js. يشرح المثال البسيط التالي مجرى Duplex والذي يخزّن مؤقتًا البيانات القادمة والمكتوبة عليه بواسطة الواجهة Writable ثم يعاد قراءتها مجدَّدًا (والتي تُقرأ تراجعيًا) بواسطة الواجهة Readable. <syntaxhighlight lang="javascript"> | يشرح التالي مثال بسيط عن مجرى من النوع المزدوج (<code>Duplex</code>) والذي يغلّف كائن مصدر افتراضي منخفض المستوى إلى أي بيانات يمكن أن تُكتب وأن تُقرأ، ولو باستخدام واجهات برمجية غير متوافقة مع مجاري Node.js. يشرح المثال البسيط التالي مجرى <code>Duplex</code> والذي يخزّن مؤقتًا البيانات القادمة والمكتوبة عليه بواسطة الواجهة [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Writable|Writable]] ثم يعاد قراءتها مجدَّدًا (والتي تُقرأ تراجعيًا) بواسطة الواجهة [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Readable|Readable]]. <syntaxhighlight lang="javascript"> | ||
const { Duplex } = require('stream'); | const { Duplex } = require('stream'); | ||
const kSource = Symbol('source'); | const kSource = Symbol('source'); | ||
سطر 1٬575: | سطر 1٬477: | ||
} | } | ||
</syntaxhighlight>النقطة الأكثر أهمية في مجاري Duplex هي أنّ كل طرف من الطرفين Readable و Writable يعمل بشكل مستقل عن الآخر رغم تواجدهما داخل نسخة كائن واحدة. | </syntaxhighlight>النقطة الأكثر أهمية في مجاري <code>Duplex</code> هي أنّ كل طرف من الطرفين <code>Readable</code> و <code>Writable</code> يعمل بشكل مستقل عن الآخر رغم تواجدهما داخل نسخة كائن واحدة. | ||
== تشغيل المجاري المزدوجة بنمط الكائن == | === تشغيل المجاري المزدوجة بنمط الكائن === | ||
من أجل المجاري المزدوجة (Duplex)، يمكن أن يُضبط الخيار objectMode على وجه الحصر إمّا على الطرف Readable أو الطرف Writable باستخدام الخيارات readableObjectMode و writableObjectMode على التوالي. | من أجل المجاري المزدوجة (<code>Duplex</code>)، يمكن أن يُضبط الخيار <code>objectMode</code> على وجه الحصر إمّا على الطرف <code>Readable</code> أو الطرف <code>Writable</code> باستخدام الخيارات <code>readableObjectMode</code> و <code>writableObjectMode</code> على التوالي. | ||
في المثال التالي، على سبيل المثال، يُنشأ مجرى جديد من النوع Transform (والذي هو نوع من المجاري المزدوجة (Duplex)) والذي يمتلك طرف قابل للكتابة (Writable) بنمط الكائن والذي يقبل أرقام JavaScript حُوّلت إلى سلاسل نصية ستة عشرية على الطرف Readable.<syntaxhighlight lang="javascript"> | في المثال التالي، على سبيل المثال، يُنشأ مجرى جديد من النوع <code>Transform</code> (والذي هو نوع من المجاري [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Duplex|المزدوجة (Duplex)]]) والذي يمتلك طرف قابل للكتابة (<code>Writable</code>) بنمط الكائن والذي يقبل أرقام JavaScript حُوّلت إلى سلاسل نصية ستة عشرية على الطرف <code>Readable</code>.<syntaxhighlight lang="javascript"> | ||
const { Transform } = require('stream'); | const { Transform } = require('stream'); | ||
سطر 1٬619: | سطر 1٬521: | ||
== تطبيق مجرى التحويل (Transform) == | == تطبيق مجرى التحويل (Transform) == | ||
مجرى التحويل (Transform) هو مجرى مزدوج (Duplex) حيث أن الخرج محسوب بطريقة ما من الدخل. مجاري الوحدة zlib أو مجاري الوحدة crypto هي مثال على هذا النوع من المجاري والتي تضغط أو تشفّر أو تفك تشفير البيانات. | مجرى [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81: stream.Transform|التحويل (Transform)]] هو مجرى [https://wiki.hsoub.com/Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81_stream.Duplex مزدوج (Duplex)] حيث أن الخرج محسوب بطريقة ما من الدخل. مجاري الوحدة [[Node.js/zlib|zlib]] أو مجاري الوحدة [[Node.js/crypto|crypto]] هي مثال على هذا النوع من المجاري والتي تضغط أو تشفّر أو تفك تشفير البيانات. | ||
لايوجد ضرورة أن يكون الخرج بنفس حجم الدخل أو نفس عدد القطع أو أن يصل في نفس الوقت. على سبيل المثال، المجرى Hash سوف يملك دائًما وأبدًا قطعة خرج واحدة والتي تُوفّر عندما يُنهَى الدخل. سوف ينتج المجرى zlib خرجًا إمّا أصغر بكثير أو أكبر بكثير من الدخل. | لايوجد ضرورة أن يكون الخرج بنفس حجم الدخل أو نفس عدد القطع أو أن يصل في نفس الوقت. على سبيل المثال، المجرى <code>Hash</code> سوف يملك دائًما وأبدًا قطعة خرج واحدة والتي تُوفّر عندما يُنهَى الدخل. سوف ينتج المجرى <code>zlib</code> خرجًا إمّا أصغر بكثير أو أكبر بكثير من الدخل. | ||
وُسّع الصنف stream.Transform لينفّذ مجرى التحويل (Transform). | وُسّع الصنف <code>stream.Transform</code> لينفّذ مجرى [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81: stream.Transform|التحويل (Transform)]]. | ||
يرث الصنف stream.Transform بشكل نموذجي من stream.Duplex ويطبّق إصداراته الخاصة من التابعين writable._write() و readable._read() .يجب أن تنفّذ تطبيقات مجاري Transform المخصصة التابع transform._transform() وربما تنفّذ أيضًا التابع transform._flush() . | يرث الصنف <code>stream.Transform</code> بشكل نموذجي من <code>stream.Duplex</code> ويطبّق إصداراته الخاصة من التابعين <code>writable._write()</code> و <code>readable._read()</code> .يجب أن تنفّذ تطبيقات مجاري <code>Transform</code> المخصصة التابع [[Node.js/stream#transform. transform.28chunk.2C encoding.2C callback.29.E2.80.8E|transform._transform()]] وربما تنفّذ أيضًا التابع [[Node.js/stream#transform. flush.28callback.29.E2.80.8E|transform._flush()]]. | ||
يجب توخي الحذر عند استخدام مجاري | يجب توخي الحذر عند استخدام مجاري <code>Transform</code>، إذ أنّ البيانات المكتوبة إلى المجرى قد تؤدي إلى أن يصبح الطرف <code>Writable</code> للمجرى متوقفًا إذا كان الخرج على الطرف <code>Readable</code> غير مُستهلك. | ||
== new stream.Transform([options]) == | === new stream.Transform([options]) === | ||
* | * [[JavaScript/Object|<Object>]] :options مُمرّرة إلى كلا بانيي <code>Writable</code> و <code>Readable</code>. وتحوي أيضًا الحقول التالية: | ||
** <Function> :transform تنفيذ لتابع stream._transform() | ** [[JavaScript/Function|<Function>]] :<code>transform</code> تنفيذ لتابع [[Node.js/stream#transform. transform.28chunk.2C encoding.2C callback.29.E2.80.8E|stream._transform()]] | ||
** <Function> :flush تنفيذ لتابع stream._flush() | ** [[JavaScript/Function|<Function>]] :<code>flush</code> تنفيذ لتابع [[Node.js/stream#transform. flush.28callback.29.E2.80.8E|stream._flush()]] | ||
<syntaxhighlight lang="javascript"> | <syntaxhighlight lang="javascript"> | ||
const { Transform } = require('stream'); | const { Transform } = require('stream'); | ||
سطر 1٬664: | سطر 1٬566: | ||
</syntaxhighlight> | </syntaxhighlight> | ||
== الحدثان 'finish' و 'end' == | === الحدثان <code>'finish'</code> و <code>'end'</code> === | ||
الحدثان 'finish' و 'end' هما من الصنفين stream.Writable و stream.Readable على التوالي. يُطلق الحدث 'finish' بعد أن يُستدعى stream.end() وقد عُولجت كل القطع من قبل stream._transform() .يُطلق الحدث 'end' بعد أن أُخرجت كل البيانات والذي يحصل بعدما استُدعي رد النداء في التابع transform._flush(). | الحدثان [[Node.js/stream#.D8.A7.D9.84.D8.AD.D8.AF.D8.AB .27finish.27|'finish']] و [[Node.js/stream#.D8.A7.D9.84.D8.AD.D8.AF.D8.AB: .27end.27|'end']] هما من الصنفين <code>stream.Writable</code> و <code>stream.Readable</code> على التوالي. يُطلق الحدث <code>'finish'</code> بعد أن يُستدعى [[Node.js/stream#writable.end.28.5Bchunk.5D.5B.2C encoding.5D.5B.2C callback.5D.E2.80.8E.29.E2.80.8E|stream.end()]] وقد عُولجت كل القطع من قبل [[Node.js/stream#transform. transform.28chunk.2C encoding.2C callback.29.E2.80.8E|stream._transform()]] .يُطلق الحدث <code>'end'</code> بعد أن أُخرجت كل البيانات والذي يحصل بعدما استُدعي رد النداء في التابع [[Node.js/stream#transform. flush.28callback.29.E2.80.8E|transform._flush()]]. | ||
== transform._flush(callback) == | === <code>transform._flush(callback)</code> === | ||
* <Function> : | * [[JavaScript/Function|<Function>]] :<code>callback</code> دالة رد نداء (مع وسيط خطأ وبيانات بشكل اختياري) لتُستدعى عندما تُدفع البيانات المتبقية. | ||
يجب ألّا تُستدعى الدالة من قبل شيفرة التطبيق مباشرةً. يجب أن تُنفّذ من قبل أصناف ابن، وتُستدعى من قبل توابع الصنف Readable الداخلية فقط. | يجب ألّا تُستدعى الدالة من قبل شيفرة التطبيق مباشرةً. يجب أن تُنفّذ من قبل أصناف ابن، وتُستدعى من قبل توابع الصنف <code>Readable</code> الداخلية فقط. | ||
في بعض الحالات، قد تحتاج عملية التحويل إلى إطلاق بت إضافي من البيانات في نهاية المجرى. على سبيل المثال، سوف يخزّن مجرى الضغط | في بعض الحالات، قد تحتاج عملية التحويل إلى إطلاق بت إضافي من البيانات في نهاية المجرى. على سبيل المثال، سوف يخزّن مجرى الضغط <code>zlib</code> كمية من الحالة الداخلية المستخدمة لضغط الخرج بشكل مثالي. ولكن عندما ينتهي المجرى، تحتاج البيانات الإضافية إلى أن تُدفَع بحيث ستكتمل البيانات المضغوطة. | ||
قد تنفّذ مجاري تحويل (Transform) مخصصة التابع transform._flush(). سوف يُستدعى هذا عندما لا يكون هناك المزيد من البيانات المكتوب لتُستهلك (لتُقرأ)، ولكن قبل إطلاق الحدث 'end' مشيرًا إلى نهاية المجرى القابل للقراءة. | قد تنفّذ مجاري [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81: stream.Transform|تحويل (Transform)]] مخصصة التابع <code>transform._flush()</code>. سوف يُستدعى هذا عندما لا يكون هناك المزيد من البيانات المكتوب لتُستهلك (لتُقرأ)، ولكن قبل إطلاق الحدث [[Node.js/stream#.D8.A7.D9.84.D8.AD.D8.AF.D8.AB: .27end.27|'end']] مشيرًا إلى نهاية المجرى [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Readable|القابل للقراءة]]. | ||
خلال تنفيذ transform._flush() | خلال تنفيذ <code>transform._flush()</code>، قد لا يُستدعى التابع <code>readable.push()</code> مطلقًا أو يستدعى أكثر من مرة حسب الحاجة. يجب أن تُستدعى الدالة <code>callback</code> عند انتهاء عملية دفع البيانات خارج المجرى. | ||
التابع transform._flush() مسبوق بالرمز _ لأنّه داخلي للصنف الذي عرّفه، وينبغي ألّا يُستدعى مباشرةً من قبل برامج المستخدم. | التابع <code>transform._flush()</code> مسبوق بالرمز _ لأنّه داخلي للصنف الذي عرّفه، وينبغي ألّا يُستدعى مباشرةً من قبل برامج المستخدم. | ||
== transform._transform(chunk, encoding, callback) == | === <code>transform._transform(chunk, encoding, callback)</code> === | ||
* <Buffer> | <string> | <any>: chunk قطعة البيانات المراد تحويلها. ستكون دائمًا كائنًا من النوع Buffer إلّا إذا ضُبط الخيار decodeStrings إلى القيمة false أو أن المجرى يعمل في نمط الكائن. | * [[Node.js/buffer#.D8.A7.D9.84.D8.B5.D9.86.D9.81 Buffer|<Buffer>]] | [[JavaScript/String|<string>]] | [[تلبرنعبنتعفب|<any>]]: <code>chunk</code> قطعة البيانات المراد تحويلها. ستكون دائمًا كائنًا من النوع Buffer إلّا إذا ضُبط الخيار <code>decodeStrings</code> إلى القيمة <code>false</code> أو أن المجرى يعمل في نمط الكائن. | ||
* <string> | * [[JavaScript/String|<string>]] <code>encoding</code> إذا كانت القطعة سلسلة نصية، عندها ستكون قيمة encoding هي نوع الترميز. إذا كانت القطعة buffer ومن ثم ستكون قيمة encoding هي قيمة خاصة 'buffer'، تتجاهلها في هذه الحالة. | ||
* <Function>: callback دالة رد نداء (مع وسيط خطأ وبيانات اختيارية) لتُستدعى بعد أن تتم معالجة chunk المزوّدة. | * [[JavaScript/Function|<Function>]]: <code>callback</code> دالة رد نداء (مع وسيط خطأ وبيانات اختيارية) لتُستدعى بعد أن تتم معالجة <code>chunk</code> المزوّدة. | ||
يجب ألّا تُستدعى الدالة من قبل شيفرة التطبيق مباشرةً. يجب أن تُنفّذ من قبل الأصناف الأبناء وتُستدعى من قبل توابع الصنف Readable الداخلية فقط. | يجب '''ألّا تُستدعى''' الدالة من قبل شيفرة التطبيق مباشرةً. يجب أن تُنفّذ من قبل الأصناف الأبناء وتُستدعى من قبل توابع الصنف <code>Readable</code> الداخلية فقط. | ||
كل تطبيقات المجاري Transform يجب أن تقدّم التابع _transform() ليقبل دخلًا وينتج خرجًا. تنفيذ transform._transform() يعالج البايتات التي تُكتب، يحسب الخرج، ومن ثم يمرر ذاك الخرج خارجًا إلى الجزء القابل للقراءة باستخدام التابع readable.push(). | كل تطبيقات المجاري <code>Transform</code> يجب أن تقدّم التابع <code>_transform()</code> ليقبل دخلًا وينتج خرجًا. تنفيذ <code>transform._transform()</code> يعالج البايتات التي تُكتب، يحسب الخرج، ومن ثم يمرر ذاك الخرج خارجًا إلى الجزء القابل للقراءة باستخدام التابع <code>readable.push()</code>. | ||
قد لا يُستدعى التابع transform.push() أبدًا أو يستدعى أكثر من مرة لتوليد خرج من قطعة دخل واحدة، معتمدًا على الكمية المراد إخراجها كنتيجة للقطعة. | قد لا يُستدعى التابع <code>transform.push()</code> أبدًا أو يستدعى أكثر من مرة لتوليد خرج من قطعة دخل واحدة، معتمدًا على الكمية المراد إخراجها كنتيجة للقطعة. | ||
من الممكن ألّا يولّد خرجٌ من أي قطعة مكتوبة من بيانات الدخل. | من الممكن ألّا يولّد خرجٌ من أي قطعة مكتوبة من بيانات الدخل. | ||
يجب أن تُستدعى الدالة callback فقط عندما تُستهلك القطعة الحالية بالكامل. يجب أن يكون الوسيط الأول المُمرر إلى callback هو كائن من النوع Error إذا حصل خطأ أثناء معالجة الدخل أو null بصورة أخرى. إذا مُرر وسيط ثاني إلى الدالة | يجب أن تُستدعى الدالة <code>callback</code> فقط عندما تُستهلك القطعة الحالية بالكامل. يجب أن يكون الوسيط الأول المُمرر إلى <code>callback</code> هو كائن من النوع <code>Error</code> إذا حصل خطأ أثناء معالجة الدخل أو <code>null</code> بصورة أخرى. إذا مُرر وسيط ثاني إلى الدالة <code>callback</code>، فسوف يُحال إلى التابع <code>readable.push()</code>. بعبارة أخرى يعادل التالي:<syntaxhighlight lang="javascript"> | ||
transform.prototype._transform = function(data, encoding, callback) { | transform.prototype._transform = function(data, encoding, callback) { | ||
this.push(data); | this.push(data); | ||
سطر 1٬702: | سطر 1٬604: | ||
</syntaxhighlight> | </syntaxhighlight> | ||
يُسبق التابع transform._transform() بالرمز _ لأنّه داخلي بالنسبة للصنف الذي عرّفه، وينبغي ألّا يُستدعى مباشرةً من قبل برامج المستخدم. | يُسبق التابع <code>transform._transform()</code> بالرمز _ لأنّه داخلي بالنسبة للصنف الذي عرّفه، وينبغي ألّا يُستدعى مباشرةً من قبل برامج المستخدم. | ||
لايُستدعى التابع transform._transform() على التفرّع مطلقًا؛ تطبّق المجاري آلية الطوابير ويجب أن تستدعى الدالة callback لاستقبال القطعة التالية إمّا بشكل متزامن أو غير متزامن. | لايُستدعى التابع <code>transform._transform()</code> على التفرّع مطلقًا؛ تطبّق المجاري آلية الطوابير ويجب أن تستدعى الدالة <code>callback</code> لاستقبال القطعة التالية إمّا بشكل متزامن أو غير متزامن. | ||
== الصنف: stream.PassThrough == | === الصنف: <code>stream.PassThrough</code> === | ||
الصنف stream.PassThrough هو تنفيذ بسيط لمجرى التحويل (Transform) والذي يمرر بايتات الدخل ببساطة عبره إلى الخرج. الغرض منه في المقام الأول هو الأمثلة والفحص، ولكن هناك بعض حالات الاستخدام حيث يكون stream.PassThrough مفيدًا ككتلة بناء لأنواع غير مألوفة من المجاري. | الصنف <code>stream.PassThrough</code> هو تنفيذ بسيط لمجرى [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81: stream.Transform|التحويل (Transform)]] والذي يمرر بايتات الدخل ببساطة عبره إلى الخرج. الغرض منه في المقام الأول هو الأمثلة والفحص، ولكن هناك بعض حالات الاستخدام حيث يكون <code>stream.PassThrough</code> مفيدًا ككتلة بناء لأنواع غير مألوفة من المجاري. | ||
== ملاحظات إضافية == | == ملاحظات إضافية == | ||
=== التوافق مع إصدارات Node.js الأقدم === | === التوافق مع إصدارات Node.js الأقدم === | ||
قبل الإصدار Node.js 0.10، كانت واجهة المجرى Readable أبسط، ولكن أيضًا أقل قوةً وأقل فائدة. | قبل الإصدار Node.js 0.10، كانت واجهة المجرى <code>Readable</code> أبسط، ولكن أيضًا أقل قوةً وأقل فائدة. | ||
* بدل الإنتظار حتى استدعاء التابع stream.read()، سوف يبدأ إطلاق الأحداث 'data' فورًا. التطبيقات التي ستحتاج إلى إنجاز بعض كمية العمل لتقرر كيفية معالجة البيانات تكون مطالبةً بتخزين البيانات المقروءة إلى ذاكرة تخزين مؤقتة داخلية لكي لا تفقد البيانات. | * بدل الإنتظار حتى استدعاء التابع [[Node.js/stream#readable.read.28.5Bsize.5D.E2.80.8E.29.E2.80.8E|stream.read()]]، سوف يبدأ إطلاق الأحداث [[Node.js/stream#.D8.A7.D9.84.D8.AD.D8.AF.D8.AB:.27data.27|'data']] فورًا. التطبيقات التي ستحتاج إلى إنجاز بعض كمية العمل لتقرر كيفية معالجة البيانات تكون مطالبةً بتخزين البيانات المقروءة إلى ذاكرة تخزين مؤقتة داخلية لكي لا تفقد البيانات. | ||
* كان التابع stream.pause() توجيهيًا، بدلًا من أن يكون مضمونًا. عنى هذا أنّه بقي من الضروري أن يكون مستعدًّا لاستقبال الأحداث 'data' حتى عندما يكون المجرى في حالة التوقف المؤقت. | * كان التابع [[Node.js/stream#readable.pause.28.29.E2.80.8E|stream.pause()]] توجيهيًا، بدلًا من أن يكون مضمونًا. عنى هذا أنّه بقي من الضروري أن يكون مستعدًّا لاستقبال الأحداث [[Node.js/stream#.D8.A7.D9.84.D8.AD.D8.AF.D8.AB:.27data.27|'data']] حتى عندما يكون المجرى في حالة التوقف المؤقت. | ||
في الإصدار Node.js 0.10، أُضيف الصنف Readable. من أجل التوافقية مع برامج Node.js أقدم، تنقلب المجاري Readable إلى "نمط التدفق" عندما يُضاف معالج الحدث 'data' أو عندما يُستدعى التابع | في الإصدار Node.js 0.10، أُضيف الصنف [[Node.js/stream#.D8.A7.D9.84.D8.B5.D9.86.D9.81 stream.Readable|Readable]]. من أجل التوافقية مع برامج Node.js أقدم، تنقلب المجاري <code>Readable</code> إلى "نمط التدفق" عندما يُضاف معالج الحدث [[Node.js/stream#.D8.A7.D9.84.D8.AD.D8.AF.D8.AB:.27data.27|'data']] أو عندما يُستدعى التابع <nowiki/>[[Node.js/stream#readable.resume.28.29.E2.80.8E|stream.resume()]]. التأثير هو أنّه حتى عند عدم استخدام التابع [[Node.js/stream#readable.read.28.5Bsize.5D.E2.80.8E.29.E2.80.8E|stream.read()]] و الحدث [[Node.js/stream#.D8.A7.D9.84.D8.AD.D8.AF.D8.AB: .27readable.27|'readable']]، لم يعد من الضروري القلق حول فقد قطع [[Node.js/stream#.D8.A7.D9.84.D8.AD.D8.AF.D8.AB:.27data.27|'data']]. | ||
بينما سوف تستمر معظم التطبيقات بالعمل بشكل اعتيادي إلا أن ذلك قد خلق حالةً حديةً في الحالات التالية: | بينما سوف تستمر معظم التطبيقات بالعمل بشكل اعتيادي إلا أن ذلك قد خلق حالةً حديةً في الحالات التالية: | ||
* لن يُضاف مستمع للحدث 'data'. | * لن يُضاف مستمع للحدث [[Node.js/stream#.D8.A7.D9.84.D8.AD.D8.AF.D8.AB:.27data.27|'data']]. | ||
* لن يُستدعى التابع أبدًا | * لن يُستدعى التابع أبدًا [[Node.js/stream#readable.resume.28.29.E2.80.8E|stream.resume()]]. | ||
* لن توصل المجاري مع أي وجهة قابلة للكتابة. | * لن توصل المجاري مع أي وجهة قابلة للكتابة. | ||
على سبيل المثال، ألق نظرة فاحصة على الشيفرة التالية:<syntaxhighlight lang="javascript"> | على سبيل المثال، ألق نظرة فاحصة على الشيفرة التالية:<syntaxhighlight lang="javascript"> | ||
تحذير! معطل | // !تحذير! معطل | ||
سطر 1٬729: | سطر 1٬631: | ||
// ولكن لن نستهلك البيانات أبدًا'end' سوف نضيف مُنصتًا لحدث | |||
socket.on('end', () => { | socket.on('end', () => { | ||
سطر 1٬742: | سطر 1٬644: | ||
</syntaxhighlight>قبل الإصدار Node.js 0.10، ستُحذف رسالة البيانات القادمة ببساطة. ولكن، في الإصدار Node.js 0.10 وما بعده، يبقى المقبس متوقفًا إلى الأبد. | </syntaxhighlight>قبل الإصدار Node.js 0.10، ستُحذف رسالة البيانات القادمة ببساطة. ولكن، في الإصدار Node.js 0.10 وما بعده، يبقى المقبس متوقفًا إلى الأبد. | ||
الحل البديل لهذه المشكلة هو استدعاء التابع stream.resume() لبدء تدفق البيانات:<syntaxhighlight lang="javascript"> | الحل البديل لهذه المشكلة هو استدعاء التابع [[Node.js/stream#readable.resume.28.29.E2.80.8E|stream.resume()]] لبدء تدفق البيانات:<syntaxhighlight lang="javascript"> | ||
// الحل البديل | // الحل البديل | ||
net.createServer((socket) => { | net.createServer((socket) => { | ||
سطر 1٬756: | سطر 1٬658: | ||
</syntaxhighlight> | </syntaxhighlight> | ||
بالإضافة إلى قلب المجاري Readable الجديدة إلى نمط التدفق، يمكن تغليف المجاري ذات النمط السابق ل 0.10 في الصنف Readable باستخدام التابع readable.wrap() . | بالإضافة إلى قلب المجاري <code>Readable</code> الجديدة إلى نمط التدفق، يمكن تغليف المجاري ذات النمط السابق ل 0.10 في الصنف <code>Readable</code> باستخدام التابع [[Node.js/stream#readable.wrap.28stream.29.E2.80.8E|readable.wrap()]] . | ||
== readable.read(0) == | === <code>readable.read(0)</code> === | ||
يوجد بعض الحالات حيث من الضروري إطلاق التحديث لآليات المجاري الأساسية القابلة للقراءة، دون استهلاك فعلي لأية بيانات. في هذه الحالات، من الممكن استدعاء readable.read(0) | يوجد بعض الحالات حيث من الضروري إطلاق التحديث لآليات المجاري الأساسية القابلة للقراءة، دون استهلاك فعلي لأية بيانات. في هذه الحالات، من الممكن استدعاء <code>readable.read(0)</code>،والتي ستعيد <code>null</code> دائمًا. | ||
إذا كانت ذاكرة التخزين المؤقتة الداخلية أقل من الحد | إذا كانت ذاكرة التخزين المؤقتة الداخلية أقل من الحد <code>highWaterMark</code>، ولا يقرأ المجرى الآن، ومن ثمّ سيثير استدعاء التابع <code>stream.read(0)</code> استدعاء تابع [[Node.js/stream#readable. read.28size.29.E2.80.8E|stream._read()]] منخفض المستوى. | ||
لما كانت معظم التطبيقات على الغالب ليست بحاجة لفعل ذلك أبدًا، يوجد حالات داخلNode.js حيث يُفعل ذلك، خصوصًا داخل مجاري الصنف Readable. | لما كانت معظم التطبيقات على الغالب ليست بحاجة لفعل ذلك أبدًا، يوجد حالات داخلNode.js حيث يُفعل ذلك، خصوصًا داخل مجاري الصنف <code>Readable</code>. | ||
== readable.push(<nowiki>''</nowiki>) == | === <code>readable.push(<nowiki>''</nowiki>)</code> === | ||
لا يُنصح باستخدام readable.push(<nowiki>''</nowiki>). | لا يُنصح باستخدام <code>readable.push(<nowiki>''</nowiki>)</code>. | ||
دفع سلسلة نصية أو كائن من النوع Buffer أو Uint8Array بصفر من البايتات إلى مجرى لا يعمل في نمط الكائن له تأثير جانبي مثير للانتباه. ذلك لأنّ استدعاء التابع readable.push() | دفع سلسلة نصية أو كائن من النوع <code>Buffer</code> أو <code>Uint8Array</code> بصفر من البايتات إلى مجرى لا يعمل في نمط الكائن له تأثير جانبي مثير للانتباه. ذلك لأنّ استدعاء التابع [[Node.js/stream#readable.push.28chunk.5B.2C encoding.5D.E2.80.8E.29.E2.80.8E|readable.push()]]، سوف ينهي الاستدعاء عملية القراءة. على أي حال، لأنّ الوسيط هو سلسلة فارغة، لن تضاف أي بيانات إلى الذاكرة المؤقتة القابلة للقراءة وبذلك لا يوجد شيء ليستهلكه المستخدم. | ||
== اختلاف highWaterMark بعد استدعاء readable.setEncoding() == | === اختلاف highWaterMark بعد استدعاء readable.setEncoding() === | ||
سوف يغير استخدام التابع readable.setEncoding() السلوك الذي تسلكه القيمة highWaterMark مع مجرى يعمل بدون تفعيل نمط الكائن. | سوف يغير استخدام التابع <code>readable.setEncoding()</code> السلوك الذي تسلكه القيمة <code>highWaterMark</code> مع مجرى يعمل بدون تفعيل نمط الكائن. | ||
عادةً، يقاس حجم الذاكرة المؤقتة الحالية اعتمادًا على القيمة highWaterMark بالبايت. ولكن بعد أن يُستدعى | عادةً، يقاس حجم الذاكرة المؤقتة الحالية اعتمادًا على القيمة <code>highWaterMark</code> بالبايت. ولكن بعد أن يُستدعى <code>setEncoding()</code>، ستبدأ دالة المقارنة بقياس حجم الذاكرة المؤقتة بالمحارف. | ||
هذه ليست اشكالية في الحالات الشائعة مع الترميز latin1 أو ascii. ولكن يُنصح بأن تكون مدركًا لهذا السلوك عند العمل مع سلاسل نصية يمكن أن تحتوي محارف متعددة البايتات. | هذه ليست اشكالية في الحالات الشائعة مع الترميز <code>latin1</code> أو <code>ascii</code>. ولكن يُنصح بأن تكون مدركًا لهذا السلوك عند العمل مع سلاسل نصية يمكن أن تحتوي محارف متعددة البايتات. | ||
== مصادر == | == مصادر == | ||
[https://nodejs.org/dist/latest-v10.x/docs/api/stream.html صفحة Stream في توثيق Node.js الرسمي.] | [https://nodejs.org/dist/latest-v10.x/docs/api/stream.html صفحة Stream في توثيق Node.js الرسمي.] |
المراجعة الحالية بتاريخ 06:07، 2 يناير 2019
الاستقرار: 2-مستقر
المجرى هو واجهة مجرّدة للعمل مع البيانات المتدفقة في Node.js. توفّر الوحدة stream
واجهة برمجية (API) أساسية تجعل من السهل بناء كائنات تتعامل مع واجهة المجرى.
يوجد العديد من كائنات المجرى التي توفرها Node.js. على سبيل المثال، http.IncomingMessage (طلبيات الخادم HTTP) و process.stdout هما نسخ من الصنف stream.
يمكن أن تكون المجاري قابلة للقراءة (readable)، أو قابلة للكتابة (writable)، أو كليهما. كل المجاري هي نسخ من الصنف EventEmitter.
يمكن الوصول إلى الوحدة stream
باستخدام:
const stream = require('stream');
لمَّا كان من الضروري فهم كيفية عمل المجاري، فإنَّ الوحدة stream
بحد ذاتها هي أكثر فائدةً للمطورين الذين ينشئون أنواعًا جديدةً من نسخ المجاري. المطورون الذين يستهلكون كائنات المجاري بالمقام الأول نادرًا ما يحتاجون لاستخدام الوحدة stream
بشكل مباشر.
تنظيم هذا المستند
يُقسّم هذا التوثيق إلى قسمين رئيسيين مع قسم ثالث للملاحظات الإضافية. يشرح القسم الأول عناصر الواجهة البرمجية للمجرى (stream API) التي تكون مطلوبة لاستخدام المجاري ضمن أي تطبيق. يشرح القسم الثاني عناصر الواجهة البرمجية (API) التي تُكون مطلوبةً لتطبيق أنواع جديدة من المجاري.
أنواع المجاري
يوجد أربعة أنواع رئيسية للمجاري ضمن Node.js هي:
- Writable: المجاري التي يمكن أن تُكتب عليها البيانات (مثل fs.createWriteStream())، و
- Readable: المجاري التي يمكن أن تُقرأ منها البيانات. (مثل fs.createReadStream())، و
- Duplex: المجاري التي تجمع بين النوع
Readable
والنوعWritable
(مثل net.Socket) - Transform: المجاري
Duplex
التي يمكن أن تعدّل أو تحوّل البيانات حسبما تُكتب أو تُقرأ (مثل zlib.createDeflate()).
بالإضافة إلى ذلك، تحوي هذه الوحدة الدوال الخدمية pipeline و finished.
نمط الكائن
كل المجاري المُنشأة من قبل واجهات Node.js تعمل حصريًا على السلاسل النصية والكائنات Buffer
(أو Uint8Array
). مع ذلك، فمن الممكن لتطبيقات المجاري أن تعمل مع أنواع بيانات أخرى في JavaScript (باستثناء النوع null
، الذي يخدم غرضًا خاصًا ضمن المجاري). مثل هذه المجاري يؤخذ بالحسبان تشغيلها في "نمط الكائن" (object mode).
تُبدَّل نُسَخ المجاري إلى نمط الكائن باستخدام الخيار objectMode
عند إنشاء المجرى. محاولة قلب مجرى موجود إلى نمط الكائن ليست آمنةً.
التخزين المؤقت Buffering
سوف يخزن كلا المجريين Writable و Readable البيانات في مخزن مؤقت داخلي لكي يمكن استعادتها باستخدام التابع writable.writableBuffer
أو readable.readableBuffer
على التوالي.
كمية البيانات القابلة للتخزين تعتمد على الخيار highWaterMark
المُمرر إلى باني المجرى. من أجل المجاري القياسية، يحدد الخيار highWaterMark
العدد الكلي من البايتات. أمَّا من أجل المجاري المُشغلة في نمط الكائن، فيحدد الخيار highWaterMark
العدد الكلي من الكائنات.
تُخزّن البيانات مؤقتًا في المجاري التي من النوع Readable
عندما يستدعي التابع stream.push(chunk). إذا لم يستدعي من يستخدم المجرى التابع stream.read()، فستتوضع البيانات ضمن طابور داخلي إلى أن تُستخدَم.
حالما يصل الحجم الكلي لذاكرة القراءة الداخلية المؤقتة عتبةَ محددة من قبل highWaterMark
، سيوقفُ المجرى مؤقتًا قراءة البيانات من المصادر الأساسية حتى تُستهلك البيانات الحالية المخزنة مؤقتًا (ذلك أنّ المجرى سيوقف استدعاء التابع readable._read()
المحلي الذي يُستخدم لوضع البيانات في ذاكرة القراءة المؤقتة).
تُخزّن البيانات مؤقتًا في المجاري التي من النوع Writable
عندما يُستدعى التابع writable.write(chunk) مرارًا وتكرارًا. طالما أنّ الحجم الكلي لذاكرة الكتابة المؤقتة الداخلية هو أدنى من العتبة المضبوطة بمقدار highWaterMark
، سوف يعيد استدعاء writable.write()
القيمة true
. حالما يصل أو يتجاوز حجم ذاكرة التخزين المؤقت الداخلية القيمة highWaterMark
، سوف تُعاد القيمة false
.
الهدف الأساسي من واجهات الوحدة stream
البرمجية - على وجه الخصوص التابع stream.pipe() - هو تقييد حجم التخزين المؤقت للبيانات إلى مستويات مقبولة من أجل التوافق بين المصادر (source) والوجهات (destination) ذات السرعات المختلفة لكي لا تمتلئ الذاكرة المتوافرة.
بما أن كلا المجريين Duplex و Transform يجمعان بين النوعين Readable
و Writable
، يمتلك كل واحد منها ذاكرتي تخزين مؤقتة. هاتان الذاكراتان داخليتين ومنفصلتين عن بعضهما وتستخدمان للقراءة والكتابة يف آن واحد، مما يسمح لكل طرف من المجرى بالتشغيل بشكل مستقل عن الآخر بينما يضمن تدفق بيانات مناسب وفعّال. على سبيل المثال، النُسخ net.Socket هي مجاري من النوع Duplex طرفها القابل للقراءة (Readable
) يسمح باستهلاك البيانات المُستقبَلة من المقبس وطرفها القابل للكتابة (Writable
) يسمح بكتابة البيانات على المقبس. بسبب أن البيانات قد تُكتب إلى المقبس بمعدل أسرع أو أبطأ من البيانات التي تُستقبل وتكتب على المجرى، فمن الضروري لكل طرف أن يعمل (ويخزِّن البيانات) بشكل مستقل عن الآخر.
الواجهات البرمجية لمستخدمي المجرى
كل تطبيقات Node.js تقريبًا، مهما كانت بسيطة، تستخدم المجاري بطريقةٍ ما. الآتي هو مثال لاستخدام المجاري في تطبيق Node.js والذي ينفّذ مُخدّم HTTP:
const http = require('http');
const server = http.createServer((req, res) => {
// والذي هو مجرى قابل للقراءة ،http.IncomingMessage هو Req
//والذي هو مجرى قابل للكتابة http.ServerResponse هو Req
let body = '';
//utf8 احصل على البيانات كسلاسل
//إذا لم يُجهّز الترميز ستُستقبل كائنات ذاكرة مؤقتة
req.setEncoding('utf8');
//حالما يُضاف مستمع 'data'تطلق المجاري القابلة للقراءة أحداث
req.on('data', (chunk) => {
body += chunk;
});
يشير الحدث 'end' أن الجسم الكامل قد استُقبل.
req.on('end', () => {
try {
const data = JSON.parse(body);
// اكتب شيئًا يثير اهتمام المستخدم
res.write(typeof data);
res.end();
} catch (er) {
// !سيئة json ،أوه
res.statusCode = 400;
return res.end(`error: ${er.message}`);
}
});
});
server.listen(1337);
// $ curl localhost:1337 -d "{}"
// object
// $ curl localhost:1337 -d "\"foo\""
// string
// $ curl localhost:1337 -d "not json"
// error: Unexpected token o in JSON at position 1
المجاري ذات النوع Writable (مثل res
في المثال) توفر توابع مثل write()
و end()
والتي تُستخدم لكتابة البيانات على المجرى.
المجاري ذات النوع Readable تستخدم واجهات الصنف EventEmitter البرمجية من أجل اشعار شيفرة التطبيق عندما تكون البيانات متوفرة للقراءة من المجرى. يمكن قراءة هذه البيانات المتوفرة من المجرى بعدة طرق.
تستخدم المجاري التي من النوع Writable و Readable واجهات الصنف EventEmitter البرمجية بطرق متنوعة للبقاء على اتصال دائم بالحالة الحالية للمجرى.
تذكر أن المجاري التي من النوع Duplex و Transform هي مجاري قابلة للكتاية (Writable) والقراءة (Readable).
لا يُطلب من التطبيقات التي إمّا تكتب البيانات أو تقرؤها من المجرى تنفيذ واجهات الوحدة stream بشكل مباشر، وبذلك لا يوجد عمومًا سبب لاستدعاء الوحدة stream عبر require('stream')
.
ينبغي على المطورين الراغبين بإنشاء أنواع جديدة من المجاري الرجوع إلى القسم الواجهات البرمجية لمنفذي المجاري.
المجاري القابلة للكتابة
المجاري القابلة للكتابة إجمالًا تمثِّل مكانًا قابلًا لكتابة البيانات فيه.
تتضمن الأمثلة التالية مجارٍ قابلة للكتابة:
- طلب HTTP، من طرف العميل.
- استجابة HTTP، من طرف الخادم.
- مجاري الوحدة fs القابلة للكتابة.
- مجاري الوحدة zlib.
- مجاري الوحدة crypto.
- المقابس TCP.
- العملية الابن للمجرى stdin (هي subprocess.stdin).
- المجرى process.stdout و process.stderr.
بعض هذه الأمثلة هي فعليًا مجاري من النوع Duplex والتي تنفّذ الواجهة Writable.
كل المجاري التي من النوع Writable تنفّذ الواجهة المعرّفة بالصنف stream.Writable
.
بينما قد تختلف نسخ المجاري التي من النوع Writable بطرق متنوعة، كل المجاري Writable
تتبع نمط الاستخدام الأساسي كما هو موضّح في المثال أدناه:
const myStream = getWritableStreamSomehow();
myStream.write('some data');
myStream.write('some more data');
myStream.end('done writing data');
الصنف stream.Writable
أضيف في الإصدار: 0.9.4.
الحدث 'close'
يُطلَق الحدث 'close'
عندما يكون المجرى أو أحد موارده الأساسية (واصف الملفات مثلًا) قد أُغلق. يشير هذا الحدث أنّه لن تُطلق المزيد من الأحداث، ولن تحصل المزيد من العمليات المتعلقة بالمجرى.
لا تُطلق كل المجاري Writable
الحدث 'close'
.
الحدث 'drain'
أُضيف في الإصدار:0.9.4.
إذا أعاد التابع stream.write(chunk) القيمة false
، فسوف يُطلَق الحدث 'drain'
عندما يكون من المناسب استئناف كتابة البيانات على المجرى.
//كتابة البيانات إلى المجرى القابل للكتابة المزوّد لمليون مرّة
//كن منتبهًا للضغط العائد
function writeOneMillionTimes(writer, data, encoding, callback) {
let i = 1000000;
write();
function write() {
let ok = true;
do {
i--;
if (i === 0) {
//آخر مرّة
writer.write(data, encoding, callback);
} else {
//انظر إذا كان ينبغي لنا الإستمرار أو الإنتظار
//لا تمرر دالة رد النداء لأنّنا لم ننته بعد
ok = writer.write(data, encoding);
}
} while (i > 0 && ok);
if (i > 0) {
//وجب التوقف باكرًا
//اكتب بعض المزيد حالما تنضب
writer.once('drain', write);
}
}
}
الحدث 'error'
أضيف في الإصدار: 0.9.4.
يُطلق الحدث 'error'
إذا حصل خطأ أثناء الكتابة على أو إرسال البيانات إلى المجرى. سوف يُمرَّر إلى دالة رد النداء المنصتة الوسيط Error
فقط عند استدعائها.
لن يُغلق المجرى عندما يُطلق الحدث 'error'
.
الحدث 'finish'
أضيف في الإصدار: 0.9.4.
سوف يُطلق الحدث 'finish'
بعد استدعاء التابع stream.end()، وبعد أن دُفعت كل البيانات للنظام الأساسي.
const writer = getWritableStreamSomehow();
for (let i = 0; i < 100; i++) {
writer.write(`hello, #${i}!\n`);
}
writer.end('This is the end\n');
writer.on('finish', () => {
console.error('All writes are now complete.');
});
الحدث 'pipe'
أُضيف في الإصدار: 0.9.4.
src
: <stream.Readable>مجرى القراءة والذي يُوصل إلى هذا المجرى القابل للكتابة.
يٌطلَق الحدث 'pipe'
عندما يُستدعى التابع stream.pipe() على مجرًى قابلٍ للقراءة مُضيفًا المجرى القابل للكتابة إلى مجموعة وجهاته.
const writer = getWritableStreamSomehow();
const reader = getReadableStreamSomehow();
writer.on('pipe', (src) => {
console.error('something is piping into the writer');
assert.equal(src, reader);
});
reader.pipe(writer);
الحدث 'unpipe'
أضيف في الإصدار: 0.9.4.
src
: <stream.Readable> مجرى المصدر الذي لغى الاتصال بهذا المجرى القابل للكاتبة.
يُطلَق الحدث 'unpipe'
عندما يُستدعى التابع stream.unpipe() على مجرًى قابلٍ للقراءة، مزيلًا المجرى القابل لكتابة من مجموعة وجهاته.
يُطلق هذا الحدث أيضًا في الحالة التي يطلق فيها المجرى القابل للكاتبة خطأً عندما يُوصل به مجرى قابل للقراءة.
const writer = getWritableStreamSomehow();
const reader = getReadableStreamSomehow();
writer.on('unpipe', (src) => {
console.error('Something has stopped piping into the writer.');
assert.equal(src, reader);
});
reader.pipe(writer);
reader.unpipe(writer);
writable.cork()
أضيف في الإصدار: 0.11.2.
يجبر التابع writable.cork()
كل البيانات المكتوبة على أن تُخزّن مؤقتًا في الذاكرة. ستُدفع البيانات المخزّنة عندما يُستدعى أحد التابعين stream.uncork() أو stream.end().
المقصد الأساسي من التابع writable.cork()
هو تجنب حالة يحدث فيها كتابة العديد من القطع الصغيرة من البيانات إلى المجرى دون جلب نسخةٍ منها إلى المخزن المؤقت الداخلي، إذ سيكون لذلك تأثير عكسي على الأداء. في مثل هذه الحالات، يمكن أن تنجز التطبيقات التي تنّفذ التابع writable._writev()
عملية الكتابة المخزّنة مؤقتًا بأسلوب أكثر مثاليةً.
انظر أيضًا: التابع writable.uncork().
writable.destroy([error])
أُضيف في الإصدار: 8.0.0
يهدم التابع المجرى، ويطلق الأحداث 'error'
و 'close'
المُمررة. بعد هذا الاستدعاء، تكون المجاري القابلة للكتابة قد انتهت والاستدعاءات اللاحقة للتابع write()
والتابع end()
سوف تُفضي إلى خطأ ERR_STREAM_DESTROYED
. لا ينبغي أن يعيد المنفّذون تعريف هذا التابع، ولكن يمكنهم أن يستعملوا التابع writable._destroy() بدلًا عن ذلك.
writable.end([chunk][, encoding][, callback])
سجل التغييرات
الإصدار | التغييرات | |
---|---|---|
10.0.0 | يعيد هذا التابع الآن مرجعًا إلى مجرًى من النوع writable .
| |
8.0.0 | يمكن الآن أن يكون الوسيط chunk نسخة من النوع Uint8Array .
| |
0.9.4 | أضيف هذا التابع في الإصدار 0.9.4 |
chunk
: <string> | <Buffer> | <Uint8Array> | <any> بيانات اختيارية يراد كتابتها. من أجل المجاري التي لا تعمل في وضع الكائن، يجب أن يكون الوسيطchunk
سلسلةً نصيةً (string) أو كائنًا من النوعBuffer
أو النوعUint8Array
. من أجل المجاري في نمط الكائن، يمكن أن تكونchunk
أي قيمة غيرnull
.encoding
: <string> التشفير، إذا كانتchunk
سلسلة نصية.- <Function> :
callback
دالة رد نداء اختيارية عندما يُنهى المجرى. - القيمة المعادة: <this>
يشير استدعاء التابع writable.end()
أنّه لا مزيد من البيانات يراد كتابتها على المجرى القابل للكتابة (Writable). سيسمح الوسيطان الاختياريان chunk
و encoding
أن تُكتب قطعة إضافية واحدة أخيرة من البيانات مباشرةً قبل إغلاق المجرى. إذا زُودت، سترفق دالة رد نداء callback
اختيارية كمنصت للحدث 'finish'.
سيثير استدعاء التابع stream.write() بعد استدعاء stream.end() خطأً.
// 'world!' ومن ثم أنهِ ب ' hello,' اكتب
const fs = require('fs');
const file = fs.createWriteStream('example.txt');
file.write('hello, ');
file.end('world!');
//كتابة المزيد ليست مسموحة الآن
writable.setDefaultEncoding(encoding)
سجل التغييرات
الإصدار | التغييرات | |
---|---|---|
6.1.0 | يعيد هذا التابع الآن مرجعًا إلى مجرًى من النوع writable .
| |
0.11.15 | أضيف في 0.11.15 | |
يضبط التابع writable.setDefaultEncoding()
الترميزَ الافتراضي (encoding
) للمجرى القابل للكتابة (Writable).
writable.uncork()
أُضيف في الإصدار: 0.11.2.
يفرِّغ التابع writable.uncork()
كل البيانات التي خُزِّنت منذ أن استُدعي stream.cork().
عند استخدام التابع writable.cork() والتابع ()writable.uncork
لإدارة التخزين المؤقت أثناء الكتابة على مجرًى، يُنصح بأن يؤخر استدعاء writable.uncork()
باستخدام process.nextTick()
. يسمح القيام بذلك بتجميع كل استدعاءات التابع writable.write()
التي تحصل خلال طورٍ واحدٍ لحلقة أحداث Node.js مُعطاة.
stream.cork();
stream.write('some ');
stream.write('data ');
process.nextTick(() => stream.uncork());
إذا استُدعي التابع writable.cork() عدة مرات لمجرىً ما، فيجب أن يُستدعى التابع writable.uncork()
نفس عدد الاستدعاءات لتفريغ البيانات المخزنة.
stream.cork();
stream.write('some ');
stream.cork();
stream.write('data ');
process.nextTick(() => {
stream.uncork();
//مرّة ثانية uncork()لن تُفرّغ البيانات حتى يُستدعى التابع
stream.uncork();
});
انظر أيضًا: التابع writable.cork().
writable.writableHighWaterMark
أُضيفت في الإصدار: 9.3.0.
- <number>
يعيد القيمة highWaterMark
التي مُرِّرت عند إنشاء المجرى القابل للكتابة (Writable
).
writable.writableLength
أُضيفت في الإصدار: 9.4.0.
تحوي هذه الخاصية عدد البايتات (أو الكائنات) الجاهزة للكتابة والموجودة في الطابور. توفر هذه القيمة بيانات داخلية متعلقة بحالة القيمة highWaterMark
.
writable.write(chunk[, encoding][, callback])
سجل التغييرات
الإصدار | التغييرات | |
---|---|---|
8.0.0 | يمكن الآن أن يكون الوسيط نسخة من النوع Uint8Array. | |
6.0.0 | سوف يعتبر دائمًا تمريرالقيمة null كمعامل chunk غير صالح الآن، حتى في نمط الكائن. | |
0.9.4 | أُضيف هذا التابع. |
-
chunk
: <string> | <Buffer> | <Uint8Array> | <any> بيانات اختيارية للكتابة. من أجل المجاري التي لا تعمل في نمط الكائن، يجب أن تكونchunk
سلسلة نصية أوBuffer
أوUint8Array
. من أجل مجاري نمط الكائن، يمكن أن تكون chunk أي نوع من أنواع بيانات JavaScript غيرnull
. - <string> :
encoding
الترميز، إذا كان الوسيطchunk
سلسلةً نصيةً. callback
: <Function> دالة رد النداء المراد استدعاؤها عندما تُدفع قطعة البيانات.- القيمة المُعادة: <boolean> قيمة منطقية تكون
false
إذا كان المجرى يريد للشيفرة المستدعية أن تنتظر انطلاق الحدث'drain'
قبل الاستمرار بكتابة المزيد من البيانات؛ وإلّا تكونtrue
.
يكتب التابع writable.write()
بعض البيانات إلى المجرى. ويستدعي رد النداء (callback
) المُزوّد حالما تكون البيانات المكتوبة قد عولجت بشكل كامل. إذا حصل خطأ، ربما يُستدعى callback
أو لا مع تمرير الخطأ كوسيطه الأول. للكشف عن أخطاء الكتابة بشكل موثوق، أضف مُنصتًا إلى الحدث 'error'
.
القيمة المعادة هي true
إذا كانت ذاكرة التخزين الداخلية أقل من القيمة highWaterMark
المُهيئة عندما أُنشئ المجرى بعد إقرار chunk
. إذا أُعيد false
، ينبغي أن تتوقف المحاولات الإضافية لكتابة البيانات على المجرى حتى يُطلق الحدث 'drain'.
طالما أن المجرى لم يفرغ، فإن استدعاء write()
سوف يخزّن chunk
مؤقتًا، ويعيد خطأ. حالما تُستهلك كل القطع الحالية المخزنة مؤقتًا (قُبلَت للتسليم من قبل نظام التشغيل)، سوف يُطلق الحدث 'drain'
. حين يعيد write()
القيمة false
، فمن المستحسن ألّا تكتب المزيد من القطع حتى يُطلق الحدث 'drain'
. بينما يُسمح باستدعاء write()
على مجرًى لم يفرغ بعد، فسوف تخزن Node.js مؤقتًا كل القطع المكتوبة حتى يحصل استخدام ذاكرة أعظمي،عند هذه النقطة ستتوقف دون قيود. حتى قبل حصول التوقف، سيسبب الاستخدام المرتفع للذاكرة أداءًا ضعيفًا لجامع القمامة (منظّف الذاكرة) وارتفاع مجموعة الذاكرة المُقِيمة (RSS) [والذي لن يعاد استرجاعه للنظام بشكل طبيعي، حتى بعد عدم الحاجة للذاكرة]. بما أن المقابس TCP قد لا تنضب أبدًا إذا لم يقرأ الند (peer) البعيد البيانات، ربما تقود الكتابة على مقبس لا يُصرّف لحصول ثغرة قابلة للاستغلال (exploitable vulnerability).
كتابة البيانات على مجرى لا ينفد هو معضلة خصوصًا مع مجاري التحويل ( Transform)، لأنّ مجاري Transform
تتوقف افتراضيًا حتى تنقل تُتصل بمجرى آخر أو يضاف معالج للحدث 'data'
أو الحدث 'readable'
.
اذا أمكن توليد البيانات التي ستُكتب أو جلبها عند الطلب، فمن المستحسن تغليفها في نفس المجرى ثم تحويلها إلى المجرى القابل للقراءة عبر استخدام التابع stream.pipe().
ولكن إذا كان رُجّج استدعاء write()
، فمن الممكن مراعاة الضغط العائد وتجنب مشاكل الذاكرة باستخدام الحدث 'drain':
function write(data, cb) {
if (!stream.write(data)) {
stream.once('drain', cb);
} else {
process.nextTick(cb);
}
}
//قبل القيام بأي كتابة أخرى cb الانتظار حتى يُستدعى
write('hello', () => {
console.log('write completed, do more writes now');
});
سوف يهمل المجرى Writable
في نمط الكائن الوسيط encoding
دائمًا.
المجاري القابلة للقراءة
المجاري القابلة للقراءة هي فكرة مجرّدة عن مصدرٍ تُستهلك منه البيانات.
أمثلة عن المجاري المقروءة (Readable
) هي:
- استجابة لطلبيات HTTP من طرف العميل.
- طلبيات HTTP من طرف الخادم.
- مجاري الوحدة fs القابلة للقراءة.
- مجاري الوحدة zlib.
- مجاري الوحدة crypto.
- مقابس TCP.
- المجرى stdout والمجرى stderr للعملية الابن.
- المجرى Process.stdin للعملية الحالية.
كل المجاري القابلة للقراءة (Readable) تطبّق الواجهة المعرّفة من قبل الصنف stream.Readable
.
نمطا القراءة
تعمل المجاري القابلة للقراءة (Readable
) بفاعلية في أحد النمطين التاليين: نمط التدفق (flowing mode) ونمط التوقف المؤقتً (paused mode). هذه الأنماط منفصلة عن نمط الكائن. يمكن أن يكون المجرى المقروء في نمط الكائن أو لا، بغض النظر عن إذا كان في نمط التدفق أو نمط التوقف المؤقت.
- في نمط التدفق، تُقرأ البيانات من النظام الرئيسي بشكل تلقائي وتُقدّم للتطبيق بأسرع ما يمكن باستخدام الأحداث عبر الواجهة EventEmitter.
- في نمط التوقف المؤقت، يجب أن يُستدعى stream.read() بشكل صريح لقراءة قطع البيانات من المجرى.
تبدأ كل المجاري القابلة للقراءة في نمط التوقف المؤقت ولكن يمكن تبديلها إلى نمط التدفق بإحدى الطرق التالية:
- إضافة معالج للحدث 'data'.
- استدعاء التابع stream.resume().
- استدعاء التابع stream.pipe() لإرسال البيانات إلى المجرى القابل للكتابة (Writable).
يمكن نقل Readable
بالعكس إلى نمط التوقف باستخدام أحد التالي:
- إذا لم يكن هناك أنبوب متصل بالوجهات، فيمكن ذلك عبر استدعاء التابع stream.pause().
- إذا كان هناك أنبوب متصل بالوجهات، فيمكن ذلك عبر إزالة كل أنابيب الوجهات. قد تُزال العديد من أنابيب الوجهات باستدعاء التابع stream.unpipe().
المبدأ الأساسي الذي يجب تذكره دومًا هو أنّ المجرى القابل للقراءة (Readable
) لن يولّد بيانات حتى تتوافر آلية إمّا لاستهلاك تلك البيانات أو تجاهلها. إذا كانت آلية الاستهلاك معطلة أو مُستبعدة، سوف يحاول Readable
إيقاف توليد البيانات.
من أجل التوافقية مع الإصدارات السابقة، لن توقف ازالة معالج حدث 'data' المجرى تلقائيًا. أيضًا، إذا كان هنالك ممرات للوجهات، فحين ذلك لن يضمَن استدعاء stream.pause() أن يبقى المجرى متوقفًا حالما تفرغ هذه الوجهات وتسأل عن المزيد من البيانات.
إذا قُلب Readable إلى نمط التدفق ولم يتوفر مستهلكون لمعالجة البيانات، ستضيع تلك البيانات. يمكن أن يحصل هذا على سبيل المثال، عندما يُستدعى تابع readable.resume()
دون منصت مرفق بحدث 'data'
، أو عندما يُزال معالج حدث 'data'
من المجرى.
إضافة معالج حدث 'readable' يجعل توقف تدفق المجرى تلقائيًا، وتُتستهلك البيانات بواسطة readable.read(). إذا أُزيل معالج حدث 'readable'، حينذاك سيبدأ المجرى بالتدفق مُجدّدًا إذا كان هناك معالج حدث 'data' موجود.
الحالات الثلاث
هناك "وضعا" تشغيل للمجرى القابل للقراءة (Readable
)، هما تلخيص مبسّط لإدارة حالات داخلية كثر تعقيدًا والتي تحصل خلال تطبيق مجرى قابل للقراءة (Readable
).
على وجه التحديد وفي أي نقطة معطاة من الزمن يأخذ كل مجرى القابل للقراءة (Readable
) إحدى الحالات الثلاث التالية:
readable.readableFlowing === null
readable.readableFlowing === false
readable.readableFlowing === true
عندما تكون readable.readableFlowing
هي null
، لن تكون هناك آلية مُقدمة لاستهلاك بيانات المجرى. لذلك، لن يولد المجرى بيانات. بينما في هذه الحالة، سيبدل إرفاق مُنصت إلى الحدث 'data'
أو استدعاء التابع readable.pipe()
أو استدعاء التابع readable.resume()
قيمة الخاصية readable.readableFlowing
إلى القيمة true
، متسببًا ببدء Readable
بإطلاق الأحداث بفاعلية حين توّلد البيانات.
سوف يتسبب استدعاء readable.pause()
أو readable.unpipe()
أو استقبال الضغط العائد بضبط قيمة الخاصية readable.readableFlowing
إلى false
، قاطعًا تدفق الأحداث مؤقتًا ولكن غير قاطعٍ لتدفق البيانات. بينما لن يقلب إرفاق منصت بالحدث 'data'
قيمة الخاصية readable.readableFlowing
إلى true
في هذه الحالة.
const { PassThrough, Writable } = require('stream');
const pass = new PassThrough();
const writable = new Writable();
pass.pipe(writable);
pass.unpipe(writable);
//التدفق القابل للقراءة متوقف الآن
pass.on('data', (chunk) => { console.log(chunk.toString()); });
pass.write('ok');
// 'data'لن تطلق الحدث
pass.resume();
//'data' يجب أن تُستدعى لجعل المجرى يطلق الحدث
بينما تكون قيمة readable.readableFlowing
هي false
، قد تتراكم البيانات داخل الذاكرة المؤقتة الداخلية للمجرى.
اختر أسلوب واجهة برمجية وحيد
تطورت الواجهات البرمجية (API) للمجرى القابل للقراءة (Readable
) عبر عدة إصدارات Node.js وقدّمت عدّة توابع لاستهلاك بيانات المجرى. بشكل عام، ينبغي أن يختار المطورون واحدة من توابع استهلاك البيانات ولا ينبغي استخدام عدة توابع لاستهلاك البيانات من مجرى واحد. بشكل خاص، قد يقود استخدام مزيج من on('data')
أو on('readable')
أو pipe()
أو توابع تكرارية غير متزامنة إلى سلوك غير متوقع.
ينصح باستخدام التابع readable.pipe()
لأغلب المستخدمين بما أنه نُفّذ لتقديم أسهل طريقة لاستهلاك بيانات المجرى.
يمكن أن يستخدم المطورون الذين يطلبون المزيد من التحكم الدقيق على نقل وتوليد البيانات الصنف EventEmitter و readable.on('readable')
/readable.read()
أو الواجهات readable.pause()
/readable.resume()
.
الصنف stream.Readable
أُضيف في الإصدار: 0.9.4.
الحدث:'close'
أُضيف في الإصدار: 0.9.4.
يُطلق الحدث'close'
عندما يكون المجرى أو أحد موارده الأساسية (واصف الملف مثلًا) قد أُغلق. يشير الحدث أنّه لن تُطلق مزيد من الأحداث ولن يحصل المزيد من العمليات المتعلقة بالمجرى.
لا تُطلق كل المجاري القابلة للقراءة الحدث 'close'
.
الحدث:'data'
أُضيف في الإصدار: 0.9.4.
- <Buffer> | <string> | <any> :
chunk
قطع البيانات. من أجل المجاري التي لا تعمل في نمط الكائن، ستكون القطعة إمّا سلسلة نصية أو كائنًا من النوع Buffer. من أجل المجاري في نمط الكائن، يمكن أن تكون القطعة أي نوع من أنواع البيانات باستثناء null.
سيُطلق الحدث 'data'
كلما تخلى المجرى عن حيازة قطعة من البيانات إلى المستهلك. قد يحصل هذا كلّما بُدِّل المجرى في نمط الكائن باستدعاء readable.pipe()
أو readable.resume()
أو بإرفاق رد نداء منصت للحدث 'data'
. سوف يُطلق الحدث 'data'
أيضًا كلّما استُدعي التابع readable.read()
وكانت قطعة من البيانات متوفرة لتُعاد.
إرفاق مُنصت بالحدث 'data'
لمجرىً لم يُوقَف بشكل صريح سوف يقلب المجرى إلى نمط التدفق. ستُمرر البيانات بعدئذٍ حالما تكون متوفرة.
سوف يمرر ردُ نداء المُنصت قطعةَ البيانات كسلسلة نصية إذا كان التشفير الافتراضي مُحدّدًا باستخدام التابع readable.setEncoding()
، وإلّا ستمرر البيانات ضمن كائن من النوع Buffer
.
const readable = getReadableStreamSomehow();
readable.on('data', (chunk) => {
console.log(`Received ${chunk.length} bytes of data.`);
});
الحدث: 'end'
أُضيف في الإصدار: 0.9.4.
يُطلق الحدث 'end'
عندما لا يكون هناك المزيد من البيانات لتُقرَأ من المجرى.
لن يُطلق الحدث 'end'
مالم تُستهلك البيانات بالكامل. يمكن أن يُحقَّق ذلك بقلب المجرى إلى نمط التدفق، أو باستدعاء stream.read() مرارًا حتى يتم استهلاك كل البيانات.
const readable = getReadableStreamSomehow();
readable.on('data', (chunk) => {
console.log(`Received ${chunk.length} bytes of data.`);
});
readable.on('end', () => {
console.log('There will be no more data.');
});
الحدث: 'error'
أُضيف في الإصدار: 0.9.4.
- <Error>
ربما يُطلق الحدث 'error'
من قبل إجراء مجرى قابل للقراءة (Readable
) في أي وقت. عادةً، قد يحصل ذلك إذا كان المجرى الأساسي غير قادر على توليد بيانات بسبب فشل داخلي أساسي، أو عندما يحاول إجراء المجرى دفع قطعة بيانات غير صالحة.
سيُمرَّر إلى دالة رد نداء المُنصت كائنٌ من النوع Error
فقط.
الحدث: 'readable'
سجل التغييرات
الإصدار | التغييرات | |
---|---|---|
10.0.0 | يُطلق الحدث 'readable' دائمًا في النبضة التالية بعد أن يُستدعى .push()
| |
10.0.0 | يتطلب استخدام 'readable' استدعاءَ .read()
| |
0.9.4 | أُضيف في الإصدار: 0.9.4. |
يُطلق التابع 'readable'
عندما يكون هناك بيانات متوفرة للقراءة من المجرى. في بعض الحالات، سوف يسبب ربط مُنصت بالحدث 'readable'
أن تُقرَأ بعض كمية البيانات إلى ذاكرة مؤقتة داخلية.
const readable = getReadableStreamSomehow();
readable.on('readable', function() {
//يوجد بعض البيانات لتُقرأ الآن
let data;
while (data = this.read()) {
console.log(data);
}
});
سوف يُطلق الحدث 'readable'
أيضًا حالما يتم التوصل إلى نهاية بيانات المجرى ولكن قبل أن يُطلق الحدث 'end'
.
فعليًا، يشير الحدث 'readable'
أن المجرى يملك معلومات جديدة وهي: إمّا بيانات جديدة متوفرة أو تم الوصول إلى نهاية المجرى. في الحالة الأولى، سوف يعيد stream.read() البيانات المتوفرة. في الحالة الثانية، سوف يعيد stream.read() القيمة null
. على سبيل المثال، في المثال التالي، foo.txt
هو ملف فارغ:
const fs = require('fs');
const rr = fs.createReadStream('foo.txt');
rr.on('readable', () => {
console.log(`readable: ${rr.read()}`);
});
rr.on('end', () => {
console.log('end');
});
خرج تشغيل هذا السكربت هو:
$ node test.js
readable: null
end
بشكل عام، فإنّ آليات readable.pipe()
والحدث 'data'
هي أسهل للفهم من الحدث 'readable'
. ولكن، معالجة 'readable'
قد تنتج زيادة بمعدل الانتاجية .
إذا استُخدم كلا الحدثين 'readable'
و 'data' في نفس الوقت، فإن الحدث 'readable'
يأخذ الأولوية في التحكم بالتدفق؛ أي أنّ الحدث 'data'
سوف يُطلق فقط إذا استُدعي stream.read(). قد تصبح قيمة الخاصية readableFlowing
هي false
. إذا كان هناك مُنصتات للحدث 'data'
عندما يُزال الحدث 'readable'
، سيبدأ المجرى بالتدفق؛ أي أن الأحداث 'data'
سوف تُطلق دون استدعاء .resume()
readable.destroy([error])
أُضيف في الإصدار: 8.0.0.
يهدم التابع المجرى ويطلق الحدث 'error'
و 'close'
. بعد هذا الاستدعاء، سوف يحرر المجرى القابل للقراءة أيّة موارد داخلية وسوف تُتجاهل الاستدعاءات اللاحقة للتابع push()
. لا ينبغي على المنفذين إعادة تعريف هذا التابع، ولكن يمكنهم تطبيق readable._destroy() بدلًا عنه.
readable.isPaused()
أُضيف في الإصدار: 0.11.14.
- القيمة المُعادة: <boolean>
يعيد التابع readable.isPaused()
حالة التشغيل الحالية للمجرى Readable
. يُستخدم هذا التابع بالمقام الأول من قبل آليات ترتكز على التابع readable.pipe()
. في معظم الحالات الطبيعية، لا يوجد حاجة لاستخدام هذا التابع بشكل مباشر.
const readable = new stream.Readable();
readable.isPaused(); // === false
readable.pause();
readable.isPaused(); // === true
readable.resume();
readable.isPaused(); // === false
readable.pause()
أضيف في الإصدار: 0.9.4.
القيمة المُعادة: <this>
سوف يسبب التابع readable.pause()
وقف إطلاق أحداث 'data' في نمط التدفق، مع إخراج المجرى من نمط التدفق. أية بيانات تصبح متوفرة ستبقى في الذاكرة المؤقتة الداخلية.
const readable = getReadableStreamSomehow();
readable.on('data', (chunk) => {
console.log(`Received ${chunk.length} bytes of data.`);
readable.pause();
console.log('There will be no additional data for 1 second.');
setTimeout(() => {
console.log('Now data will start flowing again.');
readable.resume();
}, 1000);
});
لا يملك التابع readable.pause()
أي تأثير إذا كان هناك مُنصت للحدث 'readable'
.
readable.pipe(destination[, options])
أُضيف في الإصدار: 0.9.4.
- <stream.Writable> :
destination
الوجهة التي ستُكتَب فيها البيانات. - <Object> :
options
خيارات النقل.- <boolean> :
end
إنهاء الكاتب عندما ينتهي القارئ. القيمة الافتراضية:true
.
- <boolean> :
- القيمة المُعادة: <stream.Writable> تعاد الوجهة destination سامحةً بسلسلة من الأنابيب والممرات إذا كانت المجاري من النوع المزدوج (Duplex) أو مجاري تحويل (Transform).
يربط التابع readable.pipe()
مجرًى قابلًا للكتابة مع مجرًى قابلٍ للقراءة (Readable
) مسببًا قلبه تلقائيّا إلى نمط التدفق ودفع جميع بياناته إلى المجرى القابل للكتابة المرتبط به. أي كأن هذا التابع يضع أنبوبًا بين المجريين لتتدفق البيانات من أحدهما إلى الآخر عبره. سيضبط تدفق البيانات تلقائيًا لذلك لن يُغمر مجرى Writable
الهدف من قبل مجرى قابل للقراءة(Readable
) أسرع منه.
ينقل المثال التالي كل البيانات من المجرى readable
إلى ملف اسمه file.txt
:
const fs = require('fs');
const readable = getReadableStreamSomehow();
const writable = fs.createWriteStream('file.txt');
//'file.txt' تذهب كل البيانات من المجرى القابل للقراءة إلى
readable.pipe(writable);
من الممكن ربط عدة مجاري قابلة للكتابة (Writable
) مع مجرى واحد قابل للقراءة (Readable
).
يعيد التابع readable.pipe()
مرجعًا إلى مجرى الوجهة جاعلًا من الممكن إقامة سلسلة من المجاري المتصلة ببعضها بأنابيب:
const fs = require('fs');
const r = fs.createReadStream('file.txt');
const z = zlib.createGzip();
const w = fs.createWriteStream('file.txt.gz');
r.pipe(z).pipe(w);
بشكل افتراضي، يُستدعى stream.end() على مجرى الوجهة القابل للكتابة (Writable
) عندما يطلق مجرى Readable
المصدر الحدث 'end'. لذلك لن تكون الوجهة قابلةً للكتابة بعد الآن. لتعطيل هذا السلوك الافتراضي، يمكن أن تمرير القيمة false
إلى الخيار end
مسبّبًا بقاء الوجهة مفتوحة:
reader.pipe(writer, { end: false });
reader.on('end', () => {
writer.end('Goodbye\n');
});
أحد التحذيرات المهمة أنّه إذا أطلق المجرى Readable
خطأً أثناء المعالجة، لن يُغلق مجرى الوجهة Writable
تلقائيًا، لذا من الضروري في هذه الحالة إغلاق كل مجرى يدويًا لمنع التسريب في الذاكرة.
لن يغلق المجريان process.stderr و process.stdout القابلان للكتابة (Writable
) أبدًا حتى تنتهي عملية Node.js بغض النظر عن الخيارات المحدّدة.
readable.read([size])
أضيف في الإصدار: 0.9.4.
- <number> :
size
وسيط اختياري يحدد كمية البيانات للقراءة. - القيمة المعادة: <string> | <Buffer> | <null> | <any>
يسحب التابع readable.read()
بعض البيانات من الذاكرة المؤقتة الداخلية ويعيدها. إذا لم يكن هناك بيانات متوافرة للقراءة، يعيد التابع القيمة null
. بشكل افتراضي، ستعاد البيانات في كائن من النوع Buffer
إلّا إذا حُدد الترميز باستخدام التابع readable.setEncoding()
أو إذا كان المجرى يشتغل في نمط الكائن.
يحدد الوسيط الاختياري size
عدد البايتات المحدد للقراءة. إذا لم يكن هنالك بايتات بالحجم size
متوافرة للقراءة، ستُعاد القيمة null
إلّا إذا كان المجرى قد انتهى. في هذه الحالة، ستُعاد كل البيانات المتبقية في ذاكرة التخزين المؤقت الداخلية.
إذا لم يكن الوسيط size
محدّدًا، ستُعاد كل البيانات المحتواة في ذاكرة التخزين المؤقت الداخلية.
ينبغي أن يُستدعى التابع readable.read()
على مجاري Readable
المُشغّلة في نمط التوقف المؤقت فقط، في نمط التدفق، يُستدعى readable.read()
تلقائيًا حتى فراغ ذاكرة التخزين المؤقت الداخلية بالكامل.
const readable = getReadableStreamSomehow();
readable.on('readable', () => {
let chunk;
while (null !== (chunk = readable.read())) {
console.log(`Received ${chunk.length} bytes of data.`);
}
})
سيعيد المجرى Readable
في نمط الكائن عنصرًا وحيدًا دائمًا من استدعاء readable.read(size) بغض النظر عن عن قيمة الوسيط size
.
إذا أعاد التابع readable.read()
قطعة من البيانات، سيُطلق أيضًا الحدث 'data'
.
استدعاء stream.read([size]) بعد أن يكون الحدث 'end' قد أُطلق سيعيد null
. لن تُطلَق أي أخطاء وقت التشغيل (runtime).
readable.readableHighWaterMark
أضيفت في الإصدار: 9.3.0.
- القيمة المعادة: <number>
تعيد قيمة الخاصية highWaterMark
المُمرّرة عند إنشاء مجرى Readable
هذا.
readable.readableLength
أُضيفت في الإصدار:9.4.0.
- القيمة المعادة:<number>
تحوي هذه الخاصية عدد البايتات (أو الكائنات) الجاهزة للقراءة في الطابور. تقدم هذه القيمة البيانات الداخلية المخزنة بما يتوافق مع حالة highWaterMark
.
readable.resume()
سجل التغييرات
الإصدار | التغييرات | |
---|---|---|
10.0.0 | لا يملك resume() تأثير إذا كان هناك استماع للحدث 'readable' .
| |
0.9.4 | أُضيف في الإصدار 0.9.4. |
- القيمة المعادة: <this>
يسبب التابع readable.resume()
إيقاف صريح لمجرى Readable
لاستئناف إطلاق أحداث 'data'، قالبًا المجرى إلى نمط التدفق.
يمكن أن يُستخدم التابع readable.resume()
لاستهلاك كامل للبيانات من المجرى دون معالجة أي منها فعليًا.
getReadableStreamSomehow()
.resume()
.on('end', () => {
console.log('Reached the end, but did not read anything.');
});
لا يملك التابع readable.resume()
تأثير إذا كان هناك مُنصت للحدث 'readable'
.
readable.setEncoding(encoding)
أُضيف في الإصدار: 0.9.4.
يضبط التابع readable.setEncoding()
ترميز المحارف للبيانات المقروءة من المجرى Readable
.
بشكل افتراضي، لا يُخصص أي ترميز وستعاد بيانات المجرى في كائنات من النوع Buffer
، ضبط الترميز يتسبب بأن تُعاد بيانات المجرى كسلسلة نصية مرمّزة بالترميز المحدد بدلًا من إعادتها في كائنات من النوع Buffer
. على سبيل المثال، سيسسب استدعاء readable.setEncoding('utf8')
بأن تُفسر بيانات الخرج كبيانات مرمزة بالترميز UTF-8، وتُمرر كسلاسل نصية. سوف يسبب استدعاء readable.setEncoding('hex')
أن تُرمّز البيانات بشكل سلسلة نصية ستة عشرية.
سيعالج المجرى Readable
المحارف متعددة البايتات المُستلمة خلال المجرى بصورة صحيحة وإلّا قد تصبح مرمّزة بشكل غير صحيح إذا سُحبت من المجرى ببساطة في كائنات من النوع Buffer
.
const readable = getReadableStreamSomehow();
readable.setEncoding('utf8');
readable.on('data', (chunk) => {
assert.equal(typeof chunk, 'string');
console.log('got %d characters of string data', chunk.length);
});
readable.unpipe([destination])
أضيف في الإصدار:0.9.4.
- <stream.Writable> :
destination
مجرى محدد اختياريًا - القيمة المُعادة: <this>
يفصل التابع readable.unpipe()
مجرى Writable
المربوط مؤخرًا باستخدام التابع stream.pipe().
إذا لم يُحدد مجرى الوجهة (destination
)، ستفصل حينذاك كل الأنابيب(الممرات).
إذا كان destination
محددًا، ولكن لا يوجد أنابيب(قنوات) مُنصّبة عليه، لن يفعل التابع حينذاك شيئًا.
const fs = require('fs');
const readable = getReadableStreamSomehow();
const writable = fs.createWriteStream('file.txt');
// 'file.txt' إلى readable تذهب كل البيانات من المجرى القابل للقراءة
//ولكن فقط للثانية الأولى
readable.pipe(writable);
setTimeout(() => {
console.log('Stop writing to file.txt');
readable.unpipe(writable);
console.log('Manually close the file stream');
writable.end();
}, 1000);
readable.unshift(chunk)
سجل التغييرات
الإصدار | التغييرات | |
---|---|---|
8.0.0 | يمكن أن يكون الآن الوسيط chunk نسخةَ Uint8Array .
| |
0.9.11 | أُضيف في الإصدار: 0.9.11. | |
- <Buffer> | <Uint8Array> | <string> | <any> :chunk قطع البيانات المراد منعها من القفز (الإزاحة) فوق الطابور. من أجل المجاري غير المشغّلة في نمط الكائن، يجب أن تكون
chunk
سلسلة نصية أو كائنًا من النوعBuffer
أو النوعUint8Array
. من أجل المجاري في نمط الكائن، يمكن أن يقبلchunk
أي نوع من أنواع بيانات JavaScript باستثناءnull
.
يدفع التابع readable.unshift()
قطع البيانات خلفًا إلى ذاكرة التخزين المؤقتة الداخلية. هذا مفيد في حالات معيّنة تُقرَأ فيها بيانات محدَّدة لا حاجة لقراءتها من المجرى، لذلك يُمكن أن تُزاح هذه البيانات لتُمرر أخرى إلى طرف آخر.
هذا مفيد في حالات معينة حيث يُستهلك المجرى بشيفرات تتطلب "عدم استهلاك" لبعض كمية البيانات المسحوبة بشكل مثالي من المصدر، لذلك يمكن أن تُمرر البيانات إلى بعض الأطراف الأخرى.
لايمكن أن يُستدعى التابع stream.unshift(chunk)
بعد أن أُطلق الحدث 'end' أو سوف يُرمى خطأ وقت التشغيل.
ينبغي على المطورين المستخدمين للتابع stream.unshift()
التبديل إلى استعمال مجرى التحويل (Transform) بدلًا عن المجرى Readable. انظر مقطع الواجهات البرمجية لمنفذي المجاري للمزيد من المعلومات.
// \n\nانتزاع العنوان المحدّد ب
//إذا حصلت على الكثير unshift()استخدم
//(error, header, stream) (استدعِ دالة رد النداء مع (الخطأ، المجرى، العنوان
const { StringDecoder } = require('string_decoder');
function parseHeader(stream, callback) {
stream.on('error', callback);
stream.on('readable', onReadable);
const decoder = new StringDecoder('utf8');
let header = '';
function onReadable() {
let chunk;
while (null !== (chunk = stream.read())) {
const str = decoder.write(chunk);
if (str.match(/\n\n/)) {
//أوجد حدود العنوان
const split = str.split(/\n\n/);
header += split.shift();
const remaining = split.join('\n\n');
const buf = Buffer.from(remaining, 'utf8');
stream.removeListener('error', callback);
// unshift() قبل استدعاء 'readable' أزل مُنصت
stream.removeListener('readable', onReadable);
if (buf.length)
stream.unshift(buf);
// يمكن الآن قراءة جسم الرسالة من المجرى
callback(null, header, stream);
} else {
//لا يزال يقرأ العنوان
header += str;
}
}
}
}
خلافًا للتابع stream.push(chunk)، لن ينهي التابع stream.unshift(chunk)
عملية القراءة بإعادة تصفير حالة القراءة الداخلية للمجرى. يمكن أن يسبب هذا نتائج غير متوقعة إذا استُدعي readable.unshift()
أثناء القراءة (أي من داخل تنفيذ stream._read() على مجرى مخصص).
اتباع استدعاء التابع readable.unshift()
بالتابع stream.push('') مباشرةً سوف يعيد تصفير حالة القراءة بشكل مناسب، ولكن الأفضل ببساطة هو تجنب استدعاء readable.unshift()
أثناء عملية تنفيذ القراءة.
readable.wrap(stream)
أضيف في الإصدار: 0.9.4.
قبل الإصدار Node.js 0.10، لم تطبّق المجاري كامل الواجهة البرمجية للوحدة stream
كما هي معرّفة الآن. (انظر التوافق مع إصدارات Node.js الأقدم للمزيد من المعلومات.)
عند استخدام مكتبات Node.js أقدم والتي تطلق الأحداث 'data' وفيها التابع stream.pause() الذي هو إرشادي فقط، يمكن أن يُستخدم التابع readable.wrap()
لإنشاء مجرًى قابل للقراءة يستخدم المجرى القديم كمصدر لبياناته.
سيكون من النادر الحاجة إلى استخدام readable.wrap()
ولكن التابع قُدّم كملائمة للتفاعل مع تطبيقات Node.js أقدم.
const { OldReader } = require('./old-api-module.js');
const { Readable } = require('stream');
const oreader = new OldReader();
const myReader = new Readable().wrap(oreader);
myReader.on('readable', () => {
myReader.read(); //...الخ
});
readable[Symbol.asyncIterator]()
أضيف في الإصدار: 10.0.0.
الاستقرار: 1- تجريبي
القيمة المعادة: <AsyncIterator> لاستهلاك كامل المجرى.
const fs = require('fs');
async function print(readable) {
readable.setEncoding('utf8');
let data = '';
for await (const k of readable) {
data += k;
}
console.log(data);
}
print(fs.createReadStream('file')).catch(console.log);
إذا انتهت الحلقة التكرارية ب break
أو throw
، فسوف يُدمّر المجرى. بعبارة أخرى، سوف يستهلك التكرار عبر المجرى جميع بياناته ويقرأ المجرى كاملًا. سيُقرَأ المجرى بقطع بيانات ذات حجم مساوٍ لقيمة الخيار highWaterMark
. في مثال الشيفرة في الأعلى، ستكون البيانات في قطعة وحيدة إذا حوى الملف على أقل من 64 كيلو بت من البيانات لأنه قيمة الخيار highWaterMark
لم تُقدّم للتابع fs.createReadStream().
المجاري المزدوجة (Duplex) ومجاري التحويل (Transform)
الصنف stream.Duplex
سجل التغييرات
الإصدار | التغييرات | |
---|---|---|
6.8.0 | ستعيد الآن نُسخ Duplex القيمة true عند فحص instanceof stream.Writable
| |
0.9.4 | أضيف في 0.9.4. | |
المجاري المزدوجة (Duplex
) هي المجاري التي تطبّق كلا واجهتي المجرى القابل للقراءة (Readable) والمجرى القابل للكتابة (Writable).
تتضمن أمثلة مجاري Duplex:
الصنف: stream.Transform
أُضيف في الإصدار: 0.9.4.
مجاري التحويل (Transform streams) هي مجاري من النوع المزدوج (Duplex) حيث أنَّ الخرج مرتبط بطريقة ما مع الدخل. مثل كل مجاري Duplex، تطبّق مجاري Transform
كلا واجهتي المجرى القابل للقراءة (Readable) والمجرى القابل للكتابة (Writable).
من الأمثلة عن المجاري Transform
:
transform.destroy([error])
أُضيف في الإصدار: 8.0.0.
- <Error> :error
يدّمر المجرى، ويطلق الحدث 'error'
. بعد هذا الاستدعاء، سيحرر مجرى التحويل أي مصادر داخلية، ينبغي على المستخدمين ألّا يعيدو تعريف هذا التابع، ولكن يمكنهم تطبيق readable._destroy() بدلًا عنه. التنفيذ الافتراضي لتابع _destroy()
من أجل المجرى Transform
يطلق أيضًا الحدث 'close'
.
stream.finished(stream, callback)
أُضيف في الإصدار: 10.0.0.
- <Stream> :
stream
مجرى قابل للقراءة و/أو قابل للكتابة. - <Function> :
callback
دالة رد نداء والتي تأخذ وسيط خطأ اختياري.
دالة تستعمل للحصول على إشعارات عندما لا يعود المجرى قابلًا للقراءة، أو قابلًا للكتابة أو واجه خطأً ما، أو جرى إغلاقه بشكل سابق لأوانه.
const { finished } = require('stream');
const rs = fs.createReadStream('archive.tar');
finished(rs, (err) => {
if (err) {
console.error('Stream failed', err);
} else {
console.log('Stream is done reading');
}
});
rs.resume(); //تفريغ المجرى
هذا التابع مفيد خصوصًا في حالات معالجة الخطأ الذي يُدمّر فيه المجرى بشكل سابق لأوانه (مثل طلب HTTP مُجهض) ولم يُطلق الحدث 'end'
أو 'finish'
.
الواجهة finished
البرمجية هي promisify'able قابلة للتعامل مع الوعود (promise) كذلك.
const finished = util.promisify(stream.finished);
const rs = fs.createReadStream('archive.tar');
async function run() {
await finished(rs);
console.log('Stream is done reading');
}
run().catch(console.error);
rs.resume(); // تفريغ المجرى
stream.pipeline(...streams[, callback])
أُضيف في الإصدار: 10.0.0.
- <Stream> :
...streams
مجرييان أو أكثر للربط بينهما. - <Function> :
callback
دالة رد نداء تأخذ وسيط خطأ اختياري.
تابع نموذجي للنقل (الربط) بين المجاري موجهةً الأخطاء ومنظفةً بشكل ملائم و مُزوِدةً بدالة رد نداء عندما ينتهي خط النقل.
const { pipeline } = require('stream');
const fs = require('fs');
const zlib = require('zlib');
//البرمجية لربط سلسلة مجاري معًا بسهولة والحصول على إشعارات pipeline استخدم واجهة
//عندما يتم التوصيل والنقل بشكل كامل
// :ضخمًا بكفاءة tar يحتمل ملف gzip خط النقل إلى
pipeline(
fs.createReadStream('archive.tar'),
zlib.createGzip(),
fs.createWriteStream('archive.tar.gz'),
(err) => {
if (err) {
console.error('Pipeline failed', err);
} else {
console.log('Pipeline succeeded');
}
}
);
الواجهة البرمجية pipeline
قابلة للتعامل مع الوعود (promise) أيضًا:
const pipeline = util.promisify(stream.pipeline);
async function run() {
await pipeline(
fs.createReadStream('archive.tar'),
zlib.createGzip(),
fs.createWriteStream('archive.tar.gz')
);
console.log('Pipeline succeeded');
}
run().catch(console.error);
الواجهات البرمجية لمنفذي المجاري
صُممت الواجهات البرمجية للوحدة stream
لتجعل بالإمكان تنفيذ المجاري بسهولة باستعمال وحدة الوراثة النموذجية في JavaScript.
أولًا، سيعرّف مطوّر المجرى صنف جافا سكريبت جديد والذي سيكون توسعة لأحد أصناف المجاري الأساسية الأربعة
(stream.Writable
أو stream.Readable
أو stream.Duplex
أو stream.Transform
) ضامنًا أنها تستدعي باني صنف الأب المناسب:
const { Writable } = require('stream');
class MyWritable extends Writable {
constructor(options) {
super(options);
// ...
}
}
ثم يجب أن ينفّذ صنف المجرى الجديد واحدة أو اكثر من التوابع المحددة بالاعتماد على نوع المجرى الذي يُننشأ، كما هو مفصّل في المخطط أدناه:
حالة الاستخدام | الصنف | توابع للتطبيق | |
---|---|---|---|
قراءة فقط | Readable | _read
| |
كتابة فقط | Writable | _write , _writev , _final
| |
قراءة وكتابة | Duplex | _read , _write , _writev , _final
|
لا ينبغي أبدًا أن تستدعي شيفرة التنفيذ للمجرى التوابع "العامة" (public) للمجرى والتي ضُمّنت للاستخدام من قبل المستهلكين (كما هو موصوف في مقطع الواجهات البرمجية لمستهلكي المجرى). فعل ذلك قد يقود إلى تأثيرات جانبية معاكسة في شيفرة التطبيق المستهلك للمجرى.
البناء المبسّط (Simplified Construction)
أُضيف في الإصدار: 1.2.0.
من أجل العديد من الحالات البسيطة، من الممكن انشاء مجرىً دون الإعتماد على الوراثة، يمكن أن يُنجز هذا بانشاء مباشر لنسخ من الكائنات stream.Writable
أو stream.Readable
أو stream.Duplex
أو stream.Transform
وتمرير التوابع المناسبة كخيارات للباني.
const { Writable } = require('stream');
const myWritable = new Writable({
write(chunk, encoding, callback) {
// ...
}
});
تنفيذ مجرى قابل للكتابة
وُسِع الصنف stream.Writable
لينفذ المجرى Writable.
يجب أن تستدعي مجاري Writable المخصصة الباني new stream.Writable([options])
وتنفّذ التابع
writable._write()
. ويمُكن أن يُنفّذ التابع writable._writev()
أيضًا.
new stream.Writable([options])
الإصدار | التغييرات | |
---|---|---|
10.0.0 | أضيف الخيار emitClose ليحدد فيما إذا أطلق 'close' عند الهدم.
| |
- <Object>
options
- <number> :
highWaterMark
مستوى الذاكرة المؤقتة عندما يبدأ stream.write() بإعادةfalse
. القيمة الإفتراضية:16384
(16kb) أو16
من أجل المجاريobjectMode
. - <boolean> :
decodeStrings
قيمة منطقية تحدد إذا كان يراد ترميز السلاسل النصية ككائنات من النوعBuffer
أم لا قبل تمريرها إلى stream._write()، مستخدمًا الترميز المحدد في استدعاء stream.write(). القيمة الإفتراضية:true
. - <string> :
defaultEncoding
الترميز الافتراضي الذي يُستخدم عندما لا يحدد ترميز كوسيط في stream.write(). القيمة الإفتراضية:'utf8'
. - <boolean> :
objectMode
قيمة منطقية تحدد فيما إذا كانت stream.write(anyObj) عملية صالحة أم لا. عند ضبطها، يصبح من الممكن كتابة قيم من أي نوع من أنواع JavaScript باستثناء السلاسل النصية، أوBuffer
أوUint8Array
إذا دُعمت من قبل منفّذ المجرى. القيمة الإفتراضية:false
. - <boolean> :
emitClose
قيمة منطقية تحدد فيما إذا كان ينبغي أن يطلقَ المجرى الحدثَ'close'
بعد أن يُدمّر أو لا. القيمة الإفتراضية:true
. - <Function> :
write
تنفيذ للتابع stream._write(). - <Function> :
writev
تنفيذ للتابع stream._writev() . - <Function> :
destroy
تنفيذ للتابع stream._destroy(). - <Function> :
final
تنفيذ للتابع stream._final().
- <number> :
const { Writable } = require('stream');
class MyWritable extends Writable {
constructor(options) {
//stream.Writable() استدعِ باني
super(options);
// ...
}
}
أو عند استخدام البواني وفق النمط الذي يسبق ES6:
const { Writable } = require('stream');
const util = require('util');
function MyWritable(options) {
if (!(this instanceof MyWritable))
return new MyWritable(options);
Writable.call(this, options);
}
util.inherits(MyWritable, Writable);
أو، باستخدام نهج الباني المبسّط :
const { Writable } = require('stream');
const myWritable = new Writable({
write(chunk, encoding, callback) {
// ...
},
writev(chunks, callback) {
// ...
}
});
writable._write(chunk, encoding, callback)
- <Buffer> | <string> | <any> :
chunk
قطعة البيانات المراد كتابتها. سوف تكون دائمًا من النوع buffer إلا إذا كان الخيارdecodeStrings
مضبوطًا إلى القيمةfalse
أو أن المجرى يعمل في نمط الكائن. - <string> :
encoding
إذا كانت القطعة هي سلسلة نصية فإنencoding
سيكون هو ترميز المحارف لهذه السلسلة. إذا كانت القطعة chunck من النوعBuffer
، أو أن المجرى يعمل في نمط الكائن، سيُتجاهل الوسيطencoding
. - <Function> :
callback
يستدعي هذه الدالة (مع وسيط خطأ اختياري) عندما تنتهي معالجة القطع المعطاة.
يجب أن توفر كل مجاري Writable
التابع writable._write() لإرسال البيانات إلى المصادر الأساسية.
تقدّم مجاري التحويل (Transform) تنفيذها الخاص للتابع writable._write().
يجب ألّا تُستدعَى هذه الدالة من قبل الشيفرة التطبيق مباشرةً. يجب أن تُنفّذ باستخدام صنف ابن، وتُستدعى من قبل توابع الصنف Writable
الداخلية فقط.
يجب أن تستدعَى الدالة callback
لتشير إمّا إلى انتهاء الكتابة بنجاح أو فشلها مع خطأ. يجب أن يكون الوسيط الأول المُمرر إلى callback
هو كائن Error
إذا فشل الإستدعاء أو null
إذا نجحت الكتابة.
كل الاستدعاءات للتابع writable.write()
التي تحصل بين وقت استدعاء writable._write()
واستدعاء الدالة callback
سوف تؤدي إلى تخزين البيانات المكتوبة مؤقتًا. عندما تستدعى الدالة callback، قد يطلق المجرى الحدث 'drain'. إذا كان تطبيق المجرى قادرًا على معالجة قطع متعددة من البيانات مرّة واحدة، فينبغي أن يُنفَّذ التابع writable._writev()
.
إذا كانت الخاصية decodeStrings
مضبوطة بشكل صريح إلى false
في خيارات الباني. عند ذلك ستبقى chunk
نفس الكائن الذي يمُرر إلى التابع .write()
وقد تكون سلسلة نصية بدلًا من Buffer
. هذا من أجل دعم التطبيقات التي تمتلك معالجة مثالية لبعض ترميزات البيانات النصية. في هذه الحالة، سوف يحدد الوسيط encoding
ترميز محارف السلسلة النصية. وإلّا، سوف يُهمل الوسيط encoding
بشكل آمن.
سيكون التابع writable._write()
مسبوقًا بالرمز _ لأنه داخلي للصنف الذي عرّفه، وينبغي ألّا يُستدعى مطلقًا بشكل مباشر من قبل برامج المستخدم.
writable._writev(chunks, callback)
- <Object[]> :
chunks
القطع التي ستُكتب. كل قطعة لها الشكل التالي:{ chunk: ..., encoding: ... }
. - <Function> :
callback
دالة رد نداء (مع وسيط خطأ بشكل اختياري) لتُستدعى عند انتهاء المعالجة للقطع المُزوّدة.
يجب ألّا تُستدعى هذه الدالة من قبل شيفرة التطبيق مباشرةً، يجب أن تُنفّذ من قبل صنف ابن، وتُستدعى من قبل توابع الصنف Writable
الداخلية فقط.
يمكن أن يُنفّذ التابع writable._writev()
بالإضافة إلى writable._write()
في تطبيقات المجاري والتي تمتلك القدرة على معالجة قطع متعددة من البيانات في المرة الواحدة. إذا نُفّذت، سوف يُستدعى التابع مع كل قطع البيانات المخزّنة حاليًا في طابور الكتابة.
يُسبق التابع writable._writev()
بالرمز _ لأنّه داخلي للصنف الذي عرّفه، وينبغي ألّا يُستدعى مطلقًا من قبل من قبل برامج المستخدم.
writable._destroy(err, callback)
أضيف في الإصدار: 8.0.0.
- <Error> :
err
خطأ محتمل. - <Function> :
callback
دالة رد نداء والتي تأخذ وسيط خطأ اختياري.
يُستدعى التابع _destroy()
من قبل التابع writable.destroy(). يمكن أن يُعاد تعريفه من قبل صنف ابن ولكن يجب ألّا يُستدعى بشكل مباشر.
writable._final(callback)
أضيف في الإصدار: 8.0.0.
<Function> :callback
يستدعي هذه الدالة (مع وسيط خطأ اختياري) عند انتهاء كتابة أي بيانات متبقية.
يجب ألّا يُستدعى التابع _final()
بشكل مباشر. يمكن تنفيذه من قبل صنف ابن؛ وإذا كان ذلك، سوف يُستدعى مع توابع الصنف Writable
الداخلية فقط.
سوف تُستدعى هذه الدالة الاختيارية قبل أن يغلق المجرى، مؤخرةً الحدث 'finish'
حتى تستدعى الدالة callback
. هذا مفيد لإغلاق المصادر أو كتابة البيانات المخزّنة مؤقتًا قبل أن ينتهي المجرى.
أخطاء أثناء الكتابة
يُنصح بأن يُكتب تقرير بالأخطاء الحاصلة أثناء معالجة التابعين writable._write()
و writable._writev()
عن طريق استدعاء دالة رد النداء وتمرير الخطأ كأول وسيط. هذا سوف يسبب انطلاق الحدث 'error'
من قِبَل المجرى Writable
. رمي خطأ Error
أثناء writable._write()
يمكن أن ينتج سلوك غير متوقّع ومتضارب بالاعتماد على كيفية استعمال المجرى. يضمن استخدام دالة رد النداء معالجة أخطاء متناسقة ومتوقعة.
إذا كان المجرى Readable
موصولًا بأنبوب مع المجرى Writable
وأطلق Writable
خطأً في هذه الأثناء، فسوف ينفصل المجرى Readable
.
const { Writable } = require('stream');
const myWritable = new Writable({
write(chunk, encoding, callback) {
if (chunk.toString().indexOf('a') >= 0) {
callback(new Error('chunk is invalid'));
} else {
callback();
}
}
});
مثالٌ عن مجرى قابل للكتابة (Writable)
يشرح التالي تنفيذ مجرى مخصص من النوع Writable
وبسيط إلى حد ما (عديم الجدوى إلى حدٍ ما). بينما تكون نسخة المجرى Writable
المحدد ليست ذات فائدة محددة حقيقية. يشرح المثال كل العناصر المطلوبة لنسخة مجرى Writable مخصصة:
const { Writable } = require('stream');
class MyWritable extends Writable {
constructor(options) {
super(options);
// ...
}
_write(chunk, encoding, callback) {
if (chunk.toString().indexOf('a') >= 0) {
callback(new Error('chunk is invalid'));
} else {
callback();
}
}
}
فك ترميز الذواكر المؤقتة في المجرى القابل للكتابة (Writable)
فك ترميز الذواكر المؤقتة هو مهمة شائعة، على سبيل المثال، عند استخدام محوّلات دخلها هو سلسلة نصية. فهي ليست عملية بديهية عند استخدام ترميز محارف متعدد البايتات، مثل UTF-8. يعرض المثال التالي كيفية فك ترميز سلاسل متعددة البايتات باستخدام StringDecoder
و Writable.
const { Writable } = require('stream');
const { StringDecoder } = require('string_decoder');
class StringWritable extends Writable {
constructor(options) {
super(options);
this._decoder = new StringDecoder(options && options.defaultEncoding);
this.data = '';
}
_write(chunk, encoding, callback) {
if (encoding === 'buffer') {
chunk = this._decoder.write(chunk);
}
this.data += chunk;
callback();
}
_final(callback) {
this.data += this._decoder.end();
callback();
}
}
const euro = [[0xE2, 0x82], [0xAC]].map(Buffer.from);
const w = new StringWritable();
w.write('currency: ');
w.write(euro[0]);
w.end(euro[1]);
console.log(w.data); //€ :عملة
تنفيذ مجرى قابل للقراءة (Readable)
يُوسّع الصنف stream.Readable
لينفّذ مجرى قابل للقراءة (Readable).
يجب أن تستدعي مجاري Readable
المخصصة الباني stream.Readable([options])
وتنفّذ التابع readable._read()
.
new stream.Readable([options])
- <Object> :
options
- <number> :
highWaterMark
العدد الأعظمي من البايتات لتُخزّن في الذاكرة المؤقتة الداخلية قبل ابطال القراءة من المصادر الأساسية. القيمة الإفتراضية:16384
(16kb) أو16
لأجل مجاريobjectMode
. - <string> :encoding إذا كان محدّدًا،عندئذ سوف يُفك ترميز الذاكرة إلى سلاسل نصية باستخدام الترميز المحدد. القيمة الإفتراضية:
null
. - <boolean> :
objectMode
قيمة منطقية تحدد فيما إذا كان ينبغي أن يتصرف هذا المجرى كمجرى من الكائنات. يعني أن يعيد stream.read(n) قيمة وحيدة بدلًا منBuffer
بالحجمn
. القيمة الإفتراضية:false
. - <Function> :
read
تنفيذ للتابع stream._read(). - <Function> :
destroy
تنفيذ للتابع stream._destroy().
- <number> :
const { Readable } = require('stream');
class MyReadable extends Readable {
constructor(options) {
//stream.Readable(options) يستدعي الباني
super(options);
// ...
}
}
أو عند استخدام البواني بنمط سابق للنمط ES6:
const { Readable } = require('stream');
const util = require('util');
function MyReadable(options) {
if (!(this instanceof MyReadable))
return new MyReadable(options);
Readable.call(this, options);
}
util.inherits(MyReadable, Readable);
أو عند استخدام نهج الباني المبسّط:
const { Readable } = require('stream');
const myReadable = new Readable({
read(size) {
// ...
}
});
readable._read(size)
سجل التغييرات
الإصدار | التغييرات | |
---|---|---|
10.0.0 | استدعاء _read() لمرّة واحدة في كل جزء صغير من النبضة (microtick).
| |
0.9.4 | أُضيف في الإصدار: 0.9.4. | |
- <number> :
size
عدد البايتات المراد قراءتها بشكل غير متزامن.
يجب ألّا تُستدعى هذه الدالة من قبل شيفرة التطبيق مباشرةً. يجب أن تنفّذ من قبل أصناف ابن. وتُستدعى من قبل توابع الصنف Readable
الداخلية فقط.
كل تطبيقات المجاري القابلة للقراءة (Readable
) يجب أن توفر تنفيذًا للتابع readable._read()
لجلب البيانات من المصادر الأساسية.
عندما يُستدعى readable._read()
، إذا كانت البيانات متوافرة من المصدر، ينبغي أن يبدأ التطبيق بدفع البيانات إلى داخل طابور القراءة باستخدام التابع this.push(dataChunk). ينبغي أن يكمل _read()
القراءة من المصادر ودفع البيانات حتى يعيد readable.push()
القيمة false
. فقط عندما يُستدعى _read()
مجدّدًا بعد أن توقف، ينبغي أن يستأنف دفع بيانات إضافية فوق الطابور.
حالما استُدعي التابع readable._read()
، لن يُستدعى مجدّدًا حتى يُستدعى التابع readable.push(). يُضمن أن يُستدعى readable._read()
لمرّة واحدة فقط خلال التنفيذ المتزامن؛ أي -6 ^10 من النبضة (microtick).
الوسيط size
هو إرشادي. من أجل تطبيقات تكون فيها القراءة هي عملية وحيدة تعيد بيانات، يمكن أن تستخدم الوسيط size
لتحديد كمية البيانات المراد جلبها. قد تتجاهل تطبيقات أخرى هذا الوسيط وتقدّم البيانات ببساطة عندما تصبح متوافرة. لا يوجد حاجة للإنتظار حتى تتوافر بايتات بالحجم size
قبل استدعاء stream.push(chunk).
يُسبق التابع readable._read()
بالرمز _ لأنّه داخلي للصنف الذي عرّفه، ولا ينبغي أبدًا أن يُستدعى بشكل مباشر من قبل برامج المستخدم.
readable._destroy(err, callback)
أُضيف في الإصدار: 8.0.0
- <Error> :
err
خطأ مُحتمل. - <Function> :
callback
دالة رد نداء والتي تأخذ وسيط خطأ اختياري.
يُستدعى تابع _destroy()
من قبل readable.destroy(). يمكن أن يُعاد تعريفه من قبل صنف ابن ولكن يجب ألّا يُستدعى بشكل مباشر.
readable.push(chunk[, encoding])
سجل التغييرات
الإصدار | التغييرات | |
---|---|---|
8.0.0 | يمكن الآن أن يكون الوسيط chunk نسخة من Uint8Array .
| |
- <Buffer> | <Uint8Array> | <string> | <null> | <any> :
chunk
قطعة من البيانات للدفع إلى طابور القراءة. من أجل المجاري التي لا تعمل في نمط الكائن، يجب أن تكونchunk
هي سلسلة نصية أوBuffer
أوUint8Array
. من أجل المجاري التي تعمل في نمط الكائن، يمكن أن تكونchunk
أي نوع من أنواع بيانات JavaScript. - <string> :
encoding
ترميز قطع السلاسل النصية. يجب أن يكون ترميزBuffer
صالحًا، مثل'utf8'
أو'ascii'
. - القيمة المُعادة: <boolean> تكون القيمة
true
إذا كان هناك قطع اضافية من البيانات قد يستمر دفعها، وإلّا فستكونfalse
.
عندما تكون chunk
هي من النوع Buffer
أو Uint8Array
أو string
، ستضاف البيانات التي تحتويها إلى الطابور الداخلي لمستخدمي المجاري لتصبح متوافرة للقراءة والاستهلاك من قبل مستخدمي المجرى. تمرير القيمة null
إلى الخيار chunk
يشير إلى نهاية المجرى (EOF)، أي لا يوجد بعدها المزيد من البيانات لتكتب.
عندما يعمل Readable
في نمط التوقف المؤقت (paused mode)، يمكن قراءة البيانات المضافة مع التابع readable.push()
باستدعاء التابع readable.read() عندما يُطلق الحدث 'readable'.
عندما يعمل Readable
في نمط التدفق، ستُستلم كل البيانات المضافة مع readable.push()
بمجرد إطلاق الحدث 'data'
.
يُصمم التابع readable.push()
ليكون مرنًا قدر الإمكان. على سبيل المثال، عند تغليف مصدر منخفض المستوى (lower-level source) والذي يقدّم بعض أشكال آليات التوقف أو الإستئناف، ورد نداء البيانات، يمكن أن يُغلَّف المصدر منخفض المستوى من قبل نسخة Readable
مخصصة:
//readStop() و readStart() المصدر هو كائن مع التابعين
//والذي يحصل على استدعاء عندما يمتلك بيانات `ondata` ومع العضو
//والذي يحصل على استدعاء عندما تنتهي البيانات `onend` ومع العضو
class SourceWrapper extends Readable {
constructor(options) {
super(options);
this._source = getLowlevelSourceObject();
//كل مرّة توجد بيانات ، ادفعها إلى داخل الذاكرة المؤقتة الداخلية
this._source.ondata = (chunk) => {
// عند ذلك أوقف القراءة من المصدر،false القيمة push()إذا أعاد التابع
if (!this.push(chunk))
this._source.readStop();
};
//`null`عندما ينتهي المصدر، ادفع قطعة اشارات النهاية
this._source.onend = () => {
this.push(null);
};
}
// عندما يريد المجرى سحب المزيد من البيانات إليه _read سوف يُستدعى
//يُتجاهل وسيط الحجم الموصى به في هذه الحالة
_read(size) {
this._source.readStart();
}
}
يُعَدّ التابع eadable.push()
للاستدعاء فقط من قبل مُنفّذات المجرى Readable
، وفقط من داخل التابع readable._read()
.
من أجل المجاري التي لا تعمل في نمط الكائن، إذا كان معامل chunk
الخاص بالتابع readable.push()
هو undefined
، سيُعامل كسلسلة نصية فارغة أو ذاكرة تخزين مؤقت فارغة. انظر readable.push('') للمزيد من المعلومات.
أخطاء أثناء القراءة
يوصى بأن تُطلَق الأخطاء الحاصلة خلال معالجة التابع readable._read()
باستخدام الحدث 'error'
بدلًا من أن تُرمى فقط. قد ينتج رمي خطأٍ من داخل readable._read()
إلى سلوك غير متوقع و متضارب بالإعتماد على إذا كان المجرى يعمل في نمط التدفق أو النمط المتوقف. يضمن استخدام الحدث 'error'
معالجة أخطاء متوقعة ومتناسقة.
const { Readable } = require('stream');
const myReadable = new Readable({
read(size) {
if (checkSomeErrorCondition()) {
process.nextTick(() => this.emit('error', err));
return;
}
// القيام ببعض الأعمال
}
});
مثالٌ عن مجرى عدٍّ (Counting Stream)
التالي هو مثال أساسي لمجرى قابل للقراءة (Readable
) والذي يطلق الأعداد من 1 إلى 1,000,000 بترتيب تصاعدي ومن ثم ينتهي.
const { Readable } = require('stream');
class Counter extends Readable {
constructor(opt) {
super(opt);
this._max = 1000000;
this._index = 1;
}
_read() {
const i = this._index++;
if (i > this._max)
this.push(null);
else {
const str = String(i);
const buf = Buffer.from(str, 'ascii');
this.push(buf);
}
}
}
تنفيذ مجرى مزدوج (Duplex)
المجرى المزدوج (Duplex) هو الذي ينفِّذ كلا الواجهتين Readable و Writable، مثل اتصال المقبس TCP.
لأنّ JavaScript لا تدعم الوراثة المتعددة، وُسِّع الصنف stream.Duplex
لينفّذ المجرى Duplex (كمقابل لتوسعة الصنفين stream.Readable
و stream.Writable
).
يرث stream.Duplex
من الصنف stream.Readable
بشكل نموذجي (prototypically) ومن الصنف stream.Writable
بشكل طفيلي (parasitically)، ولكن سوف يعمل instanceof
بشكل صحيح لأجل كلا الصنفين الأساسيين بسبب إعادة تعريف Symbol.hasInstance على stream.Writable
.
يجب أن تستدعي مجاري Duplex
المخصصة الباني new stream.Duplex([options])
وتنفّذ كلا التابعين readable._read()
و writable._write()
.
new stream.Duplex(options)
سجل التغييرات
الإصدار | التغييرات | |
---|---|---|
8.4.0 | دُعم الآن الخياران readableHighWaterMark و writableHighWaterMark
| |
- <Object> :
options
يمرر إلى كلا البانيينWritable
وReadable
. ويملك أيضًا الحقول الآتية:- <boolean> :
allowHalfOpen
قيمة منطقية إذا كانتfalse
،فسوف ينهي المجرى تلقائيًا الطرف القابل للكتابة عندما ينتهي الطرف القابل للقراءة. القيمة الافتراضية:true
. - <boolean> :
readableObjectMode
يضبط هذا الخيار الوضعobjectMode
للطرف القابل للقراءة في المجرى. ليس له تأثير إذا كانتobjectMode
هيtrue
. القيمة الافتراضية:false
. - <boolean> :
writableObjectMode
يضبط هذا الخيار الوضعobjectMode
للطرف القابل للكتابة في المجرى. ليس له تأثير إذا كانتobjectMode
هيtrue
. القيمة الافتراضية:false
. - <number> :
readableHighWaterMark
يضبط القيمةhighWaterMark
لأجل الطرف القابل للقراءة من المجرى. لا يملك تأثيرًا إذا كانت القيمةhighWaterMark
معطاة. - <number> :
writableHighWaterMark
يضبط القيمةhighWaterMark
لأجل الطرف القابل للكتابة من المجرى. لا يملك تأثيرًا إذا كانت القيمةhighWaterMark
معطاة.
- <boolean> :
const { Duplex } = require('stream');
class MyDuplex extends Duplex {
constructor(options) {
super(options);
// ...
}
}
أو عند استخدام البواني بنمط سابق للنمط ES6:
const { Duplex } = require('stream');
const util = require('util');
function MyDuplex(options) {
if (!(this instanceof MyDuplex))
return new MyDuplex(options);
Duplex.call(this, options);
}
util.inherits(MyDuplex, Duplex);
أو باستخدام نهج الباني المُبسّط:
const { Duplex } = require('stream');
const myDuplex = new Duplex({
read(size) {
// ...
},
write(chunk, encoding, callback) {
// ...
}
});
مثال على مجرى مزدوج (Duplex)
يشرح التالي مثال بسيط عن مجرى من النوع المزدوج (Duplex
) والذي يغلّف كائن مصدر افتراضي منخفض المستوى إلى أي بيانات يمكن أن تُكتب وأن تُقرأ، ولو باستخدام واجهات برمجية غير متوافقة مع مجاري Node.js. يشرح المثال البسيط التالي مجرى Duplex
والذي يخزّن مؤقتًا البيانات القادمة والمكتوبة عليه بواسطة الواجهة Writable ثم يعاد قراءتها مجدَّدًا (والتي تُقرأ تراجعيًا) بواسطة الواجهة Readable.
const { Duplex } = require('stream');
const kSource = Symbol('source');
class MyDuplex extends Duplex {
constructor(source, options) {
super(options);
this[kSource] = source;
}
_write(chunk, encoding, callback) {
// المصادر الأساسية تتعامل فقط مع السلاسل النصية
if (Buffer.isBuffer(chunk))
chunk = chunk.toString();
this[kSource].writeSomeData(chunk);
callback();
}
_read(size) {
this[kSource].fetchSomeData(size, (data, encoding) => {
this.push(Buffer.from(data, encoding));
});
}
}
النقطة الأكثر أهمية في مجاري Duplex
هي أنّ كل طرف من الطرفين Readable
و Writable
يعمل بشكل مستقل عن الآخر رغم تواجدهما داخل نسخة كائن واحدة.
تشغيل المجاري المزدوجة بنمط الكائن
من أجل المجاري المزدوجة (Duplex
)، يمكن أن يُضبط الخيار objectMode
على وجه الحصر إمّا على الطرف Readable
أو الطرف Writable
باستخدام الخيارات readableObjectMode
و writableObjectMode
على التوالي.
في المثال التالي، على سبيل المثال، يُنشأ مجرى جديد من النوع Transform
(والذي هو نوع من المجاري المزدوجة (Duplex)) والذي يمتلك طرف قابل للكتابة (Writable
) بنمط الكائن والذي يقبل أرقام JavaScript حُوّلت إلى سلاسل نصية ستة عشرية على الطرف Readable
.
const { Transform } = require('stream');
كل مجاري التحويل (Transform) هي أيضًا مجاري مزدوجة (Duplex)//
const myTransform = new Transform({
writableObjectMode: true,
transform(chunk, encoding, callback) {
//ضغط القطعة إلى رقم إذا كان ضروريًا
chunk |= 0;
// تحويل القطعة إلى شيء مختلف
const data = chunk.toString(16);
//ادفع البيانات إلى الطابور القابل للقراءة
callback(null, '0'.repeat(data.length % 2) + data);
}
});
myTransform.setEncoding('ascii');
myTransform.on('data', (chunk) => console.log(chunk));
myTransform.write(1);
// يطبع: 01
myTransform.write(10);
// 0a :يطبع
myTransform.write(100);
// يطبع: 64
تطبيق مجرى التحويل (Transform)
مجرى التحويل (Transform) هو مجرى مزدوج (Duplex) حيث أن الخرج محسوب بطريقة ما من الدخل. مجاري الوحدة zlib أو مجاري الوحدة crypto هي مثال على هذا النوع من المجاري والتي تضغط أو تشفّر أو تفك تشفير البيانات.
لايوجد ضرورة أن يكون الخرج بنفس حجم الدخل أو نفس عدد القطع أو أن يصل في نفس الوقت. على سبيل المثال، المجرى Hash
سوف يملك دائًما وأبدًا قطعة خرج واحدة والتي تُوفّر عندما يُنهَى الدخل. سوف ينتج المجرى zlib
خرجًا إمّا أصغر بكثير أو أكبر بكثير من الدخل.
وُسّع الصنف stream.Transform
لينفّذ مجرى التحويل (Transform).
يرث الصنف stream.Transform
بشكل نموذجي من stream.Duplex
ويطبّق إصداراته الخاصة من التابعين writable._write()
و readable._read()
.يجب أن تنفّذ تطبيقات مجاري Transform
المخصصة التابع transform._transform() وربما تنفّذ أيضًا التابع transform._flush().
يجب توخي الحذر عند استخدام مجاري Transform
، إذ أنّ البيانات المكتوبة إلى المجرى قد تؤدي إلى أن يصبح الطرف Writable
للمجرى متوقفًا إذا كان الخرج على الطرف Readable
غير مُستهلك.
new stream.Transform([options])
- <Object> :options مُمرّرة إلى كلا بانيي
Writable
وReadable
. وتحوي أيضًا الحقول التالية:- <Function> :
transform
تنفيذ لتابع stream._transform() - <Function> :
flush
تنفيذ لتابع stream._flush()
- <Function> :
const { Transform } = require('stream');
class MyTransform extends Transform {
constructor(options) {
super(options);
// ...
}
}
أو عند استخدام البواني بنمط سابق للنمط ES6:
const { Transform } = require('stream');
const util = require('util');
function MyTransform(options) {
if (!(this instanceof MyTransform))
return new MyTransform(options);
Transform.call(this, options);
}
util.inherits(MyTransform, Transform);
أو باستخدام نهج الباني المبسّط:
const { Transform } = require('stream');
const myTransform = new Transform({
transform(chunk, encoding, callback) {
// ...
}
});
الحدثان 'finish'
و 'end'
الحدثان 'finish' و 'end' هما من الصنفين stream.Writable
و stream.Readable
على التوالي. يُطلق الحدث 'finish'
بعد أن يُستدعى stream.end() وقد عُولجت كل القطع من قبل stream._transform() .يُطلق الحدث 'end'
بعد أن أُخرجت كل البيانات والذي يحصل بعدما استُدعي رد النداء في التابع transform._flush().
transform._flush(callback)
- <Function> :
callback
دالة رد نداء (مع وسيط خطأ وبيانات بشكل اختياري) لتُستدعى عندما تُدفع البيانات المتبقية.
يجب ألّا تُستدعى الدالة من قبل شيفرة التطبيق مباشرةً. يجب أن تُنفّذ من قبل أصناف ابن، وتُستدعى من قبل توابع الصنف Readable
الداخلية فقط.
في بعض الحالات، قد تحتاج عملية التحويل إلى إطلاق بت إضافي من البيانات في نهاية المجرى. على سبيل المثال، سوف يخزّن مجرى الضغط zlib
كمية من الحالة الداخلية المستخدمة لضغط الخرج بشكل مثالي. ولكن عندما ينتهي المجرى، تحتاج البيانات الإضافية إلى أن تُدفَع بحيث ستكتمل البيانات المضغوطة.
قد تنفّذ مجاري تحويل (Transform) مخصصة التابع transform._flush()
. سوف يُستدعى هذا عندما لا يكون هناك المزيد من البيانات المكتوب لتُستهلك (لتُقرأ)، ولكن قبل إطلاق الحدث 'end' مشيرًا إلى نهاية المجرى القابل للقراءة.
خلال تنفيذ transform._flush()
، قد لا يُستدعى التابع readable.push()
مطلقًا أو يستدعى أكثر من مرة حسب الحاجة. يجب أن تُستدعى الدالة callback
عند انتهاء عملية دفع البيانات خارج المجرى.
التابع transform._flush()
مسبوق بالرمز _ لأنّه داخلي للصنف الذي عرّفه، وينبغي ألّا يُستدعى مباشرةً من قبل برامج المستخدم.
transform._transform(chunk, encoding, callback)
- <Buffer> | <string> | <any>:
chunk
قطعة البيانات المراد تحويلها. ستكون دائمًا كائنًا من النوع Buffer إلّا إذا ضُبط الخيارdecodeStrings
إلى القيمةfalse
أو أن المجرى يعمل في نمط الكائن. - <string>
encoding
إذا كانت القطعة سلسلة نصية، عندها ستكون قيمة encoding هي نوع الترميز. إذا كانت القطعة buffer ومن ثم ستكون قيمة encoding هي قيمة خاصة 'buffer'، تتجاهلها في هذه الحالة. - <Function>:
callback
دالة رد نداء (مع وسيط خطأ وبيانات اختيارية) لتُستدعى بعد أن تتم معالجةchunk
المزوّدة.
يجب ألّا تُستدعى الدالة من قبل شيفرة التطبيق مباشرةً. يجب أن تُنفّذ من قبل الأصناف الأبناء وتُستدعى من قبل توابع الصنف Readable
الداخلية فقط.
كل تطبيقات المجاري Transform
يجب أن تقدّم التابع _transform()
ليقبل دخلًا وينتج خرجًا. تنفيذ transform._transform()
يعالج البايتات التي تُكتب، يحسب الخرج، ومن ثم يمرر ذاك الخرج خارجًا إلى الجزء القابل للقراءة باستخدام التابع readable.push()
.
قد لا يُستدعى التابع transform.push()
أبدًا أو يستدعى أكثر من مرة لتوليد خرج من قطعة دخل واحدة، معتمدًا على الكمية المراد إخراجها كنتيجة للقطعة.
من الممكن ألّا يولّد خرجٌ من أي قطعة مكتوبة من بيانات الدخل.
يجب أن تُستدعى الدالة callback
فقط عندما تُستهلك القطعة الحالية بالكامل. يجب أن يكون الوسيط الأول المُمرر إلى callback
هو كائن من النوع Error
إذا حصل خطأ أثناء معالجة الدخل أو null
بصورة أخرى. إذا مُرر وسيط ثاني إلى الدالة callback
، فسوف يُحال إلى التابع readable.push()
. بعبارة أخرى يعادل التالي:
transform.prototype._transform = function(data, encoding, callback) {
this.push(data);
callback();
};
transform.prototype._transform = function(data, encoding, callback) {
callback(null, data);
};
يُسبق التابع transform._transform()
بالرمز _ لأنّه داخلي بالنسبة للصنف الذي عرّفه، وينبغي ألّا يُستدعى مباشرةً من قبل برامج المستخدم.
لايُستدعى التابع transform._transform()
على التفرّع مطلقًا؛ تطبّق المجاري آلية الطوابير ويجب أن تستدعى الدالة callback
لاستقبال القطعة التالية إمّا بشكل متزامن أو غير متزامن.
الصنف: stream.PassThrough
الصنف stream.PassThrough
هو تنفيذ بسيط لمجرى التحويل (Transform) والذي يمرر بايتات الدخل ببساطة عبره إلى الخرج. الغرض منه في المقام الأول هو الأمثلة والفحص، ولكن هناك بعض حالات الاستخدام حيث يكون stream.PassThrough
مفيدًا ككتلة بناء لأنواع غير مألوفة من المجاري.
ملاحظات إضافية
التوافق مع إصدارات Node.js الأقدم
قبل الإصدار Node.js 0.10، كانت واجهة المجرى Readable
أبسط، ولكن أيضًا أقل قوةً وأقل فائدة.
- بدل الإنتظار حتى استدعاء التابع stream.read()، سوف يبدأ إطلاق الأحداث 'data' فورًا. التطبيقات التي ستحتاج إلى إنجاز بعض كمية العمل لتقرر كيفية معالجة البيانات تكون مطالبةً بتخزين البيانات المقروءة إلى ذاكرة تخزين مؤقتة داخلية لكي لا تفقد البيانات.
- كان التابع stream.pause() توجيهيًا، بدلًا من أن يكون مضمونًا. عنى هذا أنّه بقي من الضروري أن يكون مستعدًّا لاستقبال الأحداث 'data' حتى عندما يكون المجرى في حالة التوقف المؤقت.
في الإصدار Node.js 0.10، أُضيف الصنف Readable. من أجل التوافقية مع برامج Node.js أقدم، تنقلب المجاري Readable
إلى "نمط التدفق" عندما يُضاف معالج الحدث 'data' أو عندما يُستدعى التابع stream.resume(). التأثير هو أنّه حتى عند عدم استخدام التابع stream.read() و الحدث 'readable'، لم يعد من الضروري القلق حول فقد قطع 'data'.
بينما سوف تستمر معظم التطبيقات بالعمل بشكل اعتيادي إلا أن ذلك قد خلق حالةً حديةً في الحالات التالية:
- لن يُضاف مستمع للحدث 'data'.
- لن يُستدعى التابع أبدًا stream.resume().
- لن توصل المجاري مع أي وجهة قابلة للكتابة.
على سبيل المثال، ألق نظرة فاحصة على الشيفرة التالية:
// !تحذير! معطل
net.createServer((socket) => {
// ولكن لن نستهلك البيانات أبدًا'end' سوف نضيف مُنصتًا لحدث
socket.on('end', () => {
// لن تصل أبدًا إلى هنا
socket.end('The message was received but was not processed.\n');
});
}).listen(1337);
قبل الإصدار Node.js 0.10، ستُحذف رسالة البيانات القادمة ببساطة. ولكن، في الإصدار Node.js 0.10 وما بعده، يبقى المقبس متوقفًا إلى الأبد. الحل البديل لهذه المشكلة هو استدعاء التابع stream.resume() لبدء تدفق البيانات:
// الحل البديل
net.createServer((socket) => {
socket.on('end', () => {
socket.end('The message was received but was not processed.\n');
});
// ابدأ تدفق البيانات، ارميها
socket.resume();
}).listen(1337);
بالإضافة إلى قلب المجاري Readable
الجديدة إلى نمط التدفق، يمكن تغليف المجاري ذات النمط السابق ل 0.10 في الصنف Readable
باستخدام التابع readable.wrap() .
readable.read(0)
يوجد بعض الحالات حيث من الضروري إطلاق التحديث لآليات المجاري الأساسية القابلة للقراءة، دون استهلاك فعلي لأية بيانات. في هذه الحالات، من الممكن استدعاء readable.read(0)
،والتي ستعيد null
دائمًا.
إذا كانت ذاكرة التخزين المؤقتة الداخلية أقل من الحد highWaterMark
، ولا يقرأ المجرى الآن، ومن ثمّ سيثير استدعاء التابع stream.read(0)
استدعاء تابع stream._read() منخفض المستوى.
لما كانت معظم التطبيقات على الغالب ليست بحاجة لفعل ذلك أبدًا، يوجد حالات داخلNode.js حيث يُفعل ذلك، خصوصًا داخل مجاري الصنف Readable
.
readable.push('')
لا يُنصح باستخدام readable.push('')
.
دفع سلسلة نصية أو كائن من النوع Buffer
أو Uint8Array
بصفر من البايتات إلى مجرى لا يعمل في نمط الكائن له تأثير جانبي مثير للانتباه. ذلك لأنّ استدعاء التابع readable.push()، سوف ينهي الاستدعاء عملية القراءة. على أي حال، لأنّ الوسيط هو سلسلة فارغة، لن تضاف أي بيانات إلى الذاكرة المؤقتة القابلة للقراءة وبذلك لا يوجد شيء ليستهلكه المستخدم.
اختلاف highWaterMark بعد استدعاء readable.setEncoding()
سوف يغير استخدام التابع readable.setEncoding()
السلوك الذي تسلكه القيمة highWaterMark
مع مجرى يعمل بدون تفعيل نمط الكائن.
عادةً، يقاس حجم الذاكرة المؤقتة الحالية اعتمادًا على القيمة highWaterMark
بالبايت. ولكن بعد أن يُستدعى setEncoding()
، ستبدأ دالة المقارنة بقياس حجم الذاكرة المؤقتة بالمحارف.
هذه ليست اشكالية في الحالات الشائعة مع الترميز latin1
أو ascii
. ولكن يُنصح بأن تكون مدركًا لهذا السلوك عند العمل مع سلاسل نصية يمكن أن تحتوي محارف متعددة البايتات.