2020-03-19 06:32:49 -04:00
|
|
|
// Copyright 2018-2020 the Deno authors. All rights reserved. MIT license.
|
|
|
|
|
2020-07-13 12:23:24 -04:00
|
|
|
const { PermissionDenied } = Deno.errors;
|
|
|
|
|
2020-03-19 06:32:49 -04:00
|
|
|
function getPermissionString(descriptors: Deno.PermissionDescriptor[]): string {
|
|
|
|
return descriptors.length
|
2020-07-14 15:24:17 -04:00
|
|
|
? ` ${
|
|
|
|
descriptors
|
2020-03-28 13:03:49 -04:00
|
|
|
.map((pd) => {
|
2020-03-19 06:32:49 -04:00
|
|
|
switch (pd.name) {
|
|
|
|
case "read":
|
|
|
|
case "write":
|
|
|
|
return pd.path
|
|
|
|
? `--allow-${pd.name}=${pd.path}`
|
|
|
|
: `--allow-${pd.name}`;
|
|
|
|
case "net":
|
2020-12-30 17:35:28 -05:00
|
|
|
return pd.host
|
|
|
|
? `--allow-${pd.name}=${pd.host}`
|
2020-03-19 06:32:49 -04:00
|
|
|
: `--allow-${pd.name}`;
|
|
|
|
default:
|
|
|
|
return `--allow-${pd.name}`;
|
|
|
|
}
|
|
|
|
})
|
2020-07-14 15:24:17 -04:00
|
|
|
.join("\n ")
|
|
|
|
}`
|
2020-03-19 06:32:49 -04:00
|
|
|
: "";
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Attempts to grant a set of permissions, resolving with the descriptors of
|
|
|
|
* the permissions that are granted.
|
|
|
|
*
|
|
|
|
* const perms = await grant({ name: "net" }, { name: "read" });
|
|
|
|
* if (perms && perms.length === 2) {
|
|
|
|
* // do something cool that connects to the net and reads files
|
|
|
|
* } else {
|
|
|
|
* // notify user of missing permissions
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
* If one of the permissions requires a prompt, the function will attempt to
|
|
|
|
* prompt for it. The function resolves with all of the granted permissions. */
|
|
|
|
export async function grant(
|
|
|
|
...descriptors: Deno.PermissionDescriptor[]
|
|
|
|
): Promise<void | Deno.PermissionDescriptor[]>;
|
|
|
|
/** Attempts to grant a set of permissions, resolving with the descriptors of
|
|
|
|
* the permissions that are granted.
|
|
|
|
*
|
|
|
|
* const perms = await grant([{ name: "net" }, { name: "read" }]);
|
|
|
|
* if (perms && perms.length === 2) {
|
|
|
|
* // do something cool that connects to the net and reads files
|
|
|
|
* } else {
|
|
|
|
* // notify user of missing permissions
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
* If one of the permissions requires a prompt, the function will attempt to
|
|
|
|
* prompt for it. The function resolves with all of the granted permissions. */
|
|
|
|
export async function grant(
|
2020-07-14 15:24:17 -04:00
|
|
|
descriptors: Deno.PermissionDescriptor[],
|
2020-03-19 06:32:49 -04:00
|
|
|
): Promise<void | Deno.PermissionDescriptor[]>;
|
|
|
|
export async function grant(
|
|
|
|
descriptor: Deno.PermissionDescriptor[] | Deno.PermissionDescriptor,
|
|
|
|
...descriptors: Deno.PermissionDescriptor[]
|
|
|
|
): Promise<void | Deno.PermissionDescriptor[]> {
|
|
|
|
const result: Deno.PermissionDescriptor[] = [];
|
|
|
|
descriptors = Array.isArray(descriptor)
|
|
|
|
? descriptor
|
|
|
|
: [descriptor, ...descriptors];
|
|
|
|
for (const descriptor of descriptors) {
|
2020-07-13 12:23:24 -04:00
|
|
|
let state = (await Deno.permissions.query(descriptor)).state;
|
2020-03-19 06:32:49 -04:00
|
|
|
if (state === "prompt") {
|
2020-07-13 12:23:24 -04:00
|
|
|
state = (await Deno.permissions.request(descriptor)).state;
|
2020-03-19 06:32:49 -04:00
|
|
|
}
|
|
|
|
if (state === "granted") {
|
|
|
|
result.push(descriptor);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return result.length ? result : undefined;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Attempts to grant a set of permissions or rejects.
|
|
|
|
*
|
|
|
|
* await grantOrThrow({ name: "env" }, { name: "net" });
|
|
|
|
*
|
|
|
|
* If the permission can be prompted for, the function will attempt to prompt.
|
|
|
|
* If any of the permissions are denied, the function will reject for the first
|
|
|
|
* permission that is denied. If all permissions are granted, the function
|
|
|
|
* will resolve. */
|
|
|
|
export async function grantOrThrow(
|
|
|
|
...descriptors: Deno.PermissionDescriptor[]
|
|
|
|
): Promise<void>;
|
|
|
|
/** Attempts to grant a set of permissions or rejects.
|
|
|
|
*
|
|
|
|
* await grantOrThrow([{ name: "env" }, { name: "net" }]);
|
|
|
|
*
|
|
|
|
* If the permission can be prompted for, the function will attempt to prompt.
|
|
|
|
* If any of the permissions are denied, the function will reject mentioning the
|
|
|
|
* the denied permissions. If all permissions are granted, the function will
|
|
|
|
* resolve. */
|
|
|
|
export async function grantOrThrow(
|
2020-07-14 15:24:17 -04:00
|
|
|
descriptors: Deno.PermissionDescriptor[],
|
2020-03-19 06:32:49 -04:00
|
|
|
): Promise<void>;
|
|
|
|
export async function grantOrThrow(
|
|
|
|
descriptor: Deno.PermissionDescriptor[] | Deno.PermissionDescriptor,
|
|
|
|
...descriptors: Deno.PermissionDescriptor[]
|
|
|
|
): Promise<void> {
|
|
|
|
const denied: Deno.PermissionDescriptor[] = [];
|
|
|
|
descriptors = Array.isArray(descriptor)
|
|
|
|
? descriptor
|
|
|
|
: [descriptor, ...descriptors];
|
|
|
|
for (const descriptor of descriptors) {
|
2020-07-13 12:23:24 -04:00
|
|
|
const { state } = await Deno.permissions.request(descriptor);
|
2020-03-19 06:32:49 -04:00
|
|
|
if (state !== "granted") {
|
|
|
|
denied.push(descriptor);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (denied.length) {
|
2020-07-13 12:23:24 -04:00
|
|
|
throw new PermissionDenied(
|
2020-07-14 15:24:17 -04:00
|
|
|
`The following permissions have not been granted:\n${
|
|
|
|
getPermissionString(
|
|
|
|
denied,
|
|
|
|
)
|
|
|
|
}`,
|
2020-03-19 06:32:49 -04:00
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|