@upstash/redis 0.0.0-ci.c37cc01e → 0.0.0-ci.c6073771-20221014
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm/deps/deno.land/x/base64@v0.2.1/base.js +100 -0
- package/esm/deps/deno.land/x/base64@v0.2.1/base64url.js +9 -0
- package/esm/deps/deno.land/x/sha1@v1.0.3/deps.js +1 -0
- package/esm/deps/deno.land/x/sha1@v1.0.3/mod.js +191 -0
- package/esm/deps/denopkg.com/chiefbiiko/std-encoding@v1.0.0/mod.js +50 -0
- package/esm/pkg/commands/scan.js +3 -0
- package/esm/pkg/commands/set.js +16 -4
- package/esm/pkg/commands/zmscore.js +10 -0
- package/esm/pkg/http.js +53 -2
- package/esm/pkg/pipeline.js +23 -5
- package/esm/pkg/redis.js +38 -1
- package/esm/pkg/script.js +77 -0
- package/package.json +1 -39
- package/script/deps/deno.land/x/base64@v0.2.1/base.js +104 -0
- package/script/deps/deno.land/x/base64@v0.2.1/base64url.js +13 -0
- package/script/deps/deno.land/x/sha1@v1.0.3/deps.js +6 -0
- package/script/deps/deno.land/x/sha1@v1.0.3/mod.js +196 -0
- package/script/deps/denopkg.com/chiefbiiko/std-encoding@v1.0.0/mod.js +55 -0
- package/script/pkg/commands/scan.js +3 -0
- package/script/pkg/commands/set.js +16 -4
- package/script/pkg/commands/zmscore.js +14 -0
- package/script/pkg/http.js +53 -2
- package/script/pkg/pipeline.js +23 -5
- package/script/pkg/redis.js +38 -1
- package/script/pkg/script.js +81 -0
- package/types/deps/deno.land/x/base64@v0.2.1/base.d.ts +5 -0
- package/types/deps/deno.land/x/base64@v0.2.1/base64url.d.ts +1 -0
- package/types/deps/deno.land/x/sha1@v1.0.3/deps.d.ts +1 -0
- package/types/deps/deno.land/x/sha1@v1.0.3/mod.d.ts +26 -0
- package/types/deps/denopkg.com/chiefbiiko/std-encoding@v1.0.0/mod.d.ts +3 -0
- package/types/pkg/commands/scan.d.ts +1 -0
- package/types/pkg/commands/set.d.ts +31 -2
- package/types/pkg/commands/zmscore.d.ts +7 -0
- package/types/pkg/pipeline.d.ts +12 -3
- package/types/pkg/redis.d.ts +17 -1
- package/types/pkg/script.d.ts +42 -0
|
@@ -0,0 +1,100 @@
|
|
|
1
|
+
function getLengths(b64) {
|
|
2
|
+
const len = b64.length;
|
|
3
|
+
// if (len % 4 > 0) {
|
|
4
|
+
// throw new TypeError("Invalid string. Length must be a multiple of 4");
|
|
5
|
+
// }
|
|
6
|
+
// Trim off extra bytes after placeholder bytes are found
|
|
7
|
+
// See: https://github.com/beatgammit/base64-js/issues/42
|
|
8
|
+
let validLen = b64.indexOf("=");
|
|
9
|
+
if (validLen === -1) {
|
|
10
|
+
validLen = len;
|
|
11
|
+
}
|
|
12
|
+
const placeHoldersLen = validLen === len ? 0 : 4 - (validLen % 4);
|
|
13
|
+
return [validLen, placeHoldersLen];
|
|
14
|
+
}
|
|
15
|
+
export function init(lookup, revLookup, urlsafe = false) {
|
|
16
|
+
function _byteLength(validLen, placeHoldersLen) {
|
|
17
|
+
return Math.floor(((validLen + placeHoldersLen) * 3) / 4 - placeHoldersLen);
|
|
18
|
+
}
|
|
19
|
+
function tripletToBase64(num) {
|
|
20
|
+
return (lookup[(num >> 18) & 0x3f] +
|
|
21
|
+
lookup[(num >> 12) & 0x3f] +
|
|
22
|
+
lookup[(num >> 6) & 0x3f] +
|
|
23
|
+
lookup[num & 0x3f]);
|
|
24
|
+
}
|
|
25
|
+
function encodeChunk(buf, start, end) {
|
|
26
|
+
const out = new Array((end - start) / 3);
|
|
27
|
+
for (let i = start, curTriplet = 0; i < end; i += 3) {
|
|
28
|
+
out[curTriplet++] = tripletToBase64((buf[i] << 16) + (buf[i + 1] << 8) + buf[i + 2]);
|
|
29
|
+
}
|
|
30
|
+
return out.join("");
|
|
31
|
+
}
|
|
32
|
+
return {
|
|
33
|
+
// base64 is 4/3 + up to two characters of the original data
|
|
34
|
+
byteLength(b64) {
|
|
35
|
+
return _byteLength.apply(null, getLengths(b64));
|
|
36
|
+
},
|
|
37
|
+
toUint8Array(b64) {
|
|
38
|
+
const [validLen, placeHoldersLen] = getLengths(b64);
|
|
39
|
+
const buf = new Uint8Array(_byteLength(validLen, placeHoldersLen));
|
|
40
|
+
// If there are placeholders, only get up to the last complete 4 chars
|
|
41
|
+
const len = placeHoldersLen ? validLen - 4 : validLen;
|
|
42
|
+
let tmp;
|
|
43
|
+
let curByte = 0;
|
|
44
|
+
let i;
|
|
45
|
+
for (i = 0; i < len; i += 4) {
|
|
46
|
+
tmp = (revLookup[b64.charCodeAt(i)] << 18) |
|
|
47
|
+
(revLookup[b64.charCodeAt(i + 1)] << 12) |
|
|
48
|
+
(revLookup[b64.charCodeAt(i + 2)] << 6) |
|
|
49
|
+
revLookup[b64.charCodeAt(i + 3)];
|
|
50
|
+
buf[curByte++] = (tmp >> 16) & 0xff;
|
|
51
|
+
buf[curByte++] = (tmp >> 8) & 0xff;
|
|
52
|
+
buf[curByte++] = tmp & 0xff;
|
|
53
|
+
}
|
|
54
|
+
if (placeHoldersLen === 2) {
|
|
55
|
+
tmp = (revLookup[b64.charCodeAt(i)] << 2) |
|
|
56
|
+
(revLookup[b64.charCodeAt(i + 1)] >> 4);
|
|
57
|
+
buf[curByte++] = tmp & 0xff;
|
|
58
|
+
}
|
|
59
|
+
else if (placeHoldersLen === 1) {
|
|
60
|
+
tmp = (revLookup[b64.charCodeAt(i)] << 10) |
|
|
61
|
+
(revLookup[b64.charCodeAt(i + 1)] << 4) |
|
|
62
|
+
(revLookup[b64.charCodeAt(i + 2)] >> 2);
|
|
63
|
+
buf[curByte++] = (tmp >> 8) & 0xff;
|
|
64
|
+
buf[curByte++] = tmp & 0xff;
|
|
65
|
+
}
|
|
66
|
+
return buf;
|
|
67
|
+
},
|
|
68
|
+
fromUint8Array(buf) {
|
|
69
|
+
const maxChunkLength = 16383; // Must be multiple of 3
|
|
70
|
+
const len = buf.length;
|
|
71
|
+
const extraBytes = len % 3; // If we have 1 byte left, pad 2 bytes
|
|
72
|
+
const len2 = len - extraBytes;
|
|
73
|
+
const parts = new Array(Math.ceil(len2 / maxChunkLength) + (extraBytes ? 1 : 0));
|
|
74
|
+
let curChunk = 0;
|
|
75
|
+
let chunkEnd;
|
|
76
|
+
// Go through the array every three bytes, we'll deal with trailing stuff later
|
|
77
|
+
for (let i = 0; i < len2; i += maxChunkLength) {
|
|
78
|
+
chunkEnd = i + maxChunkLength;
|
|
79
|
+
parts[curChunk++] = encodeChunk(buf, i, chunkEnd > len2 ? len2 : chunkEnd);
|
|
80
|
+
}
|
|
81
|
+
let tmp;
|
|
82
|
+
// Pad the end with zeros, but make sure to not forget the extra bytes
|
|
83
|
+
if (extraBytes === 1) {
|
|
84
|
+
tmp = buf[len2];
|
|
85
|
+
parts[curChunk] = lookup[tmp >> 2] + lookup[(tmp << 4) & 0x3f];
|
|
86
|
+
if (!urlsafe)
|
|
87
|
+
parts[curChunk] += "==";
|
|
88
|
+
}
|
|
89
|
+
else if (extraBytes === 2) {
|
|
90
|
+
tmp = (buf[len2] << 8) | (buf[len2 + 1] & 0xff);
|
|
91
|
+
parts[curChunk] = lookup[tmp >> 10] +
|
|
92
|
+
lookup[(tmp >> 4) & 0x3f] +
|
|
93
|
+
lookup[(tmp << 2) & 0x3f];
|
|
94
|
+
if (!urlsafe)
|
|
95
|
+
parts[curChunk] += "=";
|
|
96
|
+
}
|
|
97
|
+
return parts.join("");
|
|
98
|
+
},
|
|
99
|
+
};
|
|
100
|
+
}
|
|
@@ -0,0 +1,9 @@
|
|
|
1
|
+
import { init } from "./base.js";
|
|
2
|
+
const lookup = [];
|
|
3
|
+
const revLookup = [];
|
|
4
|
+
const code = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789-_";
|
|
5
|
+
for (let i = 0, l = code.length; i < l; ++i) {
|
|
6
|
+
lookup[i] = code[i];
|
|
7
|
+
revLookup[code.charCodeAt(i)] = i;
|
|
8
|
+
}
|
|
9
|
+
export const { byteLength, toUint8Array, fromUint8Array } = init(lookup, revLookup, true);
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export { encode, decode } from "../../../denopkg.com/chiefbiiko/std-encoding@v1.0.0/mod.js";
|
|
@@ -0,0 +1,191 @@
|
|
|
1
|
+
import { encode, decode } from "./deps.js";
|
|
2
|
+
function rotl(x, n) {
|
|
3
|
+
return (x << n) | (x >>> (32 - n));
|
|
4
|
+
}
|
|
5
|
+
/** Byte length of a SHA1 digest. */
|
|
6
|
+
export const BYTES = 20;
|
|
7
|
+
/** A class representation of the SHA1 algorithm. */
|
|
8
|
+
export class SHA1 {
|
|
9
|
+
/** Creates a SHA1 instance. */
|
|
10
|
+
constructor() {
|
|
11
|
+
Object.defineProperty(this, "hashSize", {
|
|
12
|
+
enumerable: true,
|
|
13
|
+
configurable: true,
|
|
14
|
+
writable: true,
|
|
15
|
+
value: BYTES
|
|
16
|
+
});
|
|
17
|
+
Object.defineProperty(this, "_buf", {
|
|
18
|
+
enumerable: true,
|
|
19
|
+
configurable: true,
|
|
20
|
+
writable: true,
|
|
21
|
+
value: new Uint8Array(64)
|
|
22
|
+
});
|
|
23
|
+
Object.defineProperty(this, "_bufIdx", {
|
|
24
|
+
enumerable: true,
|
|
25
|
+
configurable: true,
|
|
26
|
+
writable: true,
|
|
27
|
+
value: void 0
|
|
28
|
+
});
|
|
29
|
+
Object.defineProperty(this, "_count", {
|
|
30
|
+
enumerable: true,
|
|
31
|
+
configurable: true,
|
|
32
|
+
writable: true,
|
|
33
|
+
value: void 0
|
|
34
|
+
});
|
|
35
|
+
Object.defineProperty(this, "_K", {
|
|
36
|
+
enumerable: true,
|
|
37
|
+
configurable: true,
|
|
38
|
+
writable: true,
|
|
39
|
+
value: new Uint32Array([0x5a827999, 0x6ed9eba1, 0x8f1bbcdc, 0xca62c1d6])
|
|
40
|
+
});
|
|
41
|
+
Object.defineProperty(this, "_H", {
|
|
42
|
+
enumerable: true,
|
|
43
|
+
configurable: true,
|
|
44
|
+
writable: true,
|
|
45
|
+
value: void 0
|
|
46
|
+
});
|
|
47
|
+
Object.defineProperty(this, "_finalized", {
|
|
48
|
+
enumerable: true,
|
|
49
|
+
configurable: true,
|
|
50
|
+
writable: true,
|
|
51
|
+
value: void 0
|
|
52
|
+
});
|
|
53
|
+
this.init();
|
|
54
|
+
}
|
|
55
|
+
/** Reduces the four input numbers to a single one. */
|
|
56
|
+
static F(t, b, c, d) {
|
|
57
|
+
if (t <= 19) {
|
|
58
|
+
return (b & c) | (~b & d);
|
|
59
|
+
}
|
|
60
|
+
else if (t <= 39) {
|
|
61
|
+
return b ^ c ^ d;
|
|
62
|
+
}
|
|
63
|
+
else if (t <= 59) {
|
|
64
|
+
return (b & c) | (b & d) | (c & d);
|
|
65
|
+
}
|
|
66
|
+
else {
|
|
67
|
+
return b ^ c ^ d;
|
|
68
|
+
}
|
|
69
|
+
}
|
|
70
|
+
/** Initializes a hash instance. */
|
|
71
|
+
init() {
|
|
72
|
+
// prettier-ignore
|
|
73
|
+
this._H = new Uint32Array([
|
|
74
|
+
0x67452301, 0xEFCDAB89, 0x98BADCFE, 0x10325476, 0xC3D2E1F0
|
|
75
|
+
]);
|
|
76
|
+
this._bufIdx = 0;
|
|
77
|
+
this._count = new Uint32Array(2);
|
|
78
|
+
this._buf.fill(0);
|
|
79
|
+
this._finalized = false;
|
|
80
|
+
return this;
|
|
81
|
+
}
|
|
82
|
+
/** Updates a hash with additional message data. */
|
|
83
|
+
update(msg, inputEncoding) {
|
|
84
|
+
if (msg === null) {
|
|
85
|
+
throw new TypeError("msg must be a string or Uint8Array.");
|
|
86
|
+
}
|
|
87
|
+
else if (typeof msg === "string") {
|
|
88
|
+
msg = encode(msg, inputEncoding);
|
|
89
|
+
}
|
|
90
|
+
// process the msg as many times as possible, the rest is stored in the buffer
|
|
91
|
+
// message is processed in 512 bit (64 byte chunks)
|
|
92
|
+
for (let i = 0; i < msg.length; i++) {
|
|
93
|
+
this._buf[this._bufIdx++] = msg[i];
|
|
94
|
+
if (this._bufIdx === 64) {
|
|
95
|
+
this.transform();
|
|
96
|
+
this._bufIdx = 0;
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
// counter update (number of message bits)
|
|
100
|
+
const c = this._count;
|
|
101
|
+
if ((c[0] += msg.length << 3) < msg.length << 3) {
|
|
102
|
+
c[1]++;
|
|
103
|
+
}
|
|
104
|
+
c[1] += msg.length >>> 29;
|
|
105
|
+
return this;
|
|
106
|
+
}
|
|
107
|
+
/** Finalizes a hash with additional message data. */
|
|
108
|
+
digest(outputEncoding) {
|
|
109
|
+
if (this._finalized) {
|
|
110
|
+
throw new Error("digest has already been called.");
|
|
111
|
+
}
|
|
112
|
+
this._finalized = true;
|
|
113
|
+
// append '1'
|
|
114
|
+
const b = this._buf;
|
|
115
|
+
let idx = this._bufIdx;
|
|
116
|
+
b[idx++] = 0x80;
|
|
117
|
+
// zeropad up to byte pos 56
|
|
118
|
+
while (idx !== 56) {
|
|
119
|
+
if (idx === 64) {
|
|
120
|
+
this.transform();
|
|
121
|
+
idx = 0;
|
|
122
|
+
}
|
|
123
|
+
b[idx++] = 0;
|
|
124
|
+
}
|
|
125
|
+
// append length in bits
|
|
126
|
+
const c = this._count;
|
|
127
|
+
b[56] = (c[1] >>> 24) & 0xff;
|
|
128
|
+
b[57] = (c[1] >>> 16) & 0xff;
|
|
129
|
+
b[58] = (c[1] >>> 8) & 0xff;
|
|
130
|
+
b[59] = (c[1] >>> 0) & 0xff;
|
|
131
|
+
b[60] = (c[0] >>> 24) & 0xff;
|
|
132
|
+
b[61] = (c[0] >>> 16) & 0xff;
|
|
133
|
+
b[62] = (c[0] >>> 8) & 0xff;
|
|
134
|
+
b[63] = (c[0] >>> 0) & 0xff;
|
|
135
|
+
this.transform();
|
|
136
|
+
// return the hash as byte array (20 bytes)
|
|
137
|
+
const hash = new Uint8Array(BYTES);
|
|
138
|
+
for (let i = 0; i < 5; i++) {
|
|
139
|
+
hash[(i << 2) + 0] = (this._H[i] >>> 24) & 0xff;
|
|
140
|
+
hash[(i << 2) + 1] = (this._H[i] >>> 16) & 0xff;
|
|
141
|
+
hash[(i << 2) + 2] = (this._H[i] >>> 8) & 0xff;
|
|
142
|
+
hash[(i << 2) + 3] = (this._H[i] >>> 0) & 0xff;
|
|
143
|
+
}
|
|
144
|
+
// clear internal states and prepare for new hash
|
|
145
|
+
this.init();
|
|
146
|
+
return outputEncoding ? decode(hash, outputEncoding) : hash;
|
|
147
|
+
}
|
|
148
|
+
/** Performs one transformation cycle. */
|
|
149
|
+
transform() {
|
|
150
|
+
const h = this._H;
|
|
151
|
+
let a = h[0];
|
|
152
|
+
let b = h[1];
|
|
153
|
+
let c = h[2];
|
|
154
|
+
let d = h[3];
|
|
155
|
+
let e = h[4];
|
|
156
|
+
// convert byte buffer to words
|
|
157
|
+
const w = new Uint32Array(80);
|
|
158
|
+
for (let i = 0; i < 16; i++) {
|
|
159
|
+
w[i] =
|
|
160
|
+
this._buf[(i << 2) + 3] |
|
|
161
|
+
(this._buf[(i << 2) + 2] << 8) |
|
|
162
|
+
(this._buf[(i << 2) + 1] << 16) |
|
|
163
|
+
(this._buf[i << 2] << 24);
|
|
164
|
+
}
|
|
165
|
+
for (let t = 0; t < 80; t++) {
|
|
166
|
+
if (t >= 16) {
|
|
167
|
+
w[t] = rotl(w[t - 3] ^ w[t - 8] ^ w[t - 14] ^ w[t - 16], 1);
|
|
168
|
+
}
|
|
169
|
+
const tmp = (rotl(a, 5) +
|
|
170
|
+
SHA1.F(t, b, c, d) +
|
|
171
|
+
e +
|
|
172
|
+
w[t] +
|
|
173
|
+
this._K[Math.floor(t / 20)]) |
|
|
174
|
+
0;
|
|
175
|
+
e = d;
|
|
176
|
+
d = c;
|
|
177
|
+
c = rotl(b, 30);
|
|
178
|
+
b = a;
|
|
179
|
+
a = tmp;
|
|
180
|
+
}
|
|
181
|
+
h[0] = (h[0] + a) | 0;
|
|
182
|
+
h[1] = (h[1] + b) | 0;
|
|
183
|
+
h[2] = (h[2] + c) | 0;
|
|
184
|
+
h[3] = (h[3] + d) | 0;
|
|
185
|
+
h[4] = (h[4] + e) | 0;
|
|
186
|
+
}
|
|
187
|
+
}
|
|
188
|
+
/** Generates a SHA1 hash of the input data. */
|
|
189
|
+
export function sha1(msg, inputEncoding, outputEncoding) {
|
|
190
|
+
return new SHA1().update(msg, inputEncoding).digest(outputEncoding);
|
|
191
|
+
}
|
|
@@ -0,0 +1,50 @@
|
|
|
1
|
+
import { toUint8Array, fromUint8Array } from "../../../deno.land/x/base64@v0.2.1/base64url.js";
|
|
2
|
+
const decoder = new TextDecoder();
|
|
3
|
+
const encoder = new TextEncoder();
|
|
4
|
+
/** Serializes a Uint8Array to a hexadecimal string. */
|
|
5
|
+
function toHexString(buf) {
|
|
6
|
+
return buf.reduce((hex, byte) => `${hex}${byte < 16 ? "0" : ""}${byte.toString(16)}`, "");
|
|
7
|
+
}
|
|
8
|
+
/** Deserializes a Uint8Array from a hexadecimal string. */
|
|
9
|
+
function fromHexString(hex) {
|
|
10
|
+
const len = hex.length;
|
|
11
|
+
if (len % 2 || !/^[0-9a-fA-F]+$/.test(hex)) {
|
|
12
|
+
throw new TypeError("Invalid hex string.");
|
|
13
|
+
}
|
|
14
|
+
hex = hex.toLowerCase();
|
|
15
|
+
const buf = new Uint8Array(Math.floor(len / 2));
|
|
16
|
+
const end = len / 2;
|
|
17
|
+
for (let i = 0; i < end; ++i) {
|
|
18
|
+
buf[i] = parseInt(hex.substr(i * 2, 2), 16);
|
|
19
|
+
}
|
|
20
|
+
return buf;
|
|
21
|
+
}
|
|
22
|
+
/** Decodes a Uint8Array to utf8-, base64-, or hex-encoded string. */
|
|
23
|
+
export function decode(buf, encoding = "utf8") {
|
|
24
|
+
if (/^utf-?8$/i.test(encoding)) {
|
|
25
|
+
return decoder.decode(buf);
|
|
26
|
+
}
|
|
27
|
+
else if (/^base64$/i.test(encoding)) {
|
|
28
|
+
return fromUint8Array(buf);
|
|
29
|
+
}
|
|
30
|
+
else if (/^hex(?:adecimal)?$/i.test(encoding)) {
|
|
31
|
+
return toHexString(buf);
|
|
32
|
+
}
|
|
33
|
+
else {
|
|
34
|
+
throw new TypeError("Unsupported string encoding.");
|
|
35
|
+
}
|
|
36
|
+
}
|
|
37
|
+
export function encode(str, encoding = "utf8") {
|
|
38
|
+
if (/^utf-?8$/i.test(encoding)) {
|
|
39
|
+
return encoder.encode(str);
|
|
40
|
+
}
|
|
41
|
+
else if (/^base64$/i.test(encoding)) {
|
|
42
|
+
return toUint8Array(str);
|
|
43
|
+
}
|
|
44
|
+
else if (/^hex(?:adecimal)?$/i.test(encoding)) {
|
|
45
|
+
return fromHexString(str);
|
|
46
|
+
}
|
|
47
|
+
else {
|
|
48
|
+
throw new TypeError("Unsupported string encoding.");
|
|
49
|
+
}
|
|
50
|
+
}
|
package/esm/pkg/commands/scan.js
CHANGED
package/esm/pkg/commands/set.js
CHANGED
|
@@ -6,17 +6,29 @@ export class SetCommand extends Command {
|
|
|
6
6
|
constructor([key, value, opts], cmdOpts) {
|
|
7
7
|
const command = ["set", key, value];
|
|
8
8
|
if (opts) {
|
|
9
|
+
if ("nx" in opts && opts.nx) {
|
|
10
|
+
command.push("nx");
|
|
11
|
+
}
|
|
12
|
+
else if ("xx" in opts && opts.xx) {
|
|
13
|
+
command.push("xx");
|
|
14
|
+
}
|
|
15
|
+
if ("get" in opts && opts.get) {
|
|
16
|
+
command.push("get");
|
|
17
|
+
}
|
|
9
18
|
if ("ex" in opts && typeof opts.ex === "number") {
|
|
10
19
|
command.push("ex", opts.ex);
|
|
11
20
|
}
|
|
12
21
|
else if ("px" in opts && typeof opts.px === "number") {
|
|
13
22
|
command.push("px", opts.px);
|
|
14
23
|
}
|
|
15
|
-
if ("
|
|
16
|
-
command.push("
|
|
24
|
+
else if ("exat" in opts && typeof opts.exat === "number") {
|
|
25
|
+
command.push("exat", opts.exat);
|
|
17
26
|
}
|
|
18
|
-
else if ("
|
|
19
|
-
command.push("
|
|
27
|
+
else if ("pxat" in opts && typeof opts.pxat === "number") {
|
|
28
|
+
command.push("pxat", opts.pxat);
|
|
29
|
+
}
|
|
30
|
+
else if ("keepTtl" in opts && opts.keepTtl) {
|
|
31
|
+
command.push("keepTtl", opts.keepTtl);
|
|
20
32
|
}
|
|
21
33
|
}
|
|
22
34
|
super(command, cmdOpts);
|
package/esm/pkg/http.js
CHANGED
|
@@ -26,7 +26,11 @@ export class HttpClient {
|
|
|
26
26
|
value: void 0
|
|
27
27
|
});
|
|
28
28
|
this.baseUrl = config.baseUrl.replace(/\/$/, "");
|
|
29
|
-
this.headers = {
|
|
29
|
+
this.headers = {
|
|
30
|
+
"Content-Type": "application/json",
|
|
31
|
+
"Upstash-Encoding": "base64",
|
|
32
|
+
...config.headers,
|
|
33
|
+
};
|
|
30
34
|
this.options = { backend: config.options?.backend };
|
|
31
35
|
if (typeof config?.retry === "boolean" && config?.retry === false) {
|
|
32
36
|
this.retry = {
|
|
@@ -72,6 +76,53 @@ export class HttpClient {
|
|
|
72
76
|
if (!res.ok) {
|
|
73
77
|
throw new UpstashError(body.error);
|
|
74
78
|
}
|
|
75
|
-
return body;
|
|
79
|
+
return Array.isArray(body) ? body.map(decode) : decode(body);
|
|
80
|
+
}
|
|
81
|
+
}
|
|
82
|
+
function base64decode(b64) {
|
|
83
|
+
let dec = "";
|
|
84
|
+
try {
|
|
85
|
+
dec = atob(b64).split("").map((c) => "%" + ("00" + c.charCodeAt(0).toString(16)).slice(-2)).join("");
|
|
86
|
+
}
|
|
87
|
+
catch (e) {
|
|
88
|
+
console.warn(`Unable to decode base64 [${dec}]: ${e.message}`);
|
|
89
|
+
return dec;
|
|
90
|
+
}
|
|
91
|
+
try {
|
|
92
|
+
return decodeURIComponent(dec);
|
|
93
|
+
}
|
|
94
|
+
catch (e) {
|
|
95
|
+
console.warn(`Unable to decode URI [${dec}]: ${e.message}`);
|
|
96
|
+
return dec;
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
function decode(raw) {
|
|
100
|
+
let result = undefined;
|
|
101
|
+
switch (typeof raw.result) {
|
|
102
|
+
case "undefined":
|
|
103
|
+
return raw;
|
|
104
|
+
case "number":
|
|
105
|
+
result = raw.result;
|
|
106
|
+
break;
|
|
107
|
+
case "object":
|
|
108
|
+
if (Array.isArray(raw.result)) {
|
|
109
|
+
result = raw.result.map((v) => typeof v === "string"
|
|
110
|
+
? base64decode(v)
|
|
111
|
+
: Array.isArray(v)
|
|
112
|
+
? v.map(base64decode)
|
|
113
|
+
: v);
|
|
114
|
+
}
|
|
115
|
+
else {
|
|
116
|
+
// If it's not an array it must be null
|
|
117
|
+
// Apparently null is an object in javascript
|
|
118
|
+
result = null;
|
|
119
|
+
}
|
|
120
|
+
break;
|
|
121
|
+
case "string":
|
|
122
|
+
result = raw.result === "OK" ? "OK" : base64decode(raw.result);
|
|
123
|
+
break;
|
|
124
|
+
default:
|
|
125
|
+
break;
|
|
76
126
|
}
|
|
127
|
+
return { result, error: raw.error };
|
|
77
128
|
}
|
package/esm/pkg/pipeline.js
CHANGED
|
@@ -1,5 +1,6 @@
|
|
|
1
1
|
import { AppendCommand, BitCountCommand, BitOpCommand, BitPosCommand, DBSizeCommand, DecrByCommand, DecrCommand, DelCommand, EchoCommand, EvalCommand, EvalshaCommand, ExistsCommand, ExpireAtCommand, ExpireCommand, FlushAllCommand, FlushDBCommand, GetBitCommand, GetCommand, GetRangeCommand, GetSetCommand, HDelCommand, HExistsCommand, HGetAllCommand, HGetCommand, HIncrByCommand, HIncrByFloatCommand, HKeysCommand, HLenCommand, HMGetCommand, HMSetCommand, HScanCommand, HSetCommand, HSetNXCommand, HStrLenCommand, HValsCommand, IncrByCommand, IncrByFloatCommand, IncrCommand, KeysCommand, LIndexCommand, LInsertCommand, LLenCommand, LPopCommand, LPosCommand, LPushCommand, LPushXCommand, LRangeCommand, LRemCommand, LSetCommand, LTrimCommand, MGetCommand, MSetCommand, MSetNXCommand, PersistCommand, PExpireAtCommand, PExpireCommand, PingCommand, PSetEXCommand, PTtlCommand, PublishCommand, RandomKeyCommand, RenameCommand, RenameNXCommand, RPopCommand, RPushCommand, RPushXCommand, SAddCommand, ScanCommand, SCardCommand, ScriptExistsCommand, ScriptFlushCommand, ScriptLoadCommand, SDiffCommand, SDiffStoreCommand, SetBitCommand, SetCommand, SetExCommand, SetNxCommand, SetRangeCommand, SInterCommand, SInterStoreCommand, SIsMemberCommand, SMembersCommand, SMoveCommand, SPopCommand, SRandMemberCommand, SRemCommand, SScanCommand, StrLenCommand, SUnionCommand, SUnionStoreCommand, TimeCommand, TouchCommand, TtlCommand, TypeCommand, UnlinkCommand, ZAddCommand, ZCardCommand, ZCountCommand, ZIncrByCommand, ZInterStoreCommand, ZLexCountCommand, ZPopMaxCommand, ZPopMinCommand, ZRangeCommand, ZRankCommand, ZRemCommand, ZRemRangeByLexCommand, ZRemRangeByRankCommand, ZRemRangeByScoreCommand, ZRevRankCommand, ZScanCommand, ZScoreCommand, ZUnionStoreCommand, } from "./commands/mod.js";
|
|
2
2
|
import { UpstashError } from "./error.js";
|
|
3
|
+
import { ZMScoreCommand } from "./commands/zmscore.js";
|
|
3
4
|
/**
|
|
4
5
|
* Upstash REST API supports command pipelining to send multiple commands in
|
|
5
6
|
* batch, instead of sending each command one by one and waiting for a response.
|
|
@@ -16,7 +17,7 @@ import { UpstashError } from "./error.js";
|
|
|
16
17
|
* **Examples:**
|
|
17
18
|
*
|
|
18
19
|
* ```ts
|
|
19
|
-
* const p = redis.pipeline()
|
|
20
|
+
* const p = redis.pipeline() // or redis.multi()
|
|
20
21
|
* p.set("key","value")
|
|
21
22
|
* p.get("key")
|
|
22
23
|
* const res = await p.exec()
|
|
@@ -39,7 +40,7 @@ import { UpstashError } from "./error.js";
|
|
|
39
40
|
* ```
|
|
40
41
|
*/
|
|
41
42
|
export class Pipeline {
|
|
42
|
-
constructor(
|
|
43
|
+
constructor(opts) {
|
|
43
44
|
Object.defineProperty(this, "client", {
|
|
44
45
|
enumerable: true,
|
|
45
46
|
configurable: true,
|
|
@@ -58,6 +59,12 @@ export class Pipeline {
|
|
|
58
59
|
writable: true,
|
|
59
60
|
value: void 0
|
|
60
61
|
});
|
|
62
|
+
Object.defineProperty(this, "multiExec", {
|
|
63
|
+
enumerable: true,
|
|
64
|
+
configurable: true,
|
|
65
|
+
writable: true,
|
|
66
|
+
value: void 0
|
|
67
|
+
});
|
|
61
68
|
/**
|
|
62
69
|
* Send the pipeline request to upstash.
|
|
63
70
|
*
|
|
@@ -76,8 +83,9 @@ export class Pipeline {
|
|
|
76
83
|
if (this.commands.length === 0) {
|
|
77
84
|
throw new Error("Pipeline is empty");
|
|
78
85
|
}
|
|
86
|
+
const path = this.multiExec ? ["multi-exec"] : ["pipeline"];
|
|
79
87
|
const res = (await this.client.request({
|
|
80
|
-
path
|
|
88
|
+
path,
|
|
81
89
|
body: Object.values(this.commands).map((c) => c.command),
|
|
82
90
|
}));
|
|
83
91
|
return res.map(({ error, result }, i) => {
|
|
@@ -1011,6 +1019,15 @@ export class Pipeline {
|
|
|
1011
1019
|
writable: true,
|
|
1012
1020
|
value: (...args) => this.chain(new ZLexCountCommand(args, this.commandOptions))
|
|
1013
1021
|
});
|
|
1022
|
+
/**
|
|
1023
|
+
* @see https://redis.io/commands/zmscore
|
|
1024
|
+
*/
|
|
1025
|
+
Object.defineProperty(this, "zmscore", {
|
|
1026
|
+
enumerable: true,
|
|
1027
|
+
configurable: true,
|
|
1028
|
+
writable: true,
|
|
1029
|
+
value: (...args) => this.chain(new ZMScoreCommand(args, this.commandOptions))
|
|
1030
|
+
});
|
|
1014
1031
|
/**
|
|
1015
1032
|
* @see https://redis.io/commands/zpopmax
|
|
1016
1033
|
*/
|
|
@@ -1119,9 +1136,10 @@ export class Pipeline {
|
|
|
1119
1136
|
writable: true,
|
|
1120
1137
|
value: (...args) => this.chain(new ZUnionStoreCommand(args, this.commandOptions))
|
|
1121
1138
|
});
|
|
1122
|
-
this.client = client;
|
|
1139
|
+
this.client = opts.client;
|
|
1123
1140
|
this.commands = [];
|
|
1124
|
-
this.commandOptions = commandOptions;
|
|
1141
|
+
this.commandOptions = opts.commandOptions;
|
|
1142
|
+
this.multiExec = opts.multiExec ?? false;
|
|
1125
1143
|
}
|
|
1126
1144
|
/**
|
|
1127
1145
|
* Pushes a command into the pipelien and returns a chainable instance of the
|
package/esm/pkg/redis.js
CHANGED
|
@@ -1,5 +1,7 @@
|
|
|
1
1
|
import { AppendCommand, BitCountCommand, BitOpCommand, BitPosCommand, DBSizeCommand, DecrByCommand, DecrCommand, DelCommand, EchoCommand, EvalCommand, EvalshaCommand, ExistsCommand, ExpireAtCommand, ExpireCommand, FlushAllCommand, FlushDBCommand, GetBitCommand, GetCommand, GetRangeCommand, GetSetCommand, HDelCommand, HExistsCommand, HGetAllCommand, HGetCommand, HIncrByCommand, HIncrByFloatCommand, HKeysCommand, HLenCommand, HMGetCommand, HMSetCommand, HScanCommand, HSetCommand, HSetNXCommand, HStrLenCommand, HValsCommand, IncrByCommand, IncrByFloatCommand, IncrCommand, KeysCommand, LIndexCommand, LInsertCommand, LLenCommand, LPopCommand, LPosCommand, LPushCommand, LPushXCommand, LRangeCommand, LRemCommand, LSetCommand, LTrimCommand, MGetCommand, MSetCommand, MSetNXCommand, PersistCommand, PExpireAtCommand, PExpireCommand, PingCommand, PSetEXCommand, PTtlCommand, PublishCommand, RandomKeyCommand, RenameCommand, RenameNXCommand, RPopCommand, RPushCommand, RPushXCommand, SAddCommand, ScanCommand, SCardCommand, ScriptExistsCommand, ScriptFlushCommand, ScriptLoadCommand, SDiffCommand, SDiffStoreCommand, SetBitCommand, SetCommand, SetExCommand, SetNxCommand, SetRangeCommand, SInterCommand, SInterStoreCommand, SIsMemberCommand, SMembersCommand, SMoveCommand, SPopCommand, SRandMemberCommand, SRemCommand, SScanCommand, StrLenCommand, SUnionCommand, SUnionStoreCommand, TimeCommand, TouchCommand, TtlCommand, TypeCommand, UnlinkCommand, ZAddCommand, ZCardCommand, ZCountCommand, ZIncrByCommand, ZInterStoreCommand, ZLexCountCommand, ZPopMaxCommand, ZPopMinCommand, ZRangeCommand, ZRankCommand, ZRemCommand, ZRemRangeByLexCommand, ZRemRangeByRankCommand, ZRemRangeByScoreCommand, ZRevRankCommand, ZScanCommand, ZScoreCommand, ZUnionStoreCommand, } from "./commands/mod.js";
|
|
2
2
|
import { Pipeline } from "./pipeline.js";
|
|
3
|
+
import { Script } from "./script.js";
|
|
4
|
+
import { ZMScoreCommand } from "./commands/zmscore.js";
|
|
3
5
|
/**
|
|
4
6
|
* Serverless redis client for upstash.
|
|
5
7
|
*/
|
|
@@ -49,7 +51,30 @@ export class Redis {
|
|
|
49
51
|
enumerable: true,
|
|
50
52
|
configurable: true,
|
|
51
53
|
writable: true,
|
|
52
|
-
value: () => new Pipeline(
|
|
54
|
+
value: () => new Pipeline({
|
|
55
|
+
client: this.client,
|
|
56
|
+
commandOptions: this.opts,
|
|
57
|
+
multiExec: false,
|
|
58
|
+
})
|
|
59
|
+
});
|
|
60
|
+
/**
|
|
61
|
+
* Create a new transaction to allow executing multiple steps atomically.
|
|
62
|
+
*
|
|
63
|
+
* All the commands in a transaction are serialized and executed sequentially. A request sent by
|
|
64
|
+
* another client will never be served in the middle of the execution of a Redis Transaction. This
|
|
65
|
+
* guarantees that the commands are executed as a single isolated operation.
|
|
66
|
+
*
|
|
67
|
+
* @see {@link Pipeline}
|
|
68
|
+
*/
|
|
69
|
+
Object.defineProperty(this, "multi", {
|
|
70
|
+
enumerable: true,
|
|
71
|
+
configurable: true,
|
|
72
|
+
writable: true,
|
|
73
|
+
value: () => new Pipeline({
|
|
74
|
+
client: this.client,
|
|
75
|
+
commandOptions: this.opts,
|
|
76
|
+
multiExec: true,
|
|
77
|
+
})
|
|
53
78
|
});
|
|
54
79
|
/**
|
|
55
80
|
* @see https://redis.io/commands/append
|
|
@@ -974,6 +999,15 @@ export class Redis {
|
|
|
974
999
|
writable: true,
|
|
975
1000
|
value: (...args) => new ZLexCountCommand(args, this.opts).exec(this.client)
|
|
976
1001
|
});
|
|
1002
|
+
/**
|
|
1003
|
+
* @see https://redis.io/commands/zmscore
|
|
1004
|
+
*/
|
|
1005
|
+
Object.defineProperty(this, "zmscore", {
|
|
1006
|
+
enumerable: true,
|
|
1007
|
+
configurable: true,
|
|
1008
|
+
writable: true,
|
|
1009
|
+
value: (...args) => new ZMScoreCommand(args, this.opts).exec(this.client)
|
|
1010
|
+
});
|
|
977
1011
|
/**
|
|
978
1012
|
* @see https://redis.io/commands/zpopmax
|
|
979
1013
|
*/
|
|
@@ -1085,4 +1119,7 @@ export class Redis {
|
|
|
1085
1119
|
this.client = client;
|
|
1086
1120
|
this.opts = opts;
|
|
1087
1121
|
}
|
|
1122
|
+
createScript(script) {
|
|
1123
|
+
return new Script(this, script);
|
|
1124
|
+
}
|
|
1088
1125
|
}
|