跳到主要内容

解决Web组件本地资源跨域问题

拦截本地资源跨域

为了提高安全性,ArkWeb内核禁止file协议和resource协议访问跨域请求。因此,在使用Web组件加载本地离线资源的时候,Web组件会拦截file协议和resource协议的跨域访问。通过方法二设置一个路径列表,再使用file协议访问该路径列表中的资源,允许跨域访问本地文件。Web组件无法访问本地跨域资源时,DevTools控制台会显示报错信息:

Access to script at 'xxx' from origin 'xxx' has been blocked by CORS policy: Cross origin requests are only supported for protocol schemes: http, arkweb, data, chrome-extension, chrome, https, chrome-untrusted.

本地资源跨域问题解决方法

  • 方法一

    开发者应使用http或https协议替代file或resource协议,使Web组件成功访问跨域资源。替代的URL域名为自定义构造,仅供个人或组织使用,避免与互联网上的实际域名冲突。同时,开发者需使用Web组件的onInterceptRequest方法,对本地资源进行拦截和相应的替换。

    以下结合示例说明如何使用http或者https等协议解决本地资源跨域访问失败的问题。其中,index.html和js/script.js置于工程中的rawfile目录下。当使用resource协议访问index.html时,js/script.js将因跨域而被拦截,无法加载。在示例中,使用https://www.example.com/域名替换了原本的resource协议,同时利用onInterceptRequest接口替换资源,使得js/script.js可以成功加载,从而解决了跨域拦截的问题

    import { webview } from '@kit.ArkWeb';

    @Entry
    @Component
    struct Index {
    @State message: string = 'Hello World';
    webviewController: webview.WebviewController = new webview.WebviewController();
    // 构造域名和本地文件的映射表
    schemeMap = new Map([
    ['https://www.example.com/index.html', 'index.html'],
    ['https://www.example.com/js/script.js', 'js/script.js'],
    ]);
    // 构造本地文件和构造返回的格式mimeType
    mimeTypeMap = new Map([
    ['index.html', 'text/html'],
    ['js/script.js', 'text/javascript']
    ]);

    build() {
    Row() {
    Column() {
    // 针对本地index.html,使用http或者https协议代替file协议或者resource协议,并且构造一个属于自己的域名。
    // 本例中构造www.example.com为例。
    Web({ src: 'https://www.example.com/index.html', controller: this.webviewController })
    .javaScriptAccess(true)
    .fileAccess(true)
    .domStorageAccess(true)
    .geolocationAccess(true)
    .width('100%')
    .height('100%')
    .onInterceptRequest((event) => {
    if (!event) {
    return;
    }
    // 此处匹配自己想要加载的本地离线资源,进行资源拦截替换,绕过跨域
    if (this.schemeMap.has(event.request.getRequestUrl())) {
    let rawfileName: string = this.schemeMap.get(event.request.getRequestUrl())!;
    let mimeType = this.mimeTypeMap.get(rawfileName);
    if (typeof mimeType === 'string') {
    let response = new WebResourceResponse();
    // 构造响应数据,如果本地文件在rawfile下,可以通过如下方式设置
    response.setResponseData($rawfile(rawfileName));
    response.setResponseEncoding('utf-8');
    response.setResponseMimeType(mimeType);
    response.setResponseCode(200);
    response.setReasonMessage('OK');
    response.setResponseIsReady(true);
    return response;
    }
    }
    return null;
    })
    }
    .width('100%')
    }
    .height('100%')
    }
    }
    <!-- main/resources/rawfile/index.html -->
    <html>
    <head>
    <meta name="viewport" content="width=device-width,initial-scale=1">
    </head>
    <body>
    <script crossorigin src="./js/script.js"></script>
    </body>
    </html>
    const body = document.body;
    const element = document.createElement('div');
    element.textContent = 'success';
    body.appendChild(element);
  • 方法二

    通过setPathAllowingUniversalAccess设置一个路径列表。当使用file协议访问该列表中的资源时,允许进行跨域访问本地文件。此外,一旦设置了路径列表,file协议将仅限于访问列表内的资源(此时,fileAccess的行为将会被此接口行为覆盖)。

    setPathAllowingUniversalAccess放开目录的跨域访问限制是一个高风险操作。基于最小权限原则,当前el1,el2放开的路径是固定的,路径列表中的路径应符合以下任一路径格式:

    1.应用文件目录通过Context.filesDir获取,其子目录示例如下:

    • /data/storage/el2/base/files/example
    • /data/storage/el2/base/haps/entry/files/example

    2.应用资源目录通过Context.resourceDir获取,其子目录示例如下:

    • /data/storage/el1/bundle/entry/resources/resfile
    • /data/storage/el1/bundle/entry/resources/resfile/example

    3.从API version 21开始,还包括了应用缓存目录通过Context.cacheDir获取,其子目录示例如下:

    • /data/storage/el2/base/cache
    • /data/storage/el2/base/haps/entry/cache/example
    • 设置的目录路径中,不允许包含cache/web,否则会抛出异常码401。如果设置目录路径是cache,cache/web也不允许访问。

    4.从API version 21开始,还包括了应用临时目录通过Context.tempDir获取,其子目录示例如下:

    • /data/storage/el2/base/temp
    • /data/storage/el2/base/haps/entry/temp/example

    当路径列表中的任一路径不满足上述条件时,系统将抛出异常码401,并判定路径列表设置失败。如果路径列表设置为空,file协议的可访问范围将遵循fileAccess规则,具体示例如下。

    import { webview } from '@kit.ArkWeb';
    import { BusinessError } from '@kit.BasicServicesKit';

    @Entry
    @Component
    struct WebComponent {
    controller: WebviewController = new webview.WebviewController();
    uiContext: UIContext = this.getUIContext();

    build() {
    Row() {
    Web({ src: '', controller: this.controller })
    .onControllerAttached(() => {
    try {
    // 设置允许可以跨域访问的路径列表
    this.controller.setPathAllowingUniversalAccess([
    this.uiContext.getHostContext()!.resourceDir,
    this.uiContext.getHostContext()!.filesDir + '/example'
    ]);
    this.controller.loadUrl('file://' + this.uiContext.getHostContext()!.resourceDir + '/index.html');
    } catch (error) {
    console.error(
    `ErrorCode: ${(error as BusinessError).code}, Message: ${(error as BusinessError).message}`);
    }
    })
    .javaScriptAccess(true)
    .fileAccess(true)
    .domStorageAccess(true)
    }
    }
    }
    <!-- main/resources/resfile/index.html -->
    <!DOCTYPE html>
    <html lang="en">

    <head>
    <meta charset="utf-8">
    <title>Demo</title>
    <meta name="viewport" content="width=device-width, initial-scale=1, user-scalable=no, viewport-fit=cover">
    <script>
    function getFile() {
    var file = "file:///data/storage/el1/bundle/entry/resources/resfile/js/script.js";
    // 使用file协议通过XMLHttpRequest跨域访问本地js文件。
    var xmlHttpReq = new XMLHttpRequest();
    xmlHttpReq.onreadystatechange = function(){
    console.info("readyState:" + xmlHttpReq.readyState);
    console.info("status:" + xmlHttpReq.status);
    if(xmlHttpReq.readyState == 4){
    if (xmlHttpReq.status == 200) {
    // 如果ets侧正确设置路径列表,则此处能正常获取资源
    const element = document.getElementById('text');
    element.textContent = "load " + file + " success";
    } else {
    // 如果ets侧不设置路径列表,则此处会触发CORS跨域检查错误
    const element = document.getElementById('text');
    element.textContent = "load " + file + " failed";
    }
    }
    }
    xmlHttpReq.open("GET", file);
    xmlHttpReq.send(null);
    }
    </script>
    </head>

    <body>
    <div class="page">
    <button id="example" onclick="getFile()">loadFile</button>
    </div>
    <div id="text"></div>
    </body>

    </html>
const body = document.body;
const element = document.createElement('div');
element.textContent = 'success';
body.appendChild(element);