@upstash/redis 0.0.0-ci.c37cc01e → 0.0.0-ci.c586e271-20221116

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.
Files changed (56) hide show
  1. package/README.md +0 -21
  2. package/esm/deps/deno.land/x/base64@v0.2.1/base.js +100 -0
  3. package/esm/deps/deno.land/x/base64@v0.2.1/base64url.js +9 -0
  4. package/esm/deps/deno.land/x/sha1@v1.0.3/deps.js +1 -0
  5. package/esm/deps/deno.land/x/sha1@v1.0.3/mod.js +191 -0
  6. package/esm/deps/denopkg.com/chiefbiiko/std-encoding@v1.0.0/mod.js +50 -0
  7. package/esm/pkg/commands/getdel.js +9 -0
  8. package/esm/pkg/commands/mod.js +1 -0
  9. package/esm/pkg/commands/scan.js +3 -0
  10. package/esm/pkg/commands/set.js +16 -4
  11. package/esm/pkg/commands/zmscore.js +10 -0
  12. package/esm/pkg/http.js +73 -2
  13. package/esm/pkg/pipeline.js +33 -6
  14. package/esm/pkg/redis.js +48 -2
  15. package/esm/pkg/script.js +77 -0
  16. package/esm/platforms/cloudflare.js +1 -0
  17. package/esm/platforms/fastly.js +1 -0
  18. package/esm/platforms/node_with_fetch.js +2 -1
  19. package/esm/platforms/nodejs.js +2 -1
  20. package/package.json +1 -39
  21. package/script/deps/deno.land/x/base64@v0.2.1/base.js +104 -0
  22. package/script/deps/deno.land/x/base64@v0.2.1/base64url.js +13 -0
  23. package/script/deps/deno.land/x/sha1@v1.0.3/deps.js +6 -0
  24. package/script/deps/deno.land/x/sha1@v1.0.3/mod.js +196 -0
  25. package/script/deps/denopkg.com/chiefbiiko/std-encoding@v1.0.0/mod.js +55 -0
  26. package/script/pkg/commands/getdel.js +13 -0
  27. package/script/pkg/commands/mod.js +1 -0
  28. package/script/pkg/commands/scan.js +3 -0
  29. package/script/pkg/commands/set.js +16 -4
  30. package/script/pkg/commands/zmscore.js +14 -0
  31. package/script/pkg/http.js +73 -2
  32. package/script/pkg/pipeline.js +32 -5
  33. package/script/pkg/redis.js +47 -1
  34. package/script/pkg/script.js +81 -0
  35. package/script/platforms/cloudflare.js +1 -0
  36. package/script/platforms/fastly.js +1 -0
  37. package/script/platforms/node_with_fetch.js +2 -1
  38. package/script/platforms/nodejs.js +2 -1
  39. package/types/deps/deno.land/x/base64@v0.2.1/base.d.ts +5 -0
  40. package/types/deps/deno.land/x/base64@v0.2.1/base64url.d.ts +1 -0
  41. package/types/deps/deno.land/x/sha1@v1.0.3/deps.d.ts +1 -0
  42. package/types/deps/deno.land/x/sha1@v1.0.3/mod.d.ts +26 -0
  43. package/types/deps/denopkg.com/chiefbiiko/std-encoding@v1.0.0/mod.d.ts +3 -0
  44. package/types/pkg/commands/getdel.d.ts +7 -0
  45. package/types/pkg/commands/mod.d.ts +1 -0
  46. package/types/pkg/commands/scan.d.ts +1 -0
  47. package/types/pkg/commands/set.d.ts +31 -2
  48. package/types/pkg/commands/zmscore.d.ts +7 -0
  49. package/types/pkg/http.d.ts +35 -4
  50. package/types/pkg/pipeline.d.ts +16 -3
  51. package/types/pkg/redis.d.ts +21 -1
  52. package/types/pkg/script.d.ts +42 -0
  53. package/types/platforms/cloudflare.d.ts +3 -6
  54. package/types/platforms/fastly.d.ts +2 -6
  55. package/types/platforms/node_with_fetch.d.ts +2 -21
  56. package/types/platforms/nodejs.d.ts +3 -6
package/README.md CHANGED
@@ -81,24 +81,6 @@ data = await redis.spop('animals', 1)
81
81
  console.log(data)
82
82
  ```
83
83
 
84
- ### Upgrading to v1.4.0 **(ReferenceError: fetch is not defined)**
85
-
86
- If you are running on nodejs v17 and earlier, `fetch` will not be natively
87
- supported. Platforms like Vercel, Netlify, Deno, Fastly etc. provide a polyfill
88
- for you. But if you are running on bare node, you need to either specify a
89
- polyfill yourself or change the import path to:
90
-
91
- ```typescript
92
- import { Redis } from "@upstash/redis/with-fetch";
93
- ```
94
-
95
- ### Upgrading from v0.2.0?
96
-
97
- Please read the
98
- [migration guide](https://docs.upstash.com/redis/sdks/javascriptsdk/migration).
99
- For further explanation we wrote a
100
- [blog post](https://blog.upstash.com/upstash-redis-sdk-v1).
101
-
102
84
  ## Docs
103
85
 
104
86
  See [the documentation](https://docs.upstash.com/features/javascriptsdk) for
@@ -118,6 +100,3 @@ the url and token
118
100
  ```sh
119
101
  UPSTASH_REDIS_REST_URL=".." UPSTASH_REDIS_REST_TOKEN=".." deno test -A
120
102
  ```
121
-
122
- ```
123
- ```
@@ -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
+ }
@@ -0,0 +1,9 @@
1
+ import { Command } from "./command.js";
2
+ /**
3
+ * @see https://redis.io/commands/getdel
4
+ */
5
+ export class GetDelCommand extends Command {
6
+ constructor(cmd, opts) {
7
+ super(["getdel", ...cmd], opts);
8
+ }
9
+ }
@@ -19,6 +19,7 @@ export * from "./get.js";
19
19
  export * from "./getbit.js";
20
20
  export * from "./getrange.js";
21
21
  export * from "./getset.js";
22
+ export * from "./getdel.js";
22
23
  export * from "./hdel.js";
23
24
  export * from "./hexists.js";
24
25
  export * from "./hget.js";
@@ -11,6 +11,9 @@ export class ScanCommand extends Command {
11
11
  if (typeof opts?.count === "number") {
12
12
  command.push("count", opts.count);
13
13
  }
14
+ if (opts?.type && opts.type.length > 0) {
15
+ command.push("type", opts.type);
16
+ }
14
17
  super(command, cmdOpts);
15
18
  }
16
19
  }
@@ -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 ("nx" in opts && opts.nx) {
16
- command.push("nx");
24
+ else if ("exat" in opts && typeof opts.exat === "number") {
25
+ command.push("exat", opts.exat);
17
26
  }
18
- else if ("xx" in opts && opts.xx) {
19
- command.push("xx");
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);
@@ -0,0 +1,10 @@
1
+ import { Command } from "./command.js";
2
+ /**
3
+ * @see https://redis.io/commands/zmscore
4
+ */
5
+ export class ZMScoreCommand extends Command {
6
+ constructor(cmd, opts) {
7
+ const [key, members] = cmd;
8
+ super(["zmscore", key, ...members], opts);
9
+ }
10
+ }
package/esm/pkg/http.js CHANGED
@@ -25,9 +25,19 @@ export class HttpClient {
25
25
  writable: true,
26
26
  value: void 0
27
27
  });
28
+ this.options = {
29
+ backend: config.options?.backend,
30
+ agent: config.agent,
31
+ responseEncoding: config.responseEncoding ?? "base64", // default to base64
32
+ };
28
33
  this.baseUrl = config.baseUrl.replace(/\/$/, "");
29
- this.headers = { "Content-Type": "application/json", ...config.headers };
30
- this.options = { backend: config.options?.backend };
34
+ this.headers = {
35
+ "Content-Type": "application/json",
36
+ ...config.headers,
37
+ };
38
+ if (this.options.responseEncoding === "base64") {
39
+ this.headers["Upstash-Encoding"] = "base64";
40
+ }
31
41
  if (typeof config?.retry === "boolean" && config?.retry === false) {
32
42
  this.retry = {
33
43
  attempts: 1,
@@ -48,6 +58,7 @@ export class HttpClient {
48
58
  headers: this.headers,
49
59
  body: JSON.stringify(req.body),
50
60
  keepalive: true,
61
+ agent: this.options?.agent,
51
62
  /**
52
63
  * Fastly specific
53
64
  */
@@ -72,6 +83,66 @@ export class HttpClient {
72
83
  if (!res.ok) {
73
84
  throw new UpstashError(body.error);
74
85
  }
86
+ if (this.options?.responseEncoding === "base64") {
87
+ return Array.isArray(body) ? body.map(decode) : decode(body);
88
+ }
75
89
  return body;
76
90
  }
77
91
  }
92
+ function base64decode(b64) {
93
+ let dec = "";
94
+ try {
95
+ /**
96
+ * Using only atob() is not enough because it doesn't work with unicode characters
97
+ */
98
+ const binString = atob(b64);
99
+ const size = binString.length;
100
+ const bytes = new Uint8Array(size);
101
+ for (let i = 0; i < size; i++) {
102
+ bytes[i] = binString.charCodeAt(i);
103
+ }
104
+ dec = new TextDecoder().decode(bytes);
105
+ }
106
+ catch {
107
+ dec = b64;
108
+ }
109
+ try {
110
+ return decodeURIComponent(dec);
111
+ }
112
+ catch {
113
+ return dec;
114
+ }
115
+ }
116
+ function decode(raw) {
117
+ let result = undefined;
118
+ switch (typeof raw.result) {
119
+ case "undefined":
120
+ return raw;
121
+ case "number": {
122
+ result = raw.result;
123
+ break;
124
+ }
125
+ case "object": {
126
+ if (Array.isArray(raw.result)) {
127
+ result = raw.result.map((v) => typeof v === "string"
128
+ ? base64decode(v)
129
+ : Array.isArray(v)
130
+ ? v.map(base64decode)
131
+ : v);
132
+ }
133
+ else {
134
+ // If it's not an array it must be null
135
+ // Apparently null is an object in javascript
136
+ result = null;
137
+ }
138
+ break;
139
+ }
140
+ case "string": {
141
+ result = raw.result === "OK" ? "OK" : base64decode(raw.result);
142
+ break;
143
+ }
144
+ default:
145
+ break;
146
+ }
147
+ return { result, error: raw.error };
148
+ }
@@ -1,5 +1,6 @@
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";
1
+ import { AppendCommand, BitCountCommand, BitOpCommand, BitPosCommand, DBSizeCommand, DecrByCommand, DecrCommand, DelCommand, EchoCommand, EvalCommand, EvalshaCommand, ExistsCommand, ExpireAtCommand, ExpireCommand, FlushAllCommand, FlushDBCommand, GetBitCommand, GetCommand, GetDelCommand, 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(client, commandOptions) {
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: ["pipeline"],
88
+ path,
81
89
  body: Object.values(this.commands).map((c) => c.command),
82
90
  }));
83
91
  return res.map(({ error, result }, i) => {
@@ -250,6 +258,15 @@ export class Pipeline {
250
258
  writable: true,
251
259
  value: (...args) => this.chain(new GetBitCommand(args, this.commandOptions))
252
260
  });
261
+ /**
262
+ * @see https://redis.io/commands/getdel
263
+ */
264
+ Object.defineProperty(this, "getdel", {
265
+ enumerable: true,
266
+ configurable: true,
267
+ writable: true,
268
+ value: (...args) => this.chain(new GetDelCommand(args, this.commandOptions))
269
+ });
253
270
  /**
254
271
  * @see https://redis.io/commands/getrange
255
272
  */
@@ -1011,6 +1028,15 @@ export class Pipeline {
1011
1028
  writable: true,
1012
1029
  value: (...args) => this.chain(new ZLexCountCommand(args, this.commandOptions))
1013
1030
  });
1031
+ /**
1032
+ * @see https://redis.io/commands/zmscore
1033
+ */
1034
+ Object.defineProperty(this, "zmscore", {
1035
+ enumerable: true,
1036
+ configurable: true,
1037
+ writable: true,
1038
+ value: (...args) => this.chain(new ZMScoreCommand(args, this.commandOptions))
1039
+ });
1014
1040
  /**
1015
1041
  * @see https://redis.io/commands/zpopmax
1016
1042
  */
@@ -1119,9 +1145,10 @@ export class Pipeline {
1119
1145
  writable: true,
1120
1146
  value: (...args) => this.chain(new ZUnionStoreCommand(args, this.commandOptions))
1121
1147
  });
1122
- this.client = client;
1148
+ this.client = opts.client;
1123
1149
  this.commands = [];
1124
- this.commandOptions = commandOptions;
1150
+ this.commandOptions = opts.commandOptions;
1151
+ this.multiExec = opts.multiExec ?? false;
1125
1152
  }
1126
1153
  /**
1127
1154
  * Pushes a command into the pipelien and returns a chainable instance of the