फ़ाइल सिस्टम स्टैंडर्ड, ओरिजिन प्राइवेट फ़ाइल सिस्टम (ओपीएफ़एस) को स्टोरेज एंडपॉइंट के तौर पर पेश करता है. यह पेज के ओरिजिन के लिए निजी होता है और उस उपयोगकर्ता को नहीं दिखता जो परफ़ॉर्मेंस के लिए बेहतर तरीके से ऑप्टिमाइज़ की गई खास तरह की फ़ाइल को ऐक्सेस करने का विकल्प देता है.
ब्राउज़र समर्थन
ऑरिजिन प्राइवेट फ़ाइल सिस्टम, आधुनिक ब्राउज़र के साथ काम करता है. साथ ही, इसे Web Hypertext Application Technology Working Group (WHATWG) ने File System Living Standard में स्टैंडर्ड के तौर पर शामिल किया है.
वजह
अपने कंप्यूटर पर मौजूद फ़ाइलों के बारे में सोचते समय, आपके दिमाग़ में शायद फ़ाइल के क्रम के बारे में आता होगा: फ़ोल्डर में व्यवस्थित की गई फ़ाइलें, जिन्हें ऑपरेटिंग सिस्टम के फ़ाइल एक्सप्लोरर की मदद से एक्सप्लोर किया जा सकता है. उदाहरण के लिए, Windows पर Tom नाम के उपयोगकर्ता के लिए, उसकी 'क्या-क्या करना है' सूची C:\Users\Tom\Documents\ToDo.txt
में सेव हो सकती है. इस उदाहरण में, ToDo.txt
फ़ाइल का नाम है. वहीं, Users
, Tom
, और Documents
फ़ोल्डर के नाम हैं. Windows पर `C:` ड्राइव की रूट डायरेक्ट्री को दिखाता है.
वेब पर फ़ाइलों के साथ काम करने का पारंपरिक तरीका
वेब ऐप्लिकेशन में 'क्या-क्या करना है' सूची में बदलाव करने का सामान्य तरीका यह है:
- उपयोगकर्ता, फ़ाइल को सर्वर पर अपलोड करता है या
<input type="file">
की मदद से क्लाइंट पर खोलता है. - उपयोगकर्ता बदलाव करता है. इसके बाद, वह
<a download="ToDo.txt>
वाली फ़ाइल को डाउनलोड करता है. इस<a download="ToDo.txt>
को JavaScript के ज़रिए प्रोग्राम के हिसाब सेclick()
किया जाता है. - फ़ोल्डर खोलने के लिए,
<input type="file" webkitdirectory>
में एक खास एट्रिब्यूट का इस्तेमाल किया जाता है. इसका नाम भले ही मालिकाना हक वाला हो, लेकिन यह लगभग सभी ब्राउज़र पर काम करता है.
वेब पर फ़ाइलों के साथ काम करने का आधुनिक तरीका
इस फ़्लो से यह पता नहीं चलता कि उपयोगकर्ता, फ़ाइलों में बदलाव करने के बारे में क्या सोचते हैं. इसका मतलब है कि उपयोगकर्ताओं को उनकी इनपुट फ़ाइलों की डाउनलोड की गई कॉपी मिलती हैं. इसलिए, File System Access API ने तीन पिकर मेथड—showOpenFilePicker()
, showSaveFilePicker()
, और showDirectoryPicker()
—पेश किए हैं. ये ठीक वही काम करते हैं जो इनके नाम से पता चलता है. ये कुकी, इस तरह से फ़्लो को चालू करती हैं:
showOpenFilePicker()
की मदद सेToDo.txt
खोलें औरFileSystemFileHandle
ऑब्जेक्ट पाएं.FileSystemFileHandle
ऑब्जेक्ट से, फ़ाइल हैंडल केgetFile()
तरीके को कॉल करकेFile
पाएं.- फ़ाइल में बदलाव करें. इसके बाद, हैंडल पर
requestPermission({mode: 'readwrite'})
को कॉल करें. - अगर उपयोगकर्ता अनुमति के अनुरोध को स्वीकार करता है, तो बदलावों को वापस मूल फ़ाइल में सेव करें.
- इसके अलावा, कॉल
showSaveFilePicker()
करें और उपयोगकर्ता को नई फ़ाइल चुनने दें. (अगर उपयोगकर्ता पहले से खुली हुई किसी फ़ाइल को चुनता है, तो उसका कॉन्टेंट बदल जाएगा.) बार-बार सेव करने के लिए, फ़ाइल हैंडल को चालू रखा जा सकता है. इससे आपको फ़ाइल सेव करने का डायलॉग बॉक्स फिर से नहीं दिखाना पड़ेगा.
वेब पर फ़ाइलों के साथ काम करने से जुड़ी पाबंदियां
इन तरीकों से ऐक्सेस की जा सकने वाली फ़ाइलें और फ़ोल्डर, उपयोगकर्ता को दिखने वाले फ़ाइल सिस्टम में मौजूद होते हैं. वेब से सेव की गई फ़ाइलों और खास तौर पर एक्ज़ीक्यूटेबल फ़ाइलों को मार्क ऑफ़ द वेब के तौर पर मार्क किया जाता है. इससे ऑपरेटिंग सिस्टम, संभावित रूप से खतरनाक फ़ाइल के एक्ज़ीक्यूट होने से पहले एक और चेतावनी दिखा सकता है. सुरक्षा से जुड़ी एक अतिरिक्त सुविधा के तौर पर, वेब से मिली फ़ाइलों को भी सुरक्षित ब्राउज़िंग की मदद से सुरक्षित रखा जाता है. इस लेख के संदर्भ में, इसे आसान भाषा में क्लाउड पर आधारित वायरस स्कैनिंग की सुविधा कहा जा सकता है. File System Access API का इस्तेमाल करके किसी फ़ाइल में डेटा लिखने पर, डेटा को उसी जगह पर नहीं लिखा जाता. इसके बजाय, एक अस्थायी फ़ाइल का इस्तेमाल किया जाता है. जब तक फ़ाइल इन सभी सुरक्षा जांचों को पास नहीं कर लेती, तब तक उसमें कोई बदलाव नहीं किया जाता. जैसा कि आपको पता है, इस काम की वजह से फ़ाइल से जुड़ी कार्रवाइयां थोड़ी धीमी हो जाती हैं. हालांकि, जहां भी मुमकिन होता है वहां सुधार किए जाते हैं. उदाहरण के लिए, macOS पर. हालांकि, हर write()
कॉल में खुद का डेटा होता है. इसलिए, यह फ़ाइल को खोलता है, दिए गए ऑफ़सेट को ढूंढता है, और आखिर में डेटा लिखता है.
प्रोसेसिंग के लिए फ़ाइलें
साथ ही, फ़ाइलें डेटा रिकॉर्ड करने का एक बेहतरीन तरीका हैं. उदाहरण के लिए, SQLite पूरे डेटाबेस को एक ही फ़ाइल में सेव करता है. इमेज प्रोसेसिंग में इस्तेमाल होने वाले मिपमैप भी इसका एक उदाहरण हैं. मिपमैप, इमेज के पहले से कैलकुलेट किए गए और ऑप्टिमाइज़ किए गए सीक्वेंस होते हैं. इनमें से हर इमेज, पिछली इमेज के मुकाबले कम रिज़ॉल्यूशन वाली होती है. इससे ज़ूम करने जैसी कई कार्रवाइयां तेज़ी से होती हैं. तो फिर वेब ऐप्लिकेशन, फ़ाइलों के फ़ायदे कैसे पा सकते हैं? हालांकि, वेब पर फ़ाइल प्रोसेसिंग की परफ़ॉर्मेंस की कीमत चुकाए बिना. इसका जवाब ओरिजिन प्राइवेट फ़ाइल सिस्टम है.
उपयोगकर्ता को दिखने वाला वर्सेस ऑरिजिन प्राइवेट फ़ाइल सिस्टम
ऑपरेटिंग सिस्टम के फ़ाइल एक्सप्लोरर का इस्तेमाल करके, उपयोगकर्ता को दिखने वाले फ़ाइल सिस्टम के उलट, ऑरिजिन प्राइवेट फ़ाइल सिस्टम को उपयोगकर्ताओं के लिए नहीं बनाया गया है. इस फ़ाइल सिस्टम में मौजूद फ़ाइलों और फ़ोल्डर को पढ़ा, लिखा, एक जगह से दूसरी जगह ले जाया, और उनका नाम बदला जा सकता है. ऑरिजिन प्राइवेट फ़ाइल सिस्टम में मौजूद फ़ाइलें और फ़ोल्डर, नाम के मुताबिक निजी होते हैं. साथ ही, ये किसी साइट के ऑरिजिन के लिए निजी होते हैं. DevTools Console में location.origin
टाइप करके, किसी पेज के ऑरिजिन का पता लगाएं. उदाहरण के लिए, पेज https://developer.chrome.com/articles/
का ऑरिजिन https://developer.chrome.com
है. इसका मतलब है कि /articles
, ऑरिजिन का हिस्सा नहीं है. ऑरिजिन के सिद्धांत के बारे में ज़्यादा जानने के लिए, "सेम-साइट" और "सेम-ऑरिजिन" को समझना लेख पढ़ें. एक ही ऑरिजिन शेयर करने वाले सभी पेज, एक ही ऑरिजिन के प्राइवेट फ़ाइल सिस्टम का डेटा देख सकते हैं. इसलिए, https://developer.chrome.com/docs/extensions/mv3/getstarted/extensions-101/
को पिछले उदाहरण की तरह ही जानकारी दिखेगी. हर ऑरिजिन का अपना इंडिपेंडेंट ऑरिजिन प्राइवेट फ़ाइल सिस्टम होता है. इसका मतलब है कि https://developer.chrome.com
का ऑरिजिन प्राइवेट फ़ाइल सिस्टम, https://web.dev
के ऑरिजिन प्राइवेट फ़ाइल सिस्टम से पूरी तरह अलग होता है. Windows पर, उपयोगकर्ता को दिखने वाले फ़ाइल सिस्टम की रूट डायरेक्ट्री C:\\
होती है.
ओरिजन प्राइवेट फ़ाइल सिस्टम के लिए, हर ओरिजन के हिसाब से एक रूट डायरेक्ट्री होती है. यह डायरेक्ट्री शुरू में खाली होती है. इसे एसिंक्रोनस तरीके से navigator.storage.getDirectory()
को कॉल करके ऐक्सेस किया जाता है.
उपयोगकर्ता को दिखने वाले फ़ाइल सिस्टम और ऑरिजिन प्राइवेट फ़ाइल सिस्टम की तुलना करने के लिए, यह डायग्राम देखें. डायग्राम से पता चलता है कि रूट डायरेक्ट्री के अलावा, बाकी सब कुछ एक जैसा है. इसमें फ़ाइलों और फ़ोल्डर का क्रम होता है, ताकि आपके डेटा और स्टोरेज की ज़रूरतों के हिसाब से उन्हें व्यवस्थित किया जा सके.
ऑरिजिन के प्राइवेट फ़ाइल सिस्टम की खास बातें
ब्राउज़र में डेटा सेव करने के अन्य तरीकों (उदाहरण के लिए, localStorage या IndexedDB) की तरह, ओरिजन प्राइवेट फ़ाइल सिस्टम पर भी ब्राउज़र के कोटे से जुड़ी पाबंदियां लागू होती हैं. जब कोई उपयोगकर्ता पूरा ब्राउज़िंग डेटा मिटाता है या साइट का पूरा डेटा मिटाता है, तो ऑरिजिन प्राइवेट फ़ाइल सिस्टम भी मिट जाएगा. navigator.storage.estimate()
को कॉल करें. इसके बाद, रिस्पॉन्स ऑब्जेक्ट में usage
एंट्री देखें. इससे पता चलेगा कि आपका ऐप्लिकेशन पहले से कितना स्टोरेज इस्तेमाल कर रहा है. इसे usageDetails
ऑब्जेक्ट में स्टोरेज के तरीके के हिसाब से बांटा गया है. इसमें आपको खास तौर पर fileSystem
एंट्री देखनी है. ओरिजिन प्राइवेट फ़ाइल सिस्टम, उपयोगकर्ता को नहीं दिखता. इसलिए, अनुमति के लिए कोई प्रॉम्प्ट नहीं दिखता और सुरक्षित ब्राउज़िंग की जांच नहीं होती.
रूट डायरेक्ट्री का ऐक्सेस पाना
रूट डायरेक्ट्री को ऐक्सेस करने के लिए, यह कमांड चलाएं. आपको एक खाली डायरेक्ट्री हैंडल मिलता है. खास तौर पर, एक FileSystemDirectoryHandle
.
const opfsRoot = await navigator.storage.getDirectory();
// A FileSystemDirectoryHandle whose type is "directory"
// and whose name is "".
console.log(opfsRoot);
मुख्य थ्रेड या वेब वर्कर
ओरिजिन प्राइवेट फ़ाइल सिस्टम का इस्तेमाल दो तरीकों से किया जा सकता है: मुख्य थ्रेड पर या वेब वर्कर में. वेब वर्कर, मुख्य थ्रेड को ब्लॉक नहीं कर सकते. इसका मतलब है कि इस संदर्भ में एपीआई सिंक्रोनस हो सकते हैं. आम तौर पर, मुख्य थ्रेड पर इसकी अनुमति नहीं होती है. सिंक्रोनस एपीआई, एसिंक्रोनस एपीआई की तुलना में ज़्यादा तेज़ी से काम कर सकते हैं. ऐसा इसलिए, क्योंकि इन्हें प्रॉमिस से डील नहीं करना पड़ता. साथ ही, फ़ाइल ऑपरेशन आम तौर पर C जैसी भाषाओं में सिंक्रोनस होते हैं. इन भाषाओं को WebAssembly में कंपाइल किया जा सकता है.
// This is synchronous C code.
FILE *f;
f = fopen("example.txt", "w+");
fputs("Some text\n", f);
fclose(f);
अगर आपको फ़ाइल से जुड़े काम तुरंत करने हैं या आपको WebAssembly का इस्तेमाल करना है, तो सीधे वेब वर्कर में ओरिजन प्राइवेट फ़ाइल सिस्टम का इस्तेमाल करना पर जाएं. इसके अलावा, यहां दी गई जानकारी पढ़ें.
मुख्य थ्रेड पर, ऑरिजिन प्राइवेट फ़ाइल सिस्टम का इस्तेमाल करना
नई फ़ाइलें और फ़ोल्डर बनाना
रूट फ़ोल्डर बनाने के बाद, getFileHandle()
और getDirectoryHandle()
तरीकों का इस्तेमाल करके फ़ाइलें और फ़ोल्डर बनाएं. {create: true}
पास करने पर, अगर फ़ाइल या फ़ोल्डर मौजूद नहीं है, तो उसे बना दिया जाएगा. नई डायरेक्ट्री को शुरुआती पॉइंट के तौर पर इस्तेमाल करके, इन फ़ंक्शन को कॉल करके फ़ाइलों की एक हैरारकी बनाएं.
const fileHandle = await opfsRoot
.getFileHandle('my first file', {create: true});
const directoryHandle = await opfsRoot
.getDirectoryHandle('my first folder', {create: true});
const nestedFileHandle = await directoryHandle
.getFileHandle('my first nested file', {create: true});
const nestedDirectoryHandle = await directoryHandle
.getDirectoryHandle('my first nested folder', {create: true});
मौजूदा फ़ाइलों और फ़ोल्डर को ऐक्सेस करना
अगर आपको फ़ाइल या फ़ोल्डर का नाम पता है, तो getFileHandle()
या getDirectoryHandle()
तरीकों का इस्तेमाल करके, पहले से बनाई गई फ़ाइलों और फ़ोल्डर को ऐक्सेस करें. इसके लिए, फ़ाइल या फ़ोल्डर का नाम पास करें.
const existingFileHandle = await opfsRoot.getFileHandle('my first file');
const existingDirectoryHandle = await opfsRoot
.getDirectoryHandle('my first folder');
पढ़ने के लिए, फ़ाइल हैंडल से जुड़ी फ़ाइल पाना
FileSystemFileHandle
, फ़ाइल सिस्टम पर मौजूद किसी फ़ाइल को दिखाता है. इससे जुड़े File
को पाने के लिए, getFile()
तरीके का इस्तेमाल करें. File
ऑब्जेक्ट, एक खास तरह का Blob
होता है. इसका इस्तेमाल किसी भी ऐसे कॉन्टेक्स्ट में किया जा सकता है जहां Blob
का इस्तेमाल किया जा सकता है. खास तौर पर, FileReader
, URL.createObjectURL()
, createImageBitmap()
, और XMLHttpRequest.send()
, Blobs
और Files
, दोनों को स्वीकार करते हैं. अगर आपको FileSystemFileHandle
से File
मिलता है, तो डेटा "फ़्री" हो जाता है. इससे आपको डेटा का ऐक्सेस मिल जाता है और इसे उपयोगकर्ता को दिखने वाले फ़ाइल सिस्टम के लिए उपलब्ध कराया जा सकता है.
const file = await fileHandle.getFile();
console.log(await file.text());
स्ट्रीम करके किसी फ़ाइल में लिखना
createWritable()
को कॉल करके, डेटा को किसी फ़ाइल में स्ट्रीम करें. इससे एक FileSystemWritableFileStream
बनता है. इसके बाद, आपको कॉन्टेंट को write()
करना होता है. आखिर में, आपको स्ट्रीम को close()
करना होगा.
const contents = 'Some text';
// Get a writable stream.
const writable = await fileHandle.createWritable();
// Write the contents of the file to the stream.
await writable.write(contents);
// Close the stream, which persists the contents.
await writable.close();
फ़ाइलें और फ़ोल्डर मिटाना
फ़ाइल या डायरेक्ट्री के हैंडल के खास remove()
तरीके को कॉल करके, फ़ाइलों और फ़ोल्डर को मिटाएं. सभी सबफ़ोल्डर के साथ किसी फ़ोल्डर को मिटाने के लिए, {recursive: true}
विकल्प पास करें.
await fileHandle.remove();
await directoryHandle.remove({recursive: true});
इसके अलावा, अगर आपको किसी डायरेक्ट्री में मौजूद उस फ़ाइल या फ़ोल्डर का नाम पता है जिसे मिटाना है, तो removeEntry()
तरीके का इस्तेमाल करें.
directoryHandle.removeEntry('my first nested file');
फ़ाइलों और फ़ोल्डर को एक जगह से दूसरी जगह ले जाना और उनके नाम बदलना
move()
तरीके का इस्तेमाल करके, फ़ाइलों और फ़ोल्डर के नाम बदलें और उन्हें दूसरी जगह ले जाएं. फ़ोल्डर का नाम बदलने और उसे एक जगह से दूसरी जगह ले जाने की प्रोसेस एक साथ या अलग-अलग की जा सकती है.
// Rename a file.
await fileHandle.move('my first renamed file');
// Move a file to another directory.
await fileHandle.move(nestedDirectoryHandle);
// Move a file to another directory and rename it.
await fileHandle
.move(nestedDirectoryHandle, 'my first renamed and now nested file');
किसी फ़ाइल या फ़ोल्डर के पाथ का पता लगाना
यह जानने के लिए कि कोई फ़ाइल या फ़ोल्डर, रेफ़रंस डायरेक्ट्री के हिसाब से कहां मौजूद है, resolve()
तरीके का इस्तेमाल करें. इसके लिए, FileSystemHandle
को आर्ग्युमेंट के तौर पर पास करें. ओरिजन प्राइवेट फ़ाइल सिस्टम में किसी फ़ाइल या फ़ोल्डर का पूरा पाथ पाने के लिए, navigator.storage.getDirectory()
से मिली रूट डायरेक्ट्री को रेफ़रंस डायरेक्ट्री के तौर पर इस्तेमाल करें.
const relativePath = await opfsRoot.resolve(nestedDirectoryHandle);
// `relativePath` is `['my first folder', 'my first nested folder']`.
यह देखना कि क्या दो फ़ाइल या फ़ोल्डर हैंडल, एक ही फ़ाइल या फ़ोल्डर की ओर इशारा करते हैं
कई बार आपके पास दो हैंडल होते हैं और आपको यह नहीं पता होता कि वे एक ही फ़ाइल या फ़ोल्डर की ओर ले जाते हैं या नहीं. यह देखने के लिए कि क्या वाकई यही समस्या है, isSameEntry()
तरीके का इस्तेमाल करें.
fileHandle.isSameEntry(nestedFileHandle);
// Returns `false`.
किसी फ़ोल्डर में मौजूद कॉन्टेंट की सूची बनाना
FileSystemDirectoryHandle
एक एसिंक्रोनस इटरेटर है. इसे for await…of
लूप के साथ दोहराया जाता है. एसिंक्रोनस इटरेटर के तौर पर, यह entries()
, values()
, और keys()
तरीकों के साथ भी काम करता है. इनमें से किसी एक को अपनी ज़रूरत के हिसाब से चुना जा सकता है:
for await (let [name, handle] of directoryHandle) {}
for await (let [name, handle] of directoryHandle.entries()) {}
for await (let handle of directoryHandle.values()) {}
for await (let name of directoryHandle.keys()) {}
किसी फ़ोल्डर और सभी सबफ़ोल्डर में मौजूद कॉन्टेंट की सूची बनाना
एसिंक्रोनस लूप और रिकर्सन के साथ जोड़े गए फ़ंक्शन को हैंडल करना आसान नहीं होता. नीचे दिया गया फ़ंक्शन, किसी फ़ोल्डर और उसके सभी सबफ़ोल्डर में मौजूद कॉन्टेंट की सूची बनाने के लिए शुरुआती पॉइंट के तौर पर काम कर सकता है. इसमें सभी फ़ाइलें और उनके साइज़ शामिल हैं. अगर आपको फ़ाइल के साइज़ की ज़रूरत नहीं है, तो फ़ंक्शन को आसान बनाया जा सकता है. इसके लिए, जहां directoryEntryPromises.push
लिखा है वहां handle.getFile()
प्रॉमिस को पुश न करें, बल्कि सीधे handle
को पुश करें.
const getDirectoryEntriesRecursive = async (
directoryHandle,
relativePath = '.',
) => {
const fileHandles = [];
const directoryHandles = [];
const entries = {};
// Get an iterator of the files and folders in the directory.
const directoryIterator = directoryHandle.values();
const directoryEntryPromises = [];
for await (const handle of directoryIterator) {
const nestedPath = `${relativePath}/${handle.name}`;
if (handle.kind === 'file') {
fileHandles.push({ handle, nestedPath });
directoryEntryPromises.push(
handle.getFile().then((file) => {
return {
name: handle.name,
kind: handle.kind,
size: file.size,
type: file.type,
lastModified: file.lastModified,
relativePath: nestedPath,
handle
};
}),
);
} else if (handle.kind === 'directory') {
directoryHandles.push({ handle, nestedPath });
directoryEntryPromises.push(
(async () => {
return {
name: handle.name,
kind: handle.kind,
relativePath: nestedPath,
entries:
await getDirectoryEntriesRecursive(handle, nestedPath),
handle,
};
})(),
);
}
}
const directoryEntries = await Promise.all(directoryEntryPromises);
directoryEntries.forEach((directoryEntry) => {
entries[directoryEntry.name] = directoryEntry;
});
return entries;
};
वेब वर्कर में, ऑरिजिन प्राइवेट फ़ाइल सिस्टम का इस्तेमाल करना
जैसा कि पहले बताया गया है, वेब वर्कर मुख्य थ्रेड को ब्लॉक नहीं कर सकते. इसलिए, इस संदर्भ में सिंक्रोनस तरीकों का इस्तेमाल किया जा सकता है.
सिंक्रोनस ऐक्सेस हैंडल पाना
फ़ाइल से जुड़े सबसे तेज़ ऑपरेशन के लिए एंट्री पॉइंट, FileSystemSyncAccessHandle
होता है. इसे createSyncAccessHandle()
को कॉल करके, सामान्य FileSystemFileHandle
से हासिल किया जाता है.
const fileHandle = await opfsRoot
.getFileHandle('my highspeed file.txt', {create: true});
const syncAccessHandle = await fileHandle.createSyncAccessHandle();
सिंक्रोनस इन-प्लेस फ़ाइल के तरीके
सिंक्रोनस ऐक्सेस हैंडल मिलने के बाद, आपको फ़ाइल के लिए इन-प्लेस फ़ास्ट मेथड का ऐक्सेस मिलता है. ये सभी मेथड सिंक्रोनस होते हैं.
getSize()
: इससे फ़ाइल का साइज़ बाइट में मिलता है.write()
: यह फ़ंक्शन, बफ़र के कॉन्टेंट को फ़ाइल में लिखता है. इसके लिए, ऑफ़सेट का इस्तेमाल किया जा सकता है. साथ ही, यह लिखे गए बाइट की संख्या दिखाता है. लिखे गए बाइट की संख्या की जांच करने से, कॉल करने वाले लोग गड़बड़ियों का पता लगा सकते हैं और उन्हें ठीक कर सकते हैं. साथ ही, वे आंशिक रूप से लिखे गए डेटा को भी ठीक कर सकते हैं.read()
: यह फ़ंक्शन, फ़ाइल के कॉन्टेंट को बफ़र में पढ़ता है. इसके लिए, दिए गए ऑफ़सेट का इस्तेमाल किया जा सकता है.truncate()
: इस विकल्प का इस्तेमाल करके, फ़ाइल का साइज़ बदला जा सकता है.flush()
: इससे यह पक्का किया जाता है कि फ़ाइल के कॉन्टेंट में,write()
के ज़रिए किए गए सभी बदलाव शामिल हों.close()
: ऐक्सेस हैंडल को बंद करता है.
यहां एक ऐसा उदाहरण दिया गया है जिसमें ऊपर बताए गए सभी तरीकों का इस्तेमाल किया गया है.
const opfsRoot = await navigator.storage.getDirectory();
const fileHandle = await opfsRoot.getFileHandle('fast', {create: true});
const accessHandle = await fileHandle.createSyncAccessHandle();
const textEncoder = new TextEncoder();
const textDecoder = new TextDecoder();
// Initialize this variable for the size of the file.
let size;
// The current size of the file, initially `0`.
size = accessHandle.getSize();
// Encode content to write to the file.
const content = textEncoder.encode('Some text');
// Write the content at the beginning of the file.
accessHandle.write(content, {at: size});
// Flush the changes.
accessHandle.flush();
// The current size of the file, now `9` (the length of "Some text").
size = accessHandle.getSize();
// Encode more content to write to the file.
const moreContent = textEncoder.encode('More content');
// Write the content at the end of the file.
accessHandle.write(moreContent, {at: size});
// Flush the changes.
accessHandle.flush();
// The current size of the file, now `21` (the length of
// "Some textMore content").
size = accessHandle.getSize();
// Prepare a data view of the length of the file.
const dataView = new DataView(new ArrayBuffer(size));
// Read the entire file into the data view.
accessHandle.read(dataView);
// Logs `"Some textMore content"`.
console.log(textDecoder.decode(dataView));
// Read starting at offset 9 into the data view.
accessHandle.read(dataView, {at: 9});
// Logs `"More content"`.
console.log(textDecoder.decode(dataView));
// Truncate the file after 4 bytes.
accessHandle.truncate(4);
किसी फ़ाइल को ओरिजनल प्राइवेट फ़ाइल सिस्टम से, उपयोगकर्ता को दिखने वाले फ़ाइल सिस्टम में कॉपी करना
ऊपर बताया गया है कि फ़ाइलों को ओरिजनल प्राइवेट फ़ाइल सिस्टम से उपयोगकर्ता को दिखने वाले फ़ाइल सिस्टम में ट्रांसफ़र नहीं किया जा सकता. हालांकि, फ़ाइलों को कॉपी किया जा सकता है. showSaveFilePicker()
सिर्फ़ मुख्य थ्रेड पर दिखता है, लेकिन वर्कर थ्रेड में नहीं. इसलिए, पक्का करें कि कोड को मुख्य थ्रेड पर चलाया गया हो.
// On the main thread, not in the Worker. This assumes
// `fileHandle` is the `FileSystemFileHandle` you obtained
// the `FileSystemSyncAccessHandle` from in the Worker
// thread. Be sure to close the file in the Worker thread first.
const fileHandle = await opfsRoot.getFileHandle('fast');
try {
// Obtain a file handle to a new file in the user-visible file system
// with the same name as the file in the origin private file system.
const saveHandle = await showSaveFilePicker({
suggestedName: fileHandle.name || ''
});
const writable = await saveHandle.createWritable();
await writable.write(await fileHandle.getFile());
await writable.close();
} catch (err) {
console.error(err.name, err.message);
}
ओरिजिन के प्राइवेट फ़ाइल सिस्टम को डीबग करना
जब तक DevTools में पहले से मौजूद सहायता की सुविधा नहीं जुड़ जाती, तब तक (crbug/1284595 देखें), ऑरिजिन प्राइवेट फ़ाइल सिस्टम को डीबग करने के लिए, OPFS Explorer Chrome एक्सटेंशन का इस्तेमाल करें. ऊपर दिया गया स्क्रीनशॉट, नई फ़ाइलें और फ़ोल्डर बनाना सेक्शन से लिया गया है. यह सीधे तौर पर एक्सटेंशन से लिया गया है.
एक्सटेंशन इंस्टॉल करने के बाद, Chrome DevTools खोलें. इसके बाद, OPFS Explorer टैब चुनें. अब फ़ाइल के क्रम की जांच की जा सकती है. फ़ाइल के नाम पर क्लिक करके, ओरिजिन प्राइवेट फ़ाइल सिस्टम से उपयोगकर्ता को दिखने वाले फ़ाइल सिस्टम में फ़ाइलें सेव करें. साथ ही, ट्रैश कैन आइकॉन पर क्लिक करके फ़ाइलें और फ़ोल्डर मिटाएं.
डेमो
अगर आपने OPFS Explorer एक्सटेंशन इंस्टॉल किया है, तो डेमो में ओरिजन प्राइवेट फ़ाइल सिस्टम को ऐक्शन में देखें. यह WebAssembly में कंपाइल किए गए SQLite डेटाबेस के लिए बैकएंड के तौर पर इसका इस्तेमाल करता है. Glitch पर मौजूद सोर्स कोड को ज़रूर देखें. ध्यान दें कि नीचे दिए गए एम्बेड किए गए वर्शन में, ऑरिजिन प्राइवेट फ़ाइल सिस्टम बैकएंड का इस्तेमाल नहीं किया गया है. ऐसा इसलिए, क्योंकि iframe क्रॉस-ऑरिजिन है. हालांकि, डेमो को अलग टैब में खोलने पर, ऐसा होता है.
मीटिंग में सामने आए नतीजे
WHATWG के मुताबिक, ऑरिजिन प्राइवेट फ़ाइल सिस्टम (ओपीएफ़एस) ने वेब पर फ़ाइलों को इस्तेमाल करने और उनके साथ इंटरैक्ट करने के तरीके को बदल दिया है. इसकी मदद से, इस्तेमाल के ऐसे नए उदाहरणों को पूरा किया जा सकता है जिन्हें उपयोगकर्ता को दिखने वाले फ़ाइल सिस्टम की मदद से पूरा नहीं किया जा सकता था. ब्राउज़र बनाने वाली सभी मुख्य कंपनियां—Apple, Mozilla, और Google—इस पहल में शामिल हैं और इनका एक ही लक्ष्य है. ओरिजिन प्राइवेट फ़ाइल सिस्टम को डेवलप करने में कई लोगों ने मिलकर काम किया है. साथ ही, डेवलपर और उपयोगकर्ताओं से मिले सुझावों से इसे बेहतर बनाने में मदद मिली है. हम इस स्टैंडर्ड को लगातार बेहतर बना रहे हैं. इसलिए, अगर आपको कोई सुझाव देना है या शिकायत करनी है, तो whatwg/fs repository पर जाकर, समस्या या पुल अनुरोध के तौर पर अपना सुझाव/शिकायत सबमिट करें.
इसी विषय से जुड़े कुछ लिंक
- फ़ाइल सिस्टम के स्टैंडर्ड स्पेसिफ़िकेशन
- File System Standard repo
- The File System API with Origin Private File System WebKit post
- OPFS Explorer एक्सटेंशन
Acknowledgements
इस लेख की समीक्षा ऑस्टिन सुली, एटियन नोएल, और राहेल एंड्रू ने की है. Unsplash पर Christina Rumpf की हीरो इमेज.