tx · 8mZhCBvuWh1Eu1Ej8vzsBY3unMYWq3ikUP6VEnTbWCaa 3MwC75rhWzLpeEwLGSnF3ftHNh4fGBU48Ac: -0.03700000 Waves 2023.05.27 12:13 [2596170] smart account 3MwC75rhWzLpeEwLGSnF3ftHNh4fGBU48Ac > SELF 0.00000000 Waves
{ "type": 13, "id": "8mZhCBvuWh1Eu1Ej8vzsBY3unMYWq3ikUP6VEnTbWCaa", "fee": 3700000, "feeAssetId": null, "timestamp": 1685178829967, "version": 2, "chainId": 84, "sender": "3MwC75rhWzLpeEwLGSnF3ftHNh4fGBU48Ac", "senderPublicKey": "9yW1Jko9uspFMWxZL7MsaEnTH66KnoCqj69k36PKqZq3", "proofs": [ "3beR6juSUEY5SexAnZFaLUSMZ42hPfAE389DAsowFFetuTeVE6wkmR9icJ5Gast3k7wQKoFdeVVd6X1S21q3LwUz" ], "script": "base64:BgIQCAISAwoBCBIHCgUEBAQICA0ADWtfaW5pdGlhbGl6ZWQCDWtfaW5pdGlhbGl6ZWQAFGtfY29vcmRpbmF0b3JBZGRyZXNzAhRrX2Nvb3JkaW5hdG9yQWRkcmVzcwARa19zdGFraW5nX2FkZHJlc3MCEWtfc3Rha2luZ19hZGRyZXNzABFrX2Zhcm1pbmdfYWRkcmVzcwIRa19mYXJtaW5nX2FkZHJlc3MAD2tfdmF1bHRfYWRkcmVzcwIPa192YXVsdF9hZGRyZXNzAA9rX2FkbWluX2FkZHJlc3MCD2tfYWRtaW5fYWRkcmVzcwEJc3RyVG9MaXN0AQRfc3RyAwkAAAIFBF9zdHICAAUDbmlsCQC1CQIFBF9zdHICASwBC2Nvb3JkaW5hdG9yAAkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQCmCAEJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkAnQgCBQR0aGlzBRRrX2Nvb3JkaW5hdG9yQWRkcmVzcwIWQ29vcmRpbmF0b3Igbm90IHNldCBIQwIbQ29vcmRpbmF0b3IgYWRkcmVzcyBpbnZhbGlkAQxhZG1pbkFkZHJlc3MACQCmCAEJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkAnQgCCQELY29vcmRpbmF0b3IABQ9rX2FkbWluX2FkZHJlc3MCFUFkbWluIGFkZHJlc3Mgbm90IHNldAEOc3Rha2luZ0FkZHJlc3MACQETdmFsdWVPckVycm9yTWVzc2FnZQIJAKYIAQkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQCdCAIJAQtjb29yZGluYXRvcgAFEWtfc3Rha2luZ19hZGRyZXNzAhdTdGFraW5nIGFkZHJlc3Mgbm90IHNldAIXU3Rha2luZyBhZGRyZXNzIGludmFsaWQBDmZhcm1pbmdBZGRyZXNzAAkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQCmCAEJARN2YWx1ZU9yRXJyb3JNZXNzYWdlAgkAnQgCCQELY29vcmRpbmF0b3IABRFrX2Zhcm1pbmdfYWRkcmVzcwIXRmFybWluZyBhZGRyZXNzIG5vdCBzZXQCF0Zhcm1pbmcgYWRkcmVzcyBpbnZhbGlkAQx2YXVsdEFkZHJlc3MACQETdmFsdWVPckVycm9yTWVzc2FnZQIJAKYIAQkBE3ZhbHVlT3JFcnJvck1lc3NhZ2UCCQCdCAIJAQtjb29yZGluYXRvcgAFD2tfdmF1bHRfYWRkcmVzcwIVVmF1bHQgYWRkcmVzcyBub3Qgc2V0AhVWYXVsdCBhZGRyZXNzIGludmFsaWQBC2luaXRpYWxpemVkAAkBC3ZhbHVlT3JFbHNlAgkAmwgCBQR0aGlzBQ1rX2luaXRpYWxpemVkBwIBaQEKaW5pdGlhbGl6ZQEMX2Nvb3JkaW5hdG9yAwMJAQtpbml0aWFsaXplZAAGCQECIT0CCAUBaQZjYWxsZXIFBHRoaXMJAAIBAhRVbmFibGUgdG8gaW5pdGlhbGl6ZQkAzAgCCQELU3RyaW5nRW50cnkCBRRrX2Nvb3JkaW5hdG9yQWRkcmVzcwkApQgBCQERQGV4dHJOYXRpdmUoMTA2MikBBQxfY29vcmRpbmF0b3IJAMwIAgkBDEJvb2xlYW5FbnRyeQIFDWtfaW5pdGlhbGl6ZWQGBQNuaWwBaQETcGVyZm9ybUhvdXNla2VlcGluZwUGX3ZhdWx0CF9mYXJtaW5nCF9zdGFraW5nC19wYXlGdW5kaW5nC19zcG90VmF1bHRzBBNkb0Fja1Jld2FyZHNTdGFraW5nAwUIX3N0YWtpbmcJAPwHBAkBDnN0YWtpbmdBZGRyZXNzAAIKYWNrUmV3YXJkcwUDbmlsBQNuaWwFA25pbAMJAAACBRNkb0Fja1Jld2FyZHNTdGFraW5nBRNkb0Fja1Jld2FyZHNTdGFraW5nBBNkb0Fja1Jld2FyZHNGYXJtaW5nAwUIX2Zhcm1pbmcJAPwHBAkBDmZhcm1pbmdBZGRyZXNzAAIKYWNrUmV3YXJkcwUDbmlsBQNuaWwFA25pbAMJAAACBRNkb0Fja1Jld2FyZHNGYXJtaW5nBRNkb0Fja1Jld2FyZHNGYXJtaW5nBBFkb0Fja1Jld2FyZHNWYXVsdAMFBl92YXVsdAkA/AcECQEMdmF1bHRBZGRyZXNzAAIKYWNrUmV3YXJkcwUDbmlsBQNuaWwFA25pbAMJAAACBRFkb0Fja1Jld2FyZHNWYXVsdAURZG9BY2tSZXdhcmRzVmF1bHQEBGFtbXMJAQlzdHJUb0xpc3QBBQtfcGF5RnVuZGluZwoBB2NvbXB1dGUCBF9hY2MIX2FtbURhdGEEBXBhcnRzCQC1CQIFCF9hbW1EYXRhAgFABANhbW0JAJEDAgUFcGFydHMAAAQLcHJpY2VVcGRhdGUDCQBmAgkAkAMBBQVwYXJ0cwABCQCRAwIFBXBhcnRzAAECAAQMZG9QYXlGdW5kaW5nCQD8BwQJARFAZXh0ck5hdGl2ZSgxMDYyKQEFA2FtbQIKcGF5RnVuZGluZwkAzAgCBQtwcmljZVVwZGF0ZQUDbmlsBQNuaWwDCQAAAgUMZG9QYXlGdW5kaW5nBQxkb1BheUZ1bmRpbmcFBF9hY2MJAAIBAiRTdHJpY3QgdmFsdWUgaXMgbm90IGVxdWFsIHRvIGl0c2VsZi4ECmRvQ2FsbEFtbXMKAAIkbAUEYW1tcwoAAiRzCQCQAwEFAiRsCgAFJGFjYzAFA25pbAoBBSRmMF8xAgIkYQIkaQMJAGcCBQIkaQUCJHMFAiRhCQEHY29tcHV0ZQIFAiRhCQCRAwIFAiRsBQIkaQoBBSRmMF8yAgIkYQIkaQMJAGcCBQIkaQUCJHMFAiRhCQACAQITTGlzdCBzaXplIGV4Y2VlZHMgNQkBBSRmMF8yAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgkBBSRmMF8xAgUFJGFjYzAAAAABAAIAAwAEAAUDCQAAAgUKZG9DYWxsQW1tcwUKZG9DYWxsQW1tcwQGdmF1bHRzCQEJc3RyVG9MaXN0AQULX3Nwb3RWYXVsdHMKAQhjb21wdXRlMgIEX2FjYwZfdmF1bHQEDGRvUGF5RnVuZGluZwkA/AcECQERQGV4dHJOYXRpdmUoMTA2MikBBQZfdmF1bHQCCmFja1Jld2FyZHMFA25pbAUDbmlsAwkAAAIFDGRvUGF5RnVuZGluZwUMZG9QYXlGdW5kaW5nBQRfYWNjCQACAQIkU3RyaWN0IHZhbHVlIGlzIG5vdCBlcXVhbCB0byBpdHNlbGYuBAxkb0NhbGxWYXVsdHMKAAIkbAUGdmF1bHRzCgACJHMJAJADAQUCJGwKAAUkYWNjMAUDbmlsCgEFJGYxXzECAiRhAiRpAwkAZwIFAiRpBQIkcwUCJGEJAQhjb21wdXRlMgIFAiRhCQCRAwIFAiRsBQIkaQoBBSRmMV8yAgIkYQIkaQMJAGcCBQIkaQUCJHMFAiRhCQACAQITTGlzdCBzaXplIGV4Y2VlZHMgNQkBBSRmMV8yAgkBBSRmMV8xAgkBBSRmMV8xAgkBBSRmMV8xAgkBBSRmMV8xAgkBBSRmMV8xAgUFJGFjYzAAAAABAAIAAwAEAAUDCQAAAgUMZG9DYWxsVmF1bHRzBQxkb0NhbGxWYXVsdHMFA25pbAkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgkAAgECJFN0cmljdCB2YWx1ZSBpcyBub3QgZXF1YWwgdG8gaXRzZWxmLgECdHgBBnZlcmlmeQAEDmNvb3JkaW5hdG9yU3RyCQCdCAIFBHRoaXMFFGtfY29vcmRpbmF0b3JBZGRyZXNzAwkBCWlzRGVmaW5lZAEFDmNvb3JkaW5hdG9yU3RyBAVhZG1pbgkAnQgCCQERQGV4dHJOYXRpdmUoMTA2MikBCQEFdmFsdWUBBQ5jb29yZGluYXRvclN0cgUPa19hZG1pbl9hZGRyZXNzAwkBCWlzRGVmaW5lZAEFBWFkbWluCQELdmFsdWVPckVsc2UCCQCbCAIJARFAZXh0ck5hdGl2ZSgxMDYyKQEJAQV2YWx1ZQEFBWFkbWluCQCsAgIJAKwCAgkArAICAgdzdGF0dXNfCQClCAEFBHRoaXMCAV8JANgEAQgFAnR4AmlkBwkAAgECLnVuYWJsZSB0byB2ZXJpZnk6IGFkbWluIG5vdCBzZXQgaW4gY29vcmRpbmF0b3IJAPQDAwgFAnR4CWJvZHlCeXRlcwkAkQMCCAUCdHgGcHJvb2ZzAAAIBQJ0eA9zZW5kZXJQdWJsaWNLZXlawy1Y", "height": 2596170, "applicationStatus": "succeeded", "spentComplexity": 0 } View: original | compacted Prev: VnkVVfMnBqMde3mGUxfomWYhju3B5zVP3GJRaKzTNoX Next: 4JabVZVdZKTP36r9Y8YA5bTK8H3zoauHqRXUoKWJcDD6 Diff:
Old | New | Differences | |
---|---|---|---|
46 | 46 | ||
47 | 47 | ||
48 | 48 | @Callable(i) | |
49 | - | func performHousekeeping (_vault,_farming,_staking,_payFunding) = { | |
49 | + | func performHousekeeping (_vault,_farming,_staking,_payFunding,_spotVaults) = { | |
50 | 50 | let doAckRewardsStaking = if (_staking) | |
51 | 51 | then invoke(stakingAddress(), "ackRewards", nil, nil) | |
52 | 52 | else nil | |
75 | 75 | else throw("Strict value is not equal to itself.") | |
76 | 76 | } | |
77 | 77 | ||
78 | - | let $l = amms | |
79 | - | let $s = size($l) | |
80 | - | let $acc0 = nil | |
81 | - | func $f0_1 ($a,$i) = if (($i >= $s)) | |
82 | - | then $a | |
83 | - | else compute($a, $l[$i]) | |
78 | + | let doCallAmms = { | |
79 | + | let $l = amms | |
80 | + | let $s = size($l) | |
81 | + | let $acc0 = nil | |
82 | + | func $f0_1 ($a,$i) = if (($i >= $s)) | |
83 | + | then $a | |
84 | + | else compute($a, $l[$i]) | |
84 | 85 | ||
85 | - | func $f0_2 ($a,$i) = if (($i >= $s)) | |
86 | - | then $a | |
87 | - | else throw("List size exceeds | |
86 | + | func $f0_2 ($a,$i) = if (($i >= $s)) | |
87 | + | then $a | |
88 | + | else throw("List size exceeds 5") | |
88 | 89 | ||
89 | - | $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3), 4), 5), 6), 7), 8), 9), 10), 11), 12), 13), 14), 15), 16), 17), 18), 19), 20) | |
90 | + | $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3), 4), 5) | |
91 | + | } | |
92 | + | if ((doCallAmms == doCallAmms)) | |
93 | + | then { | |
94 | + | let vaults = strToList(_spotVaults) | |
95 | + | func compute2 (_acc,_vault) = { | |
96 | + | let doPayFunding = invoke(addressFromStringValue(_vault), "ackRewards", nil, nil) | |
97 | + | if ((doPayFunding == doPayFunding)) | |
98 | + | then _acc | |
99 | + | else throw("Strict value is not equal to itself.") | |
100 | + | } | |
101 | + | ||
102 | + | let doCallVaults = { | |
103 | + | let $l = vaults | |
104 | + | let $s = size($l) | |
105 | + | let $acc0 = nil | |
106 | + | func $f1_1 ($a,$i) = if (($i >= $s)) | |
107 | + | then $a | |
108 | + | else compute2($a, $l[$i]) | |
109 | + | ||
110 | + | func $f1_2 ($a,$i) = if (($i >= $s)) | |
111 | + | then $a | |
112 | + | else throw("List size exceeds 5") | |
113 | + | ||
114 | + | $f1_2($f1_1($f1_1($f1_1($f1_1($f1_1($acc0, 0), 1), 2), 3), 4), 5) | |
115 | + | } | |
116 | + | if ((doCallVaults == doCallVaults)) | |
117 | + | then nil | |
118 | + | else throw("Strict value is not equal to itself.") | |
119 | + | } | |
120 | + | else throw("Strict value is not equal to itself.") | |
90 | 121 | } | |
91 | 122 | else throw("Strict value is not equal to itself.") | |
92 | 123 | } |
Old | New | Differences | |
---|---|---|---|
1 | 1 | {-# STDLIB_VERSION 6 #-} | |
2 | 2 | {-# SCRIPT_TYPE ACCOUNT #-} | |
3 | 3 | {-# CONTENT_TYPE DAPP #-} | |
4 | 4 | let k_initialized = "k_initialized" | |
5 | 5 | ||
6 | 6 | let k_coordinatorAddress = "k_coordinatorAddress" | |
7 | 7 | ||
8 | 8 | let k_staking_address = "k_staking_address" | |
9 | 9 | ||
10 | 10 | let k_farming_address = "k_farming_address" | |
11 | 11 | ||
12 | 12 | let k_vault_address = "k_vault_address" | |
13 | 13 | ||
14 | 14 | let k_admin_address = "k_admin_address" | |
15 | 15 | ||
16 | 16 | func strToList (_str) = if ((_str == "")) | |
17 | 17 | then nil | |
18 | 18 | else split(_str, ",") | |
19 | 19 | ||
20 | 20 | ||
21 | 21 | func coordinator () = valueOrErrorMessage(addressFromString(valueOrErrorMessage(getString(this, k_coordinatorAddress), "Coordinator not set HC")), "Coordinator address invalid") | |
22 | 22 | ||
23 | 23 | ||
24 | 24 | func adminAddress () = addressFromString(valueOrErrorMessage(getString(coordinator(), k_admin_address), "Admin address not set")) | |
25 | 25 | ||
26 | 26 | ||
27 | 27 | func stakingAddress () = valueOrErrorMessage(addressFromString(valueOrErrorMessage(getString(coordinator(), k_staking_address), "Staking address not set")), "Staking address invalid") | |
28 | 28 | ||
29 | 29 | ||
30 | 30 | func farmingAddress () = valueOrErrorMessage(addressFromString(valueOrErrorMessage(getString(coordinator(), k_farming_address), "Farming address not set")), "Farming address invalid") | |
31 | 31 | ||
32 | 32 | ||
33 | 33 | func vaultAddress () = valueOrErrorMessage(addressFromString(valueOrErrorMessage(getString(coordinator(), k_vault_address), "Vault address not set")), "Vault address invalid") | |
34 | 34 | ||
35 | 35 | ||
36 | 36 | func initialized () = valueOrElse(getBoolean(this, k_initialized), false) | |
37 | 37 | ||
38 | 38 | ||
39 | 39 | @Callable(i) | |
40 | 40 | func initialize (_coordinator) = if (if (initialized()) | |
41 | 41 | then true | |
42 | 42 | else (i.caller != this)) | |
43 | 43 | then throw("Unable to initialize") | |
44 | 44 | else [StringEntry(k_coordinatorAddress, toString(addressFromStringValue(_coordinator))), BooleanEntry(k_initialized, true)] | |
45 | 45 | ||
46 | 46 | ||
47 | 47 | ||
48 | 48 | @Callable(i) | |
49 | - | func performHousekeeping (_vault,_farming,_staking,_payFunding) = { | |
49 | + | func performHousekeeping (_vault,_farming,_staking,_payFunding,_spotVaults) = { | |
50 | 50 | let doAckRewardsStaking = if (_staking) | |
51 | 51 | then invoke(stakingAddress(), "ackRewards", nil, nil) | |
52 | 52 | else nil | |
53 | 53 | if ((doAckRewardsStaking == doAckRewardsStaking)) | |
54 | 54 | then { | |
55 | 55 | let doAckRewardsFarming = if (_farming) | |
56 | 56 | then invoke(farmingAddress(), "ackRewards", nil, nil) | |
57 | 57 | else nil | |
58 | 58 | if ((doAckRewardsFarming == doAckRewardsFarming)) | |
59 | 59 | then { | |
60 | 60 | let doAckRewardsVault = if (_vault) | |
61 | 61 | then invoke(vaultAddress(), "ackRewards", nil, nil) | |
62 | 62 | else nil | |
63 | 63 | if ((doAckRewardsVault == doAckRewardsVault)) | |
64 | 64 | then { | |
65 | 65 | let amms = strToList(_payFunding) | |
66 | 66 | func compute (_acc,_ammData) = { | |
67 | 67 | let parts = split(_ammData, "@") | |
68 | 68 | let amm = parts[0] | |
69 | 69 | let priceUpdate = if ((size(parts) > 1)) | |
70 | 70 | then parts[1] | |
71 | 71 | else "" | |
72 | 72 | let doPayFunding = invoke(addressFromStringValue(amm), "payFunding", [priceUpdate], nil) | |
73 | 73 | if ((doPayFunding == doPayFunding)) | |
74 | 74 | then _acc | |
75 | 75 | else throw("Strict value is not equal to itself.") | |
76 | 76 | } | |
77 | 77 | ||
78 | - | let $l = amms | |
79 | - | let $s = size($l) | |
80 | - | let $acc0 = nil | |
81 | - | func $f0_1 ($a,$i) = if (($i >= $s)) | |
82 | - | then $a | |
83 | - | else compute($a, $l[$i]) | |
78 | + | let doCallAmms = { | |
79 | + | let $l = amms | |
80 | + | let $s = size($l) | |
81 | + | let $acc0 = nil | |
82 | + | func $f0_1 ($a,$i) = if (($i >= $s)) | |
83 | + | then $a | |
84 | + | else compute($a, $l[$i]) | |
84 | 85 | ||
85 | - | func $f0_2 ($a,$i) = if (($i >= $s)) | |
86 | - | then $a | |
87 | - | else throw("List size exceeds | |
86 | + | func $f0_2 ($a,$i) = if (($i >= $s)) | |
87 | + | then $a | |
88 | + | else throw("List size exceeds 5") | |
88 | 89 | ||
89 | - | $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3), 4), 5), 6), 7), 8), 9), 10), 11), 12), 13), 14), 15), 16), 17), 18), 19), 20) | |
90 | + | $f0_2($f0_1($f0_1($f0_1($f0_1($f0_1($acc0, 0), 1), 2), 3), 4), 5) | |
91 | + | } | |
92 | + | if ((doCallAmms == doCallAmms)) | |
93 | + | then { | |
94 | + | let vaults = strToList(_spotVaults) | |
95 | + | func compute2 (_acc,_vault) = { | |
96 | + | let doPayFunding = invoke(addressFromStringValue(_vault), "ackRewards", nil, nil) | |
97 | + | if ((doPayFunding == doPayFunding)) | |
98 | + | then _acc | |
99 | + | else throw("Strict value is not equal to itself.") | |
100 | + | } | |
101 | + | ||
102 | + | let doCallVaults = { | |
103 | + | let $l = vaults | |
104 | + | let $s = size($l) | |
105 | + | let $acc0 = nil | |
106 | + | func $f1_1 ($a,$i) = if (($i >= $s)) | |
107 | + | then $a | |
108 | + | else compute2($a, $l[$i]) | |
109 | + | ||
110 | + | func $f1_2 ($a,$i) = if (($i >= $s)) | |
111 | + | then $a | |
112 | + | else throw("List size exceeds 5") | |
113 | + | ||
114 | + | $f1_2($f1_1($f1_1($f1_1($f1_1($f1_1($acc0, 0), 1), 2), 3), 4), 5) | |
115 | + | } | |
116 | + | if ((doCallVaults == doCallVaults)) | |
117 | + | then nil | |
118 | + | else throw("Strict value is not equal to itself.") | |
119 | + | } | |
120 | + | else throw("Strict value is not equal to itself.") | |
90 | 121 | } | |
91 | 122 | else throw("Strict value is not equal to itself.") | |
92 | 123 | } | |
93 | 124 | else throw("Strict value is not equal to itself.") | |
94 | 125 | } | |
95 | 126 | else throw("Strict value is not equal to itself.") | |
96 | 127 | } | |
97 | 128 | ||
98 | 129 | ||
99 | 130 | @Verifier(tx) | |
100 | 131 | func verify () = { | |
101 | 132 | let coordinatorStr = getString(this, k_coordinatorAddress) | |
102 | 133 | if (isDefined(coordinatorStr)) | |
103 | 134 | then { | |
104 | 135 | let admin = getString(addressFromStringValue(value(coordinatorStr)), k_admin_address) | |
105 | 136 | if (isDefined(admin)) | |
106 | 137 | then valueOrElse(getBoolean(addressFromStringValue(value(admin)), ((("status_" + toString(this)) + "_") + toBase58String(tx.id))), false) | |
107 | 138 | else throw("unable to verify: admin not set in coordinator") | |
108 | 139 | } | |
109 | 140 | else sigVerify(tx.bodyBytes, tx.proofs[0], tx.senderPublicKey) | |
110 | 141 | } | |
111 | 142 |
github/deemru/w8io/169f3d6 41.25 ms ◑![]()