فرار از جهنم Callback با تبدیل Callback به Promise

جهنم callback
جهنم callback


برای کار با توابع Asynchronous در جاوااسکریپت، از کالبک (Callback) استفاده میشه. با اینکه این راه در خیلی از موارد ممکنه مشکل رو حل کنه، زمانی که به انجام چند عمل Async باهم و در ادامه هم میرسیم، استفاده از Callback ما رو درگیر مشکلی به اسم جهنم Callback یا ٖ Callback Hell میکنه. برای حل این مشکل خوبه که بدونیم میشه Callback ها رو تبدیل به Promise کرد.

نوشتن تبدیل

برای اینکه بدونیم چطور میشه Callback رو به Promise تبدیل کرد، باید بدونیم Promise چه خصوصیاتی داره. نمیخوام وارد جزییات تعریفش بشم. اگر نمیدونین Promise چیه میتونین به تعریف پرامیس مراجعه کنین.

برای سادگی نشون دادن این تبدیل بیایم یه عمل Async ساده رو درنظر بگیریم و باهم از حالت کالبک، به پرامیس تبدیلش کنیم.

برای مثال در استفاده از فایل سیستم در Nodejs، تابعی وجود داره به اسم fs.readFile که خیلی ساده میتونین با استفاده ازش، محتویات یه فایل در فایل سیستم رو بخونید. متناظر با این تابع، fs.writeFile هم وجود داره که میتونید با استفاده ازش یه محتوایی رو توی یه فایل ذخیره کنید.

فرض کنید دو فایل داریم توی فولدر files پروژه با نام های:

  • read.txt
  • write.txt

و میخوایم محتویات فایل read.txt رو بخونیم و توی فایل write.txt ذخیره کنیم. به صورت پیش فرض در Nodejs اکثر عملیات ها Async هستن (این برمیگرده به ذات None-blocking-IO) و پارامتری رو به عنوان کالبک دریافت میکنن. کد زیر رو ببینید:

// Basic variables
const filesBaseDir = __dirname + '/files'
const filesDir = {
  read: filesBaseDir + '/read.txt',
  write: filesBaseDir + '/write.txt'
}
const fs = require('fs')

// Utils
const readFile = (fileDir, success, fail) => {
  fs.readFile(fileDir, (err, data) => {
    if (err) return fail(err)
 
    return success(data)
  })
}
const writeFile = (fileDir, content, success, fail) => {
  fs.writeFile(fileDir, content, (err) => {
    if (err) return fail(err)

    return success()
  })
}

حالا با درنظر داشتن کد بالا عملیات مدنظر رو انجام میدیم:

readFile(filesDir.read, (data) => {
  // Now write data to write.txt
  writeFile(filesDir.write, data, () => {
    console.log('Mission accomplished!')
  }, (err) => {
    console.error('Error writing to file...)
  })
}, (err) => {
  console.error('Error reading file...')
})q

خب به خوبی و خوشی تونستیم دو تابع Async رو باهم ادغام کنیم و زنجیره وار ازشون استفاده کنیم. تا همینجاش وقتی حتی فقط 2 تابع Async وجود داشت، دردسر زیادی برای ادغامشون کشیدیم، حالا اگر قرار بود ۴تا یا حتی ۱۰تا عمل Async رو پشت هم و در ادامه هم انجام میدادیم چطور؟ ۱۰ عمل Async در ادامه هم یعنی فراخوانی 9 کالبک در کالبک قبلی!! اسم دیگه ای به جز جهنم نمیشه براش گذاشت!

برای رهایی ازین شرایط کافیه بدونیم میتونیم توابع Async ای که کالبک میپذیرن رو به Promise تبدیل کنیم. ممکنه در وهله اول این تبدیل ارزش چندانی براتون نداشته باشه، اما اجازه بدین نشونتون بدم چطور زندگی براتون ساده تر میشه وقتی از جهنم میاین بیرون :))

نوشتن دوباره با استفاده از Promise ها

همون فانکشن های قبلی رو اینبار با استفاده از پرامیس مینویسیم:

// Convert Callbacks to Promisified version
const promisifiedReadFile = (fileDir) => {
  return new Promise((resolve, reject) => {
    readFile(fileDir, (data) => resolve(data), (e) => reject(e))
  })
}
const promisifiedWriteFile = (fileDir, data) => {
  return new Promise((resolve, reject) => {
    writeFile(fileDir, data, () => resolve(), (e) => reject(e))
  })
}

حالا از ورژن Promisified استفاده میکنیم:

promisifiedReadFile(filesDir.read)
  .then((data) => promisifiedWriteFile(filesDir.write, data))
  .then(() => {
    console.log('Mission accomplished!')
  })
  .catch(e => console.error('Error'))

خاصیت پرامیس ها اینه که به تعداد نامحدود هم اگر باشن میشه به راحتی در قالب .then های پشت هم ازشون استفاده کرد. پرامیس مثل بلاک try-catch عمل میکنه، به این معنی که نیازی نیس برای هر .then یه .catch قرار بدیم درحالیکه کافیه Exception رو در هر جایی از این زنجیره return کنیم و فقط یه .catch در انتهای زنجیره قرار بدیم تا بتونه این Exception رو بگیره.

اگر هم براتون واقعا مهمه :پوکرفیس که بدونین کجای پروسه به مشکل خوردین، منظورم نوع اروریه که توی پروسه اتفاق میوفته، کافیه نوع ارور خاص خودتون رو بسازید تا با استفاده از اون بتونید نوع مشکل پیش اومده رو پیش بینی کنید:

class ReadError extends Error {}
class WriteError extends Error {}

و در نهایت توی توابع مربوطه ارور مناسبش رو return کنین تا توسط catch انتهای زنجیره گرفته بشه.

const readFile = (fileDir, success, fail) => {
  fs.readFile(fileDir, (err, data) => {
    if (err) return new ReadError(err)
 
    return success(data)
  })
}
const writeFile = (fileDir, content, success, fail) => {
  fs.writeFile(fileDir, content, (err) => {
    if (err) return new WriteError(err)

    return success()
  })
}

در نهایت اینکه (Rule of Thumb)

برای تبدیل هر کالبک به پرامیس کافیه یه تابع ساخته بشه که یه پروامیس برگردونه. هم چنین این پرامیس دو تابع resolve و reject به عنوتن پارامتر دریافت کنه. توی بدنه این تابع، اون تابع کالبک دار اولیه رو فراخونی کنین و موقعی که ارور ندارید، resolve رو صدا کنین و موقعی که به ارور میخوره reject کنین. همین!

مطمئن باشید زندگی با Promise خیلی ساده تر از کالبکه به خصوص زمانی که بخواین زنجیری از کالبک ها رو صدا کنید!

هم چنین اگر از Nodejs استفاده میکنین پیشنهاد میکنم از پکیج es6-promisifyاستفاده کنید و توابع Callback دار نود جی اس رو به پرامیس تبدیل کنید.

خوشحال میشم ایده هاتون رو در مورد این مطلب توی کامنت ها ببینم تا فیبدک هامون رو به هم منتقل کنیم.

اگر از خوندن این مطلب لذت بردین میتونین اونو با دیگران به اشتراک بزارید. همینطور میتونین از سایر مطالب بلاگ هم استفاده کنین و یا اگر دوس داشته باشید، کنارمون بنویسید. برای راهنمای انتشار مطلب در پول ریکوئست میتونین با ما در تماس باشید.