در حال خدمت کردن

یکی از جنبه های کلیدی برنامه های وب پیشرو این است که قابل اعتماد هستند. آنها می توانند دارایی ها را به سرعت بارگیری کنند، کاربران را درگیر نگه می دارند و بلافاصله بازخورد ارائه می دهند، حتی در شرایط بد شبکه. چگونه ممکن است؟ با تشکر از رویداد fetch کارگر خدمات.

پشتیبانی مرورگر

  • کروم: 40.
  • لبه: 17.
  • فایرفاکس: 44.
  • سافاری: 11.1.

منبع

رویداد fetch به ما امکان می‌دهد هر درخواست شبکه‌ای را که توسط PWA در محدوده سرویس‌دهنده ارائه می‌شود، هم برای درخواست‌های یکسان و هم برای درخواست‌های متقاطع، رهگیری کنیم. علاوه بر درخواست‌های ناوبری و دارایی، واکشی از یک سرویس‌کار نصب‌شده اجازه می‌دهد بازدید از صفحه پس از اولین بارگذاری سایت بدون تماس شبکه ارائه شود.

کنترل‌کننده fetch همه درخواست‌های یک برنامه، از جمله نشانی‌های اینترنتی و سرصفحه‌های HTTP را دریافت می‌کند و به توسعه‌دهنده برنامه اجازه می‌دهد تصمیم بگیرد که چگونه آنها را پردازش کند.

کارگر سرویس بین مشتری و شبکه می نشیند.

کارمند خدمات شما می تواند درخواستی را به شبکه ارسال کند، با پاسخی که قبلاً در حافظه پنهان شده است پاسخ دهد یا پاسخ جدیدی ایجاد کند. انتخاب با شماست. در اینجا یک مثال ساده آورده شده است:

self.addEventListener("fetch", event => {
    console.log(`URL requested: ${event.request.url}`);
});

پاسخ به یک درخواست

وقتی درخواستی به کارمند خدمات شما می رسد، دو کار وجود دارد که می توانید انجام دهید. می توانید آن را نادیده بگیرید، که به آن اجازه می دهد به شبکه برود، یا می توانید به آن پاسخ دهید. پاسخ به درخواست‌های داخل سرویس‌کار به این معناست که چگونه می‌توانید انتخاب کنید چه چیزی و چگونه به PWA خود بازگردانده شود، حتی زمانی که کاربر آفلاین است.

برای پاسخ به یک درخواست ورودی، event.respondWith() از داخل یک کنترل کننده رویداد fetch ، مانند این فراخوانی کنید:

// fetch event handler in your service worker file
self.addEventListener("fetch", event => {
    const response = .... // a response or a Promise of response
    event.respondWith(response);
});

شما باید respondWith() به طور همزمان فراخوانی کنید و باید یک شی Response را برگردانید. اما نمی‌توانید پس از اتمام واکشی واکشی کنترل‌کننده رویداد، مانند یک تماس همگام‌سازی respondWith() را فراخوانی کنید. اگر باید منتظر پاسخ کامل باشید، می توانید یک Promise را به respondWith() ارسال کنید که با یک Response حل می شود.

ایجاد پاسخ

به لطف Fetch API ، می‌توانید پاسخ‌های HTTP را در کد جاوا اسکریپت خود ایجاد کنید، و آن پاسخ‌ها را می‌توان با استفاده از Cache Storage API در حافظه پنهان ذخیره کرد و به گونه‌ای که گویی از یک وب سرور می‌آیند، بازگردانده می‌شوند.

برای ایجاد یک پاسخ، یک شی Response جدید ایجاد کنید و بدنه آن و گزینه هایی مانند وضعیت و هدرها را تنظیم کنید:

const simpleResponse = new Response("Body of the HTTP response");

const options = {
   status: 200,
   headers: {
    'Content-type': 'text/html'
   }
};
const htmlResponse = new Response("<b>HTML</b> content", options)

پاسخ از حافظه پنهان

اکنون که می‌دانید چگونه پاسخ‌های HTTP را از یک سرویس‌دهنده ارائه کنید، وقت آن است که از رابط ذخیره‌سازی حافظه پنهان برای ذخیره دارایی‌ها در دستگاه استفاده کنید.

می توانید از API ذخیره سازی کش استفاده کنید تا بررسی کنید که آیا درخواست دریافت شده از PWA در حافظه نهان موجود است یا خیر، و اگر وجود دارد، با آن به respondWith() پاسخ دهید. برای انجام این کار، ابتدا باید درون کش جستجو کنید. تابع match() که در رابط caches سطح بالا موجود است، همه فروشگاه‌ها را در مبدا یا روی یک شی کش باز جستجو می‌کند.

تابع match() یک درخواست HTTP یا یک URL را به عنوان آرگومان دریافت می کند و وعده ای را برمی گرداند که با Response مرتبط با کلید مربوطه حل می شود.

// Global search on all caches in the current origin
caches.match(urlOrRequest).then(response => {
   console.log(response ? response : "It's not in the cache");
});

// Cache-specific search
caches.open("pwa-assets").then(cache => {
  cache.match(urlOrRequest).then(response => {
    console.log(response ? response : "It's not in the cache");
  });
});

استراتژی های ذخیره سازی

ارائه فایل‌ها فقط از حافظه پنهان مرورگر برای هر موردی مناسب نیست. به عنوان مثال، کاربر یا مرورگر می تواند کش را خارج کند. به همین دلیل است که باید استراتژی های خود را برای ارائه دارایی برای PWA خود تعریف کنید. شما محدود به یک استراتژی کش نیستید. شما می توانید الگوهای مختلف را برای الگوهای URL مختلف تعریف کنید. برای مثال، می‌توانید یک استراتژی برای حداقل دارایی‌های UI، استراتژی دیگر برای تماس‌های API و استراتژی سوم برای URLهای تصویر و داده داشته باشید. برای انجام این کار، event.request.url در ServiceWorkerGlobalScope.onfetch بخوانید و آن را از طریق عبارات منظم یا یک الگوی URL تجزیه کنید. (در زمان نوشتن، الگوی URL در همه سیستم عامل ها پشتیبانی نمی شود).

رایج ترین استراتژی ها عبارتند از:

ابتدا کش
ابتدا یک پاسخ ذخیره شده در حافظه پنهان را جستجو می کند و اگر پاسخی پیدا نشد به شبکه باز می گردد.
شبکه اول
ابتدا از شبکه درخواست پاسخ می دهد و اگر هیچ پاسخی برگردانده نشد، پاسخ را در حافظه پنهان بررسی می کند.
قدیمی در حالی که اعتبار مجدد
پاسخی را از حافظه پنهان ارائه می‌کند، در حالی که در پس‌زمینه آخرین نسخه را درخواست می‌کند و آن را برای دفعه بعدی درخواست دارایی در حافظه پنهان ذخیره می‌کند.
فقط شبکه
همیشه با یک پاسخ از شبکه یا خطا پاسخ می دهد. کش هرگز مورد مشورت قرار نمی گیرد.
فقط حافظه پنهان
همیشه با یک پاسخ از حافظه پنهان یا خطاها پاسخ می دهد. شبکه هرگز مورد مشورت قرار نخواهد گرفت. دارایی هایی که با استفاده از این استراتژی ارائه می شوند باید قبل از درخواست به حافظه پنهان اضافه شوند.

ابتدا کش

با استفاده از این استراتژی، سرویس‌کار به دنبال درخواست تطبیق در حافظه پنهان می‌گردد و در صورت ذخیره‌سازی، پاسخ مربوطه را برمی‌گرداند. در غیر این صورت، پاسخ را از شبکه بازیابی می کند (به صورت اختیاری، به روز رسانی حافظه پنهان برای تماس های بعدی). اگر نه پاسخ کش وجود داشته باشد و نه پاسخ شبکه، درخواست با خطا مواجه می شود. از آنجایی که ارائه دارایی‌ها بدون رفتن به شبکه سریع‌تر است، این استراتژی عملکرد را بر تازگی ترجیح می‌دهد.

استراتژی Cache First

self.addEventListener("fetch", event => {
   event.respondWith(
     caches.match(event.request)
     .then(cachedResponse => {
       // It can update the cache to serve updated content on the next request
         return cachedResponse || fetch(event.request);
     }
   )
  )
});

اول شبکه

این استراتژی آینه استراتژی Cache First است. بررسی می‌کند که آیا درخواست می‌تواند از شبکه انجام شود یا خیر، و اگر نمی‌تواند، سعی می‌کند آن را از حافظه پنهان بازیابی کند. ابتدا کش را دوست داشته باشید. اگر نه پاسخ شبکه و نه پاسخ کش وجود داشته باشد، درخواست با خطا مواجه می شود. دریافت پاسخ از شبکه معمولا کندتر از دریافت آن از حافظه پنهان است، این استراتژی به جای عملکرد، محتوای به روز شده را در اولویت قرار می دهد.

استراتژی شبکه اول

self.addEventListener("fetch", event => {
   event.respondWith(
     fetch(event.request)
     .catch(error => {
       return caches.match(event.request) ;
     })
   );
});

بیات در حالی که اعتبار مجدد

استراتژی stale while revalidate یک پاسخ ذخیره شده را بلافاصله برمی گرداند، سپس شبکه را برای به روز رسانی بررسی می کند و در صورت یافتن پاسخ ذخیره شده جایگزین آن می شود. این استراتژی همیشه یک درخواست شبکه می دهد، زیرا حتی اگر یک منبع ذخیره شده در حافظه پنهان پیدا شود، سعی می کند آنچه را که در حافظه پنهان بود با آنچه از شبکه دریافت شده است، به روز کند تا در درخواست بعدی از نسخه به روز شده استفاده کند. بنابراین، این استراتژی راهی را برای شما فراهم می کند تا از سرویس سریع اولین استراتژی کش بهره مند شوید و کش را در پس زمینه به روز کنید.

استراتژی کهنه در حالی که اعتبار مجدد

self.addEventListener('fetch', event => {
  event.respondWith(
    caches.match(event.request).then(cachedResponse => {
        const networkFetch = fetch(event.request).then(response => {
          // update the cache with a clone of the network response
          const responseClone = response.clone()
          caches.open(url.searchParams.get('name')).then(cache => {
            cache.put(event.request, responseClone)
          })
          return response
        }).catch(function (reason) {
          console.error('ServiceWorker fetch failed: ', reason)
        })
        // prioritize cached response over network
        return cachedResponse || networkFetch
      }
    )
  )
})

فقط شبکه

استراتژی تنها شبکه شبیه به نحوه رفتار مرورگرها بدون سرویس کار یا API حافظه کش است. درخواست‌ها فقط در صورتی منبعی را برمی‌گردانند که بتوان آن را از شبکه واکشی کرد. این اغلب برای منابعی مانند درخواست‌های API فقط آنلاین مفید است.

استراتژی فقط شبکه

فقط کش

استراتژی تنها کش تضمین می کند که درخواست ها هرگز به شبکه نمی روند. تمام درخواست‌های دریافتی با یک آیتم حافظه پنهان از قبل پر شده پاسخ داده می‌شوند. کد زیر از کنترل‌کننده رویداد fetch با روش match حافظه پنهان برای پاسخگویی به حافظه پنهان استفاده می‌کند:

self.addEventListener("fetch", event => {
   event.respondWith(caches.match(event.request));
});

استراتژی فقط کش.

استراتژی های سفارشی

در حالی که موارد بالا استراتژی‌های رایج ذخیره‌سازی هستند، شما مسئول سرویس‌دهنده خود و نحوه رسیدگی به درخواست‌ها هستید. اگر هیچ یک از اینها برای نیازهای شما کار نمی کند، خود را ایجاد کنید.

برای مثال، می‌توانید از یک استراتژی شبکه اول با مهلت زمانی برای اولویت‌بندی محتوای به‌روزرسانی شده استفاده کنید، اما تنها در صورتی که پاسخ در آستانه‌ای که شما تعیین کرده‌اید ظاهر شود. همچنین می‌توانید یک پاسخ ذخیره‌شده را با یک پاسخ شبکه ادغام کنید و یک پاسخ پیچیده از سرویس‌کار بسازید.

به روز رسانی دارایی ها

به روز نگه داشتن دارایی های ذخیره شده PWA شما می تواند یک چالش باشد. در حالی که استراتژی قدیمی اعتبارسنجی مجدد یکی از راه‌های انجام این کار است، این تنها راه نیست. در فصل به‌روزرسانی ، تکنیک‌های مختلفی را برای به‌روزرسانی محتوا و دارایی‌های برنامه‌تان یاد خواهید گرفت.

منابع