File operations

To perform any file operations including read, write, create, and delete, UXP provides a couple of options. But before we look at the APIs, let's get acquainted with a few concepts.

System requirements

Please make sure your local environment uses the following application versions before proceeding.

  • InDesign v18.5 or higher
  • UDT v1.9.0 or higher
  • Manifest version v5 or higher

Concepts

Sandbox and other locations

UXP, by default, only allows access to certain locations in the user's file system. These locations are referred to as the sandbox.

However, we understand that there are circumstances when you would like to access other file locations as well. Accessing such locations is possible in UXP but you will need to seek permission first.

Manifest permission

For file system access you require permission for the localFileSystem module.

Let's understand the manifest settings a bit more in detail.

Copied to your clipboard
/* manifest.json */
{
"requiredPermissions": {
"localFileSystem": "plugin"
}
}

Allowed values for localFileSystem are:

  • plugin: Default value. Provides access to the sandbox.
  • request: Let's you request the user to select a location of their choice.
  • fullAccess: Provides full access to the user file system

Protip: Make sure you pick the most accurate permission for your use case because in the future we may ask users to provide their consent based on it. You may find 'fullAccess' to be the least restrictive and hence the safest to pick, but a user may not be comfortable giving full access to their system unless it's absolutely necessary and might deny the installation of your plugin.

Schemes

UXP provides a shorthand representation of these locations via schemes.

For sandbox, you can use plugin:/, plugin-data:/and plugin-temp:/. And, for other locations, use file:/.

Copied to your clipboard
<img src="plugin:/sample.png" />
<img src="file:/Users/user/Downloads/sample.png" /> <!-- update the path based on your system -->

Example

You have two options to access the file system - LocalFileSytem and FS module. These modules are very similar in terms of the capabilities they offer, however, there is a difference in the way they carry out the task.

LocalFileSytem APIs work with an object reference called Entry. Having an object reference makes it easier to manage and perform multiple operations. Whereas the FS APIs are very similar to NodeJS path-based file system APIs which make them ideal for carrying out single operations.

LocalFileSytem API

Available via require('uxp').storage.localFileSystem which returns an instance of FileSystemProvider.

Nomenclature

A file system is a conglomerate of files and folders. You can use the File and Folder classes to refer to them. However, these classes have a base class called Entry. The nomenclature of some of the APIs uses 'entry' in them typically when the output type can be either of the two - File or Folder and be determined only at runtime. Therefore, it's a good practice to check using isfile or isFolder before using specific APIs.

Copied to your clipboard
const { localFileSystem, types } = require('uxp').storage;
async function foo(){
// Create new folder and file
try {
const newFolderEntry = await localFileSystem.createEntryWithUrl("plugin-temp:/temp", { type: types.folder });
if (newFolderEntry.isFolder) {
const newFile = await newFolderEntry.createFile("temp.txt", {overwrite: true});
await newFile.write("Sample file created.");
}
} catch (e) {
console.error(e);
}
}

Now let's take a look at some examples to access system locations based on permission settings.

Accessing sandbox

Copied to your clipboard
const fsProvider = require('uxp').storage.localFileSystem;
async function foo() {
// Access sandbox
if (fsProvider.isFileSystemProvider) {
try {
const pluginFolder = await fsProvider.getEntryWithUrl("plugin-data:/");
console.log(`File path: ${pluginFolder.nativePath}`);
} catch (e) {
console.error(e);
}
}
}

Accessing other locations

Copied to your clipboard
const fsProvider = require('uxp').storage.localFileSystem;
async function foo() {
// Access other location
if (fsProvider.isFileSystemProvider) {
try {
const pluginFolder = await fsProvider.getEntryWithUrl("file:/Users/user/Documents"); // update the path based on your system
console.log(`File path: ${pluginFolder.nativePath}`);
} catch (e) {
console.error(e);
}
}
}

User's choice of location

These APIs are particularly handy when you want to request the user to select a folder/file location of their choice. For example, the below code snippet using getFileForOpening and getFileForSaving methods presents a file picker for the user to choose from.

Copied to your clipboard
const fsProvider = require('uxp').storage.localFileSystem;
async function foo() {
// Ask user to select a file. Show their 'Desktop' as the default folder.
if (fsProvider.isFileSystemProvider) {
const { domains, fileTypes } = require('uxp').storage;
try {
const file = await fsProvider.getFileForOpening({ initialDomain: domains.userDesktop, types: fileTypes.text });
if (!file) {
console.error("Something went wrong.");
return;
}
// read the file content
const text = await file.read();
console.log(`File content: ${text}`);
} catch (err) {
console.error(err);
}
}
}
async function bar() {
// Ask user to select a location to save a file
if (fsProvider.isFileSystemProvider) {
try {
const file = await fsProvider.getFileForSaving("sample.txt", { types: ["txt"] });
if (!file) {
console.error("Something went wrong.");
return;
}
// write content to file
await file.write("UXP saved sample file.");
} catch (err) {
console.error(err);
}
}
}

Save user's choice of location

If you would like to remember the user's choice for an extended period, you can do it with the help of a token. There are two types of tokens you can create

  • Session token - has a shorter span. It lasts until the plugin is 'Unloaded' from UDT or the script finishes its execution or the application is closed.
  • Persistent token - is more permanent in nature, and can last for multiple sessions or until the plugin is 'Uninstalled'. (Since scripts only last for the time of their execution, you should not use persistent tokens in scripts.)

The example below shows the essence of this usage but you should ideally save these tokens in the storage (more details covered in Storage section) for later use.

Copied to your clipboard
const fsProvider = require('uxp').storage.localFileSystem;
const { domains, fileTypes } = require('uxp').storage;
const entry = await fsProvider.getFileForOpening({ initialDomain: domains.userDesktop, types: fileTypes.text });
const token = await fsProvider.createPersistentToken(entry); // store for future use
async function readFileUsingTokensInLocalFileSystem() {
// In the future, access the entry using the token directly
try {
const file = await fsProvider.getEntryForPersistentToken(token);
const text = await file.read();
console.log(`File content: ${text}`);
} catch (err) {
console.error(err);
}
}

Reference material

FS Module

Based on NodeJS file system APIs, these provide direct access to file locations with the help of schema.

Accessing sandbox

Copied to your clipboard
const fs = require("fs");
async function foo() {
// Read a file from sandbox using fs module
try {
const text = await fs.readFile("plugin:/sample.txt", 'utf8');
console.log(`File content: ${text}`);
} catch (e) {
console.error(e);
}
}

Accessing other locations

Copied to your clipboard
const fs = require("fs");
async function foo() {
// Write to a arbitrary location using fs module
try {
await fs.writeFile("/Users/user/Desktop/output.txt", "This is a sample text.", {encoding: "utf-8"}); // update the path based on your system
} catch (e) {
console.error(e);
}
}

Reference material

Additional Notes

  • Despite the manifest setting fullAccess, certain files and folders may still not be accessible. It depends on the permission of the Operating System. For Win32 and Mac, these APIs can access anywhere in the file system. But, in UWP, accessing the system folder is prohibited.
padmkris123
Was this helpful?