My asset canister does not work

I fill an asset canister by myself (not through DFX).

Then, when I try to access it, I get:

Body does not pass verification

What did I do wrongly?

I generated my asset canister by copying all the files from another asset canister:

    public func copyAll({from: Asset.AssetCanister; to: Asset.AssetCanister}): async* () {
        let fromAssets = await from.list({});
        let toAssets = await to.list({});
        let fromAssetsSet = HashMap.HashMap<Asset.Key, ()>(fromAssets.size(), Text.equal, Text.hash);
        for (x in fromAssets.vals()) {
            fromAssetsSet.put(x.key, ());
        };
        let toAssetsSet = HashMap.HashMap<Text, {
            key: Asset.Key;
            content_type: Text;
            encodings: [{
                content_encoding: Text;
                sha256: ?Blob; // sha256 of entire asset encoding, calculated by dfx and passed in SetAssetContentArguments
                length: Nat; // Size of this encoding's Blob. Calculated when uploading assets.
                modified: Time.Time;
            }];
        }>(toAssets.size(), Text.equal, Text.hash);
        for (x in toAssets.vals()) {
            toAssetsSet.put(x.key, x);
        };
        let { batch_id } = await to.create_batch({});
        let buf = Buffer.Buffer<Asset.BatchOperationKind>(0);
        for (toAsset in toAssets.vals()) {
            if (fromAssetsSet.get(toAsset.key) == null) {
                buf.add(#DeleteAsset {key = toAsset.key});
            };
        };
        for (fromAsset in fromAssets.vals()) {
            let props = await to.get_asset_properties(fromAsset.key);
            switch (toAssetsSet.get(fromAsset.key)) {
                case(?toAsset) {
                    // Remove missing encodings:
                    let fromEncodings = HashMap.HashMap<Asset.Key, ()>(0, Text.equal, Text.hash);
                    for (fromEncoding in fromAsset.encodings.vals()) {
                        fromEncodings.put(fromEncoding.content_encoding, ());
                    };
                    for (encoding in toAsset.encodings.vals()) {
                        if (fromEncodings.get(encoding.content_encoding) == null) {
                            buf.add(#UnsetAssetContent {
                                key = fromAsset.key;
                                content_encoding = encoding.content_encoding;
                            })
                        };
                    };

                    buf.add(#SetAssetProperties {
                        key = fromAsset.key;
                        max_age = ?props.max_age;
                        headers = ?props.headers;
                        allow_raw_access = ?props.allow_raw_access;
                        is_aliased = ?props.is_aliased;
                    });
                };
                case null {
                    buf.add(#CreateAsset {
                        allow_raw_access = props.allow_raw_access;
                        content_type = fromAsset.content_type;
                        enable_aliasing = props.is_aliased;
                        headers = props.headers;
                        key = fromAsset.key;
                        max_age = props.max_age;
                    });
                };
            };

            for (encoding in fromAsset.encodings.vals()) {
                let got = await from.get({key = fromAsset.key; accept_encodings = [encoding.content_encoding]});
                let chunksNum = if (got.total_length == 0) {
                    0;
                } else {
                    Int.abs((got.total_length: Int - 1) / got.content.size() + 1);
                };
                let chunkIds = Buffer.Buffer<Asset.ChunkId>(chunksNum);
                let chunk_info = await to.create_chunk({batch_id; content = got.content});
                chunkIds.add(chunk_info.chunk_id);
                for (i in Iter.range(1, chunksNum - 1)) {
                    let { content } = await from.get_chunk({
                        key = fromAsset.key;
                        content_encoding = encoding.content_encoding;
                        index = i;
                        sha256 = encoding.sha256; // sha256 of entire fromAsset encoding, calculated by dfx and passed in SetAssetContentArguments
                    });
                    let chunk_info = await to.create_chunk({batch_id; content});
                    chunkIds.add(chunk_info.chunk_id);
                };
                buf.add(#SetAssetContent {
                    key = fromAsset.key;
                    content_encoding = encoding.content_encoding;
                    chunk_ids = Buffer.toArray(chunkIds);
                    sha256 = null;
                });
            };
        };
        await to.commit_batch({batch_id; operations = Buffer.toArray(buf)});
    };

That’s a bit too little information to be able to debug easily. If you call http_request manually, you can see the full response. Can you compare the same call against the two canisters and post the difference you get? And if you can narrow it down to a more concrete example that would be great, too

@Severin

$ dfx canister call package_manager_frontend http_request '(record {url="/index.html";method="get";body=blob "";headers=vec {};certificate_version=null})'
(
  record {
    body = blob "<!DOCTYPE html>\0a<html lang=\22en\22>\0a\0a<head>\0a  <meta charset=\22UTF-8\22 />\0a  <meta name=\22viewport\22 content=\22width=device-width\22 />\0a  <title>IC Package Manager</title>\0a  <base href=\22/\22 />\0a  <link rel=\22icon\22 href=\22/favicon.ico\22 />\0a  <script type=\22module\22 crossorigin src=\22/assets/index-da1db52b.js\22></script>\0a  <link rel=\22stylesheet\22 href=\22/assets/index-bdbb3561.css\22>\0a</head>\0a\0a<body>\0a  <div id=\22root\22></div>\0a  \0a</body>\0a\0a</html>";
    headers = vec {
      record { "content-type"; "text/html" };
      record { "x-xss-protection"; "1; mode=block" };
      record { "x-content-type-options"; "nosniff" };
      record {
        "permissions-policy";
        "accelerometer=(), ambient-light-sensor=(), autoplay=(), battery=(), camera=(), cross-origin-isolated=(), display-capture=(), document-domain=(), encrypted-media=(), execution-while-not-rendered=(), execution-while-out-of-viewport=(), fullscreen=(), geolocation=(), gyroscope=(), keyboard-map=(), magnetometer=(), microphone=(), midi=(), navigation-override=(), payment=(), picture-in-picture=(), publickey-credentials-get=(), screen-wake-lock=(), sync-xhr=(), usb=(), web-share=(), xr-spatial-tracking=(), clipboard-read=(), clipboard-write=(), gamepad=(), speaker-selection=(), conversion-measurement=(), focus-without-user-activation=(), hid=(), idle-detection=(), interest-cohort=(), serial=(), sync-script=(), trust-token-redemption=(), window-placement=(), vertical-scroll=()";
      };
      record { "referrer-policy"; "same-origin" };
      record { "x-frame-options"; "DENY" };
      record {
        "strict-transport-security";
        "max-age=31536000; includeSubDomains";
      };
      record {
        "content-security-policy";
        "default-src \'self\';script-src \'self\' \'unsafe-eval\';connect-src \'self\' http://localhost:* https://icp0.io https://*.icp0.io https://icp-api.io;img-src \'self\' data:;style-src * \'unsafe-inline\';style-src-elem * \'unsafe-inline\';font-src *;object-src \'none\';base-uri \'self\';frame-ancestors \'none\';form-action \'self\';upgrade-insecure-requests;";
      };
      record {
        "etag";
        "\"c67b418e3ff363bf5e6aea7d2fa46e537a5ff97c2370432e58dd7480f75d4669\"";
      };
      record {
        "IC-Certificate";
        "certificate=:2dn3omR0cmVlgwGDAYMBggRYICZ9EIuCUUnIiSJ/IgyiAROM5iHDHHw88XQRfnAVEQu3gwJIY2FuaXN0ZXKDAYMBggRYIKhIcSSuB6FaNc2wPv4HVpnTQCQkcLgfKIYRLqXhvP99gwGDAYIEWCBX6ZYLyo9zBgn/7Kko/lMeHB4a6Qkj6Oro/s7DgPcVCoMBgwJKgAAAAAAQAAoBAYMBgwGDAk5jZXJ0aWZpZWRfZGF0YYIDWCA3bOwkmiZ40cOT7lw7IG4usAq15T4MJsjEJGxGBIZubIIEWCCOnhHU9ZZF6RKb3qZ4DgMAxygTzt4ynP7MZ9HAVan+i4IEWCA/hkZ4l93JAeiYaJ6pbHp7ZBVJWCZy2UYOgcJNxOtdY4IEWCAd89v70lvgVIwgRpEY0emqgDAbvjDwq4XhAz4yRWEjF4IEWCAyzvlN4m72KEs3LHOtloohnSZCBItUCJbXl+7+NfzK6IIEWCAz83/pZXn85mBSglKnCxy+jWSpGd9av3pkQFDWk4+LOIIEWCDQ+ojmgcDgVsZaAOjEF3B8G6VvOWSki5OJMcdlJOHIaoMBggRYIAyGWGh/0sPo6G17QCBD/N/5vkkTp4KGCgC4YECQdkUugwJEdGltZYIDSfyp5IPN0LD8F2lzaWduYXR1cmVYMJIiaMgb1vWyUmnRGgAvbmy6nEd6wu7F13mRQCTfMG26teFNBZcDjylPbwiPlNZerA==:, tree=:2dn3gwGDAktodHRwX2Fzc2V0c4MBggRYIHXKYKyE/22fSRms7oQ3N9a6PiSAyhEMEwOQUBjiffONgwGCBFgg8MZ1I05dnzgHrJy2p2j+ZkDa0tEV8L6NlWwEpxAtqFWDAYIEWCBhwbMZzLO26KgRwZzb0BFbSXn7ZMWFL30m/QkMAnXcg4MBgwJLL2luZGV4Lmh0bWyCA1ggxntBjj/zY79eaup9L6RuU3pf+XwjcEMuWN10gPddRmmCBFggl6KoLVBlz9PPmfh+9LwAz4pLiIKKofAQ1pu61MK3nImCBFggux8UtVTRkFP/PTn2fZ++2lHmHQAdKJw9DYXpFTQdk2U=:";
      };
    };
    streaming_strategy = null;
    status_code = 200 : nat16;
  },
)
$ dfx canister call dlbnd-beaaa-aaaaa-qaana-cai http_request '(record {url="/index.html";method="get";body=blob "";headers=vec {};certificate_version=null})'
(
  record {
    body = blob "not found";
    headers = vec {
      record { "content-type"; "text/plain" };
      record {
        "IC-Certificate";
        "certificate=:2dn3omR0cmVlgwGDAYMBggRYICZ9EIuCUUnIiSJ/IgyiAROM5iHDHHw88XQRfnAVEQu3gwJIY2FuaXN0ZXKDAYIEWCAyh3BFlKwvxx6gXFBNgCDWo6kt6T1PGiRshiP/k3q7q4MBggRYIGt0bCaLfJiLKiZfW60twdEt6+va4qZCAan+EZqru1tFgwGCBFggeCtx1MGqvDkxeslx5QNCKGXllZf8B+Q9KYNph9vUNC+DAkqAAAAAABAAGgEBgwGDAYMCTmNlcnRpZmllZF9kYXRhggNAggRYICya7/PhX0UVnltibYJxKxTfRkPdA8D384uyZkXjjYYlggRYID+GRniX3ckB6JhonqlsentkFUlYJnLZRg6Bwk3E611jggRYIPbvBToUhSFH0k3DSyn4c76khLNYjlayUPDJnARxWUI5gwGCBFggDIZYaH/Sw+jobXtAIEP83/m+SROngoYKALhgQJB2RS6DAkR0aW1lggNJ5qTWkMrRsPwXaXNpZ25hdHVyZVgwi/W/s6N9g2lIkYfMDuHzTuPlRqVcAH07VLLLrHOO1Ru9YoeR9xO76mZIzs2cUL+X:, tree=:2dn3gQA=:";
      };
    };
    streaming_strategy = null;
    status_code = 404 : nat16;
  },
)

So, the original assets returned a document, while my copied assets returned error 404.

What may I have done wrong?

I think, I know the error: I forgot to set assets canister permissions.

BTW, what is #Prepare permission?

Sounds like the file is not properly sent over. Maybe list is helpful to debug? I don’t have time to dive in your code at the moment

No, this is probably not the error: I properly set controllers = ?[Principal.fromActor(indirectCaller)];.

I’ve made the error to be displayed:

Error from Canister fvpkz-24aaa-aaaaa-qaazq-cai: Canister called ic0.trap with message: asset not found.

What my this error mean?

I messed to and from:

let props = await to.get_asset_properties(fromAsset.key);

should be instead

let props = await from.get_asset_properties(fromAsset.key);

After this it works.