跳到主要内容

数据防泄漏服务开发指导

DLP是系统提供的系统级的数据防泄漏解决方案,提供一种称为DLP的文件格式。后缀格式为“原始文件名(包含原始文件后缀).dlp”,例如“test.docx.dlp”,文件由授权凭证和原始文件密文组成。

通过端云协同认证(需要联网)来获取文件的访问授权,授权类型包含只读、编辑、文件拥有者三种。

  • 只读:能读取文件内容但不能修改。
  • 编辑:能够读写文件内容,但不能修改文件权限配置。
  • 文件拥有者:可读写文件、修改权限配置、恢复原始文件等。

应用需要访问DLP文件时,系统会自动安装应用的DLP沙箱分身应用,相当于完全独立的应用,数据和配置会继承原应用,但相互之间并不共享。分身应用在运行时会处于DLP沙箱环境中,访问外部的权限会被限制,以防止数据的泄漏。每当打开一个新的DLP文件会生成一个应用沙箱分身,沙箱应用之间也是相互隔离的,当应用关闭后应用分身会自动卸载,沙箱期间产生的临时数据也会丢弃。

正常情况下,应用不会感知到沙箱的存在,访问的也是解密后的明文,和访问普通文件没有区别,但由于DLP沙箱会限制其访问外部的权限(例如网络、剪切板、截屏、录屏、蓝牙等)。为了更好的用户体验,需要应用进行适配,例如文件只读的情况下,不应显示“保存”按钮,不应主动联网等。

沙箱限制

当应用进入DLP沙箱状态时,可以申请的权限将受到限制,根据DLP文件授权类型不同,限制也不相同,如下表:

权限名说明授权类型:只读授权类型:编辑/文件拥有者
ohos.permission.USE_BLUETOOTH允许应用使用蓝牙。禁止禁止
ohos.permission.INTERNET允许应用访问网络。禁止禁止
ohos.permission.DISTRIBUTED_DATASYNC允许应用与远程设备交换用户数据(如图片、音乐、视频、及应用数据等)。禁止禁止
ohos.permission.WRITE_MEDIA应用读写用户媒体文件,如视频、音频、图片等,需要申请此权限。禁止允许
ohos.permission.NFC_TAG允许应用使用NFC。禁止允许

接口说明

接口名描述
isDLPFile(fd: number): Promise<boolean> isDLPFile(fd: number, callback: AsyncCallback<boolean>): void判断是否是dlp文件。
getDLPPermissionInfo(): Promise<DLPPermissionInfo> getDLPPermissionInfo(callback: AsyncCallback<DLPPermissionInfo>): void获取当前沙箱应用的权限类型。
getOriginalFileName(fileName: string): string获取dlp文件原始文件名。
getDLPSuffix(): string获取dlp文件dlp后缀名。
on(type: 'openDLPFile', listener: Callback<AccessedDLPFileInfo>): void注册dlp文件打开事件监听,用于原始应用获取dlp文件打开事件。
off(type: 'openDLPFile', listener?: Callback<AccessedDLPFileInfo>): void取消dlp文件打开事件监听。
isInSandbox(): Promise<boolean> isInSandbox(callback: AsyncCallback<boolean>): void判断当前是否是dlp沙箱应用。
getDLPSupportedFileTypes(): Promise<Array<string>> getDLPSupportedFileTypes(callback: AsyncCallback<Array<string>>): void获取当前系统支持添加权限保护的文件格式类型。
setRetentionState(docUris: Array<string>): Promise<void> setRetentionState(docUris: Array<string>, callback: AsyncCallback<void>): void设置dlp分身应用保留状态。
cancelRetentionState(docUris: Array<string>): Promise<void> cancelRetentionState(docUris: Array<string>, callback: AsyncCallback<void>): void取消dlp分身应用保留状态。
getRetentionSandboxList(bundleName?: string): Promise<Array<RetentionSandboxInfo>> getRetentionSandboxList(bundleName: string, callback: AsyncCallback<Array<RetentionSandboxInfo>>): void getRetentionSandboxList(callback: AsyncCallback<Array<RetentionSandboxInfo>>): void获取当前保留沙箱列表。
getDLPFileAccessRecords(): Promise<Array<AccessedDLPFileInfo>> getDLPFileAccessRecords(callback: AsyncCallback<Array<AccessedDLPFileInfo>>): void获取dlp文件访问记录。
setSandboxAppConfig(configInfo: string): Promise<void>设置沙箱应用配置信息。
getSandboxAppConfig(): Promise<string>查询沙箱应用配置信息。
cleanSandboxAppConfig(): Promise<void>清理沙箱应用配置信息。
startDLPManagerForResult(context: common.UIAbilityContext, want: Want): Promise<DLPManagerResult>在当前UIAbility界面以无边框形式打开DLP权限管理应用(只支持Stage模型)。

开发步骤

DLP作为HarmonyOS系统级数据防泄漏方案,可以让应用在零适配或低代码适配的情况下接入DLP能力,打开DLP文件。

当用户使用默认应用或指定应用打开DLP文件时,DLP框架将会完成:

  1. 安装此应用的DLP沙箱分身应用。
  2. 为这个DLP文件绑定一个FUSE文件。
  3. 将FUSE文件分享给DLP沙箱分身应用。

实现DLP沙箱分身在无感加解密流程下访问DLP文件解密后的内容。

当三方应用接入DLP(支持打开DLP文件)时,为了更优的体验,可从以下方面完成适配。

预置操作

本文档提供接口示例代码,如需要了解工程项目创建方式,可参考工程创建

应用接入DLP能力,支持被安装为DLP沙箱分身应用,打开DLP文件,需要具备以下条件:

  • 应用需要支持打开以下文件类型中的其中一种或几种,也就是当前DLP支持的文件类型。包括:

    ".doc", ".docm", ".docx", ".dot", ".dotm", ".dotx", ".odp", ".odt", ".pdf", ".pot", ".potm", ".potx", ".ppa",
    ".ppam", ".pps", ".ppsm", ".ppsx", ".ppt", ".pptm", ".pptx", ".rtf", ".txt", ".wps", ".xla", ".xlam", ".xls",
    ".xlsb", ".xlsm", ".xlsx", ".xlt", ".xltm", ".xltx", ".xlw", ".xml", ".xps"
  • 应用需要具备ohos.want.action.viewData或ohos.want.action.editData的skills,可在module.json5文件中增加相应配置:

    "skills":[
    {
    "entities":[
    ...
    ],
    "actions":[
    ...
    "ohos.want.action.viewData"
    ]
    }
    ]
  • 使用的设备需要具备域账号环境。

导入模块

引入dlpPermission模块。

import { dlpPermission } from '@kit.DataProtectionKit';

应用支持打开DLP文件绑定的FUSE文件

一般情况下,应用如果支持打开预置操作中指定文件类型的文件,没有对传入的Want做特定限制的情况下,不需要适配即可打开FUSE文件。

打开DLP文件时,应用被安装为DLP沙箱分身应用(后续简称为分身),分身会收到want请求,分身可以对其中一些字段进行解析:

import { Want } from '@kit.AbilityKit';

interface DLPUriObj {
name: string
};

interface DLPWriteable {
name:boolean
};

interface DLPNameObj {
dateModified: string,
displayName: string,
relativePath: string,
};

interface DLPLinkNameObj {
name: string
};

function getParams(want: Want) {
// 接收打开DLP文件传过来的参数
let dlpFuseUri: string = want.uri? want.uri : ''; // FUSE文件的uri, 存放解密后的明文
let dlpFuseWriteable: boolean = (want.parameters?.linkFileWriteable as DLPWriteable).name; // 对FUSE文件是否有写权限
let dlpUri: string = (want.parameters?.dlpUri as DLPUriObj).name; // DLP文件的uri
let dlpName: string = (want.parameters?.fileAsset as DLPNameObj).displayName; // DLP文件的文件名
let dlpFuseName: string = (want.parameters?.linkFileName as DLPLinkNameObj).name; // FUSE文件的文件名
}

分身可以通过把want.uri打开为fd,获取FUSE文件的内容:

import { fileIo } from '@kit.CoreFileKit';
import { util } from '@kit.ArkTS';
import { hilog } from '@kit.PerformanceAnalysisKit';

const TAG: string = 'dlp';

function readFileContent(dlpFuseUri: string): string {
let content: string = '';
let file: fileIo.File | undefined = undefined;
try {
file = fileIo.openSync(dlpFuseUri, fileIo.OpenMode.READ_ONLY);
} catch (err) {
hilog.error(0x0000, TAG, 'openSync failed. ' + err);
if (file) {
fileIo.closeSync(file);
}
return content;
}

try {
let buffer = new ArrayBuffer(4096);
let bytesRead = fileIo.readSync(file.fd, buffer);
let actualBuffer = buffer.slice(0, bytesRead);
content = bufferToString(actualBuffer);
} catch (err) {
hilog.error(0x0000, TAG,'readSync failed. ' + err);
} finally {
if (file) {
fileIo.closeSync(file);
}
}
return content;
}

function bufferToString(buffer: ArrayBuffer): string {
let textDecoder = new util.TextDecoder('utf-8', {
ignoreBOM: true
});
return textDecoder.decodeToString(new Uint8Array(buffer), {
stream: true
});
}

如果有FUSE文件的读写权限,也可以更新FUSE文件内容:

import { fileIo } from '@kit.CoreFileKit';
import { hilog } from '@kit.PerformanceAnalysisKit';

const TAG: string = 'dlp';

function writeFileContent(dlpFuseUri: string, content: string): void {
let file: fileIo.File | undefined = undefined;

try {
file = fileIo.openSync(dlpFuseUri, fileIo.OpenMode.CREATE | fileIo.OpenMode.READ_WRITE);
// O_RDWR: 读写模式, O_CREAT: 文件不存在时创建
let writeLen: number = fileIo.writeSync(file.fd, content);

} catch (err) {
hilog.error(0x0000, TAG, '文件操作失败: ' + err);
} finally {
if (file) {
fileIo.closeSync(file);
}
}
}

应用根据DLP文件的权限对界面进行适配

DLP沙箱分身中可以调用getDLPPermissionInfo方法查询当前系统登陆的域账号用户对本DLP文件的用户权限和操作权限,不同用户权限可以对应不同的对文档的操作权限。

import { dlpPermission } from '@kit.DataProtectionKit';
import { BusinessError } from '@kit.BasicServicesKit';
import { hilog } from '@kit.PerformanceAnalysisKit';

const TAG: string = 'dlp';

dlpPermission.getDLPPermissionInfo().then((data: dlpPermission.DLPPermissionInfo)=> {
let userAccess: dlpPermission.DLPFileAccess = data.dlpFileAccess; // 用户对本DLP文件的用户权限
let isEditable: number = data.flags & dlpPermission.ActionFlagType.ACTION_EDIT; // 用户对本DLP文件的操作权限
}).catch((err: BusinessError) => {
hilog.error(0x0000, TAG, 'getDLPPermissionInfo: ' + JSON.stringify(err));
});

getDLPPermissionInfo返回的data为DLPPermissionInfo类型,其中dlpFileAccess表示用户权限,flags表示操作权限的按位组合的结果。可以根据返回的flags字段对照ActionFlagType判断DLP沙箱分身是否具有对应的操作权限,可以用于界面按钮置灰操作等。

应用与DLP沙箱分身数据共享

DLP沙箱分身是普通应用的分身,所有数据都是全新的,如果二者之间有些数据需要实现共享,可以通过DLP框架提供的应用与DLP沙箱分身数据共享机制实现。一种典型的使用场景是原应用与DLP沙箱分身之间共用是否已经弹出过隐私声明弹框的配置信息。

一般包括下面四种读写配置信息前后顺序组合:

  • 原应用写配置,原应用读配置。
  • 原应用写配置,DLP沙箱分身读配置。
  • DLP沙箱分身写配置,DLP沙箱分身读配置。
  • DLP沙箱分身写配置,原应用读配置。

约束与限制

  • 每次调用设置配置信息接口会覆盖上次调用的设置内容。
  • 出于数据防泄漏考虑,DLP沙箱分身写配置需要在读取FUSE文件内容之前完成。

具体步骤

  1. 设置配置信息。

    把需要保存的配置信息转成string类型,调用setSandboxAppConfig接口设置配置信息。

    普通应用和DLP沙箱分身都可以调用该接口,但DLP沙箱分身必须在读取DLP文件内容之前才允许调用。

    import { dlpPermission } from '@kit.DataProtectionKit';
    import { BusinessError } from '@kit.BasicServicesKit';
    import { hilog } from '@kit.PerformanceAnalysisKit';

    const TAG: string = 'dlp';

    async function setSandboxAppConfig() {
    try {
    await dlpPermission.setSandboxAppConfig('configInfo'); // 设置配置信息
    } catch (err) {
    hilog.error(0x0000, TAG, 'setSandboxAppConfig error: ' + JSON.stringify(err)); // 失败报错
    }
    }
  2. 清理配置信息。

    调用cleanSandboxAppConfig接口清理该应用的所有配置信息。

    该接口只允许普通应用中调用。

    import { dlpPermission } from '@kit.DataProtectionKit';
    import { BusinessError } from '@kit.BasicServicesKit';
    import { hilog } from '@kit.PerformanceAnalysisKit';

    const TAG: string = 'dlp';

    async function cleanSandboxAppConfig() {
    try {
    await dlpPermission.cleanSandboxAppConfig(); // 清理配置信息
    } catch (err) {
    hilog.error(0x0000, TAG, 'cleanSandboxAppConfig error: ' + JSON.stringify(err)); // 失败报错
    }
    }
  3. 获取配置信息。

    调用getSandboxAppConfig查询该应用的所有配置信息。

    普通应用和DLP沙箱分身都可以调用。

    import { dlpPermission } from '@kit.DataProtectionKit';
    import { BusinessError } from '@kit.BasicServicesKit';
    import { hilog } from '@kit.PerformanceAnalysisKit';

    const TAG: string = 'dlp';

    async function getSandboxAppConfig() {
    try {
    let res:string = await dlpPermission.getSandboxAppConfig(); // 查询配置信息
    } catch (err) {
    hilog.error(0x0000, TAG, 'getSandboxAppConfig error: ' + JSON.stringify(err)); // 失败报错
    }
    }

应用支持更新最近打开记录

当应用有最近打开记录场景时,可以使用DLP框架提供的接口适配最近打开记录。可从以下场景适配:

  • 普通应用未启动,无法感知到DLP沙箱分身打开的DLP文件

    仅有DLP沙箱分身有打开DLP文件场景:普通应用启动时,可以通过接口获取到历史通过本应用的DLP沙箱分身打开的DLP文件。

    import { dlpPermission } from '@kit.DataProtectionKit';
    import { BusinessError } from '@kit.BasicServicesKit';
    import { hilog } from '@kit.PerformanceAnalysisKit';

    const TAG: string = 'dlp';

    async function getDLPFileAccessRecords() {
    try {
    let res:Array<dlpPermission.AccessedDLPFileInfo> = await dlpPermission.getDLPFileAccessRecords(); // 获取DLP访问列表
    hilog.info(0x0000, TAG, 'res' + JSON.stringify(res))
    } catch (err) {
    hilog.error(0x0000, TAG, 'error:' + JSON.stringify(err)); // 失败报错
    }
    }
  • 普通应用已启动,可以感知到DLP沙箱分身打开的DLP文件

    DLP沙箱分身有打开DLP文件场景:普通应用可以订阅本应用的DLP沙箱分身打开DLP文件的事件。

    import { AbilityConstant, UIAbility, Want } from '@kit.AbilityKit';
    import { dlpPermission } from '@kit.DataProtectionKit';
    import { BusinessError } from '@kit.BasicServicesKit';
    import { hilog } from '@kit.PerformanceAnalysisKit';

    const TAG: string = 'dlp';


    export default class TestAbility extends UIAbility {
    onCreate(want: Want, launchParam: AbilityConstant.LaunchParam): void {
    this.subscribe();
    }

    onDestroy(): void {
    this.unSubscribe();
    }

    event(info: dlpPermission.AccessedDLPFileInfo) {
    hilog.info(0x0000, TAG, 'openDlpFile event');
    }

    unSubscribe() {
    try {
    dlpPermission.off('openDLPFile', this.event); // 取消订阅
    } catch (err) {
    hilog.error(0x0000, TAG, 'error:' + JSON.stringify(err)); // 失败报错
    }
    }

    subscribe() {
    try {
    dlpPermission.on('openDLPFile', this.event); // 订阅
    } catch (err) {
    hilog.error(0x0000, TAG, 'error:' + JSON.stringify(err)); // 失败报错
    }
    }
    }

应用内支持打开选定的DLP文件

应用可以支持从最近打开列表、文件选择器中选择DLP文件,打开DLP文件的场景,按如下流程适配:

  1. 设置Want参数,指定action为"ohos.want.action.viewData",bundleName、abilityName分别为选择打开DLP文件的应用的bundleName、abilityName,uri为需要打开的DLP文件的uri,在parameters中设置fileName的name值为DLP文件的文件名。
  2. 获取UIAbilityContext的context。
  3. 调用context的startAbility方法传入want参数,打开dlp文件。
import { Want, common } from '@kit.AbilityKit';
import { BusinessError } from '@kit.BasicServicesKit';
import { UIContext } from '@kit.ArkUI';
import { hilog } from '@kit.PerformanceAnalysisKit';

const TAG: string = 'dlp';

function openDlpFile(dlpUri: string, fileName: string) {
let want: Want = {
"action": "ohos.want.action.viewData",
"bundleName": "com.example.example_bundle_name",
"abilityName": "exampleAbility",
"uri": dlpUri,
"parameters": {
"fileName": {
"name": fileName
}
}
}
let context = new UIContext().getHostContext() as common.UIAbilityContext; // 获取当前UIAbilityContext
try {
hilog.info(0x0000, TAG, 'openDLPFile:' + JSON.stringify(want));
hilog.info(0x0000, TAG, 'openDLPFile: delegator:' + JSON.stringify(context));
context.startAbility(want);
} catch (err) {
hilog.error(0x0000, TAG, 'openDLPFile startAbility failed:' + JSON.stringify(err));
return;
}
}

应用内支持对DLP文件权限设置

应用内可以集成权限设置按钮,当已打开一个普通文件后,点击权限设置按钮,拉起DLP管理应用的模态设置权限页面,生成DLP文件。也可以在DLP沙箱分身中查看当前正在打开的DLP文件的操作权限。

  • 普通应用内权限设置

    以无边框形式打开DLP权限管理应用。

    • 此方法只能在UIAbility上下文中调用,只支持Stage模式。
    • want参数中uri的值为普通文件uri,parameters.displayName为文件名,这两个值为必传参数。
    • 调用dlpPermission.startDLPManagerForResult拉起DLP管理应用的设置权限页面,输入相关的域账号信息,点击保存,在拉起的filepicker中选择DLP文件的保存路径,保存DLP文件。

    调用以下代码:

    import { common, Want } from '@kit.AbilityKit';
    import { BusinessError } from '@kit.BasicServicesKit';
    import { dlpPermission } from '@kit.DataLossPreventionKit';
    import { UIContext } from '@kit.ArkUI';
    import { hilog } from '@kit.PerformanceAnalysisKit';

    const TAG: string = 'dlp';

    try {
    let fileUri: string = "file://docs/storage/Users/currentUser/test.txt";
    let fileName: string = "test.txt";
    let context = new UIContext().getHostContext() as common.UIAbilityContext; // 获取当前UIAbilityContext
    let want: Want = {
    'uri': fileUri,
    'parameters': {
    'displayName': fileName
    }
    }; // 请求参数
    dlpPermission.startDLPManagerForResult(context, want).then((res: dlpPermission.DLPManagerResult) => {
    hilog.info(0x0000, TAG, 'startDLPManagerForResult res.resultCode:' + res.resultCode);
    hilog.info(0x0000, TAG, 'startDLPManagerForResult res.want:' + JSON.stringify(res.want));
    }); // 拉起DLP权限管理应用 设置权限
    } catch (err) {
    hilog.error(0x0000, TAG, 'startDLPManagerForResult error:' + (err as BusinessError).code + (err as BusinessError).message);
    }
  • DLP沙箱分身内权限修改,查看和解除

    • 如果当前的账号是DLP文档的创建者,则该用户拥有修改这个DLP文件权限或者解除这个DLP文档权限还原为普通文件的能力,调用以下代码,拉起DLP管理应用的设置权限页面,在该页面中选择更改加密进行权限修改或者解除加密;如果当前账号拥有DLP文档只读或者编辑权限,调用以下代码则可以查看当前用户权限内容。

      import { common, Want } from '@kit.AbilityKit';
      import { BusinessError } from '@kit.BasicServicesKit';
      import { dlpPermission } from '@kit.DataLossPreventionKit';
      import { UIContext } from '@kit.ArkUI';
      import { hilog } from '@kit.PerformanceAnalysisKit';

      const TAG: string = 'dlp';

      try {
      let fileUri: string = "file://docs/storage/Users/currentUser/test.txt.dlp";// DLP文件的uri
      let fileName: string = "test.txt.dlp";
      let context = new UIContext().getHostContext() as common.UIAbilityContext; // 获取当前UIAbilityContext
      let want: Want = {
      'uri': fileUri,
      'parameters': {
      'displayName': fileName
      }
      }; // 请求参数
      dlpPermission.startDLPManagerForResult(context, want).then((res: dlpPermission.DLPManagerResult) => {
      hilog.info(0x0000, TAG, 'startDLPManagerForResult res.resultCode:' + res.resultCode);
      hilog.info(0x0000, TAG, 'startDLPManagerForResult res.want:' + JSON.stringify(res.want));
      }); // 拉起DLP权限管理应用 设置权限
      } catch (err) {
      hilog.error(0x0000, TAG, 'startDLPManagerForResult error:' + (err as BusinessError).code + (err as BusinessError).message);
      }
    • DLP沙箱分身中可以调用getDLPPermissionInfo方法查询当前用户DLP文件权限,DLP沙箱分身的权限限制,参考沙箱限制

      import { dlpPermission } from '@kit.DataProtectionKit';
      import { BusinessError } from '@kit.BasicServicesKit';
      import { hilog } from '@kit.PerformanceAnalysisKit';

      const TAG: string = 'dlp';

      dlpPermission.getDLPPermissionInfo().then((data:dlpPermission.DLPPermissionInfo)=> {
      hilog.info(0x0000, TAG, 'getDLPPermissionInfo, result: ' + JSON.stringify(data));
      }).catch((err: BusinessError) => {
      hilog.error(0x0000, TAG, 'getDLPPermissionInfo: ' + JSON.stringify(err));
      });

其他DLP能力增强

  • 判断一个文件是否是DLP文件

    传入文件的fd查询对应文件是否是DLP文件,是DLP文件则按文档指导打开该文件。

    import { dlpPermission } from '@kit.DataProtectionKit';
    import { fileIo } from '@kit.CoreFileKit';
    import { BusinessError } from '@kit.BasicServicesKit';
    import { hilog } from '@kit.PerformanceAnalysisKit';

    const TAG: string = 'dlp';

    let uri = "file://docs/storage/Users/currentUser/Desktop/test.txt.dlp";
    let file = fileIo.openSync(uri);
    try {
    let res: boolean = await dlpPermission.isDLPFile(file.fd); // 是否加密DLP文件
    hilog.info(0x0000, TAG, 'res' + JSON.stringify(res));
    } catch (err) {
    hilog.error(0x0000, TAG, 'startDLPManagerForResult error:' + (err as BusinessError).code + (err as BusinessError).message); // 失败报错
    }
    fileIo.closeSync(file);
  • 判断当前所在应用是否是DLP沙箱分身

    在应用中调用isInSandbox接口判断当前是否是DLP沙箱分身,如果是DLP沙箱分身则可以结合调用接口查询权限的结果进行对应功能按钮的置灰或屏蔽。比如:如果只有只读权限,则编辑保存入口可以置灰,如果是只读或者编辑权限,则修改权限入口可以置灰。

    import { dlpPermission } from '@kit.DataProtectionKit';
    import { BusinessError } from '@kit.BasicServicesKit';
    import { hilog } from '@kit.PerformanceAnalysisKit';

    const TAG: string = 'dlp';

    dlpPermission.isInSandbox().then((data: boolean)=> {
    hilog.info(0x0000, TAG, 'isInSandbox, result: ' + JSON.stringify(data));
    }).catch((err: BusinessError) => {
    hilog.error(0x0000, TAG, 'isInSandbox: ' + JSON.stringify(err));
    });
  • 保留沙箱

    DLP沙箱分身关闭后会进行沙箱卸载,如果不希望DLP沙箱分身关闭时卸载该沙箱可以在沙箱中调用设置保留沙箱接口,只有当再次调用取消保留沙箱接口并关闭DLP沙箱分身才会触发沙箱的卸载。

    • 调用接口setRetentionState设置保留沙箱,传入参数为本沙箱内打开的dlp文件的uri列表,该接口只允许在沙箱中调用。

      import { dlpPermission } from '@kit.DataProtectionKit';
      import { BusinessError } from '@kit.BasicServicesKit';
      import { hilog } from '@kit.PerformanceAnalysisKit';

      const TAG: string = 'dlp';


      async function setRetentionSandboxList() {
      let docUris: Array<string>=["file://docs/storage/Users/currentUser/Desktop/test.txt.dlp"]
      try {
      await dlpPermission.setRetentionState(docUris); // 设置沙箱保留
      } catch (err) {
      hilog.error(0x0000, TAG, 'startDLPManagerForResult error:' + (err as BusinessError).code + (err as BusinessError).message); // 失败报错
      }
      }
    • 调用接口cancelRetentionState取消保留沙箱,该接口只允许沙箱中调用。

      import { dlpPermission } from '@kit.DataProtectionKit';
      import { BusinessError } from '@kit.BasicServicesKit';
      import { hilog } from '@kit.PerformanceAnalysisKit';

      const TAG: string = 'dlp';


      async function setRetentionSandboxList() {
      let docUris: Array<string>=["file://docs/storage/Users/currentUser/Desktop/test.txt.dlp"]
      try {
      await dlpPermission.cancelRetentionState(docUris); // 取消保留沙箱
      } catch (err) {
      hilog.error(0x0000, TAG, 'startDLPManagerForResult error:' + (err as BusinessError).code + (err as BusinessError).message); // 失败报错
      }
      }
    • 调用接口getRetentionSandboxList获取保留沙箱记录,该接口允许原应用和DLP沙箱分身中调用。

      import { dlpPermission } from '@kit.DataProtectionKit';
      import { BusinessError } from '@kit.BasicServicesKit';
      import { hilog } from '@kit.PerformanceAnalysisKit';

      const TAG: string = 'dlp';

      async function getRetentionSandboxList() {
      try {
      let res:Array<dlpPermission.RetentionSandboxInfo> = await dlpPermission.getRetentionSandboxList(); // 获取保留沙箱记录
      hilog.info(0x0000, TAG, 'res' + JSON.stringify(res))
      } catch (err) {
      hilog.error(0x0000, TAG, 'startDLPManagerForResult error:' + (err as BusinessError).code + (err as BusinessError).message);// 失败报错
      }
      }

典型问题自排查

应用可以打开正常文件,无法打开FUSE文件

  • 排查是否对want做了特定限制,导致DLP沙箱分身无法获取到FUSE文件。
  • 排查是否以读写权限打开了只读的FUSE文件。