@mtcute/node 0.16.6 → 0.16.9

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 (125) hide show
  1. package/chunks/cjs/BpvQ752Q.js +33 -0
  2. package/chunks/cjs/D7i-4e0W.js +74 -0
  3. package/chunks/cjs/HP2yqAk_.js +79 -0
  4. package/chunks/cjs/package.json +3 -0
  5. package/chunks/es/CKso6cAV.js +75 -0
  6. package/chunks/es/CnOjjhdK.js +29 -0
  7. package/chunks/es/HZgHrOPU.js +69 -0
  8. package/{cjs/client.d.ts → client.d.ts} +3 -2
  9. package/index.cjs +449 -0
  10. package/index.d.ts +8 -1
  11. package/index.js +410 -1
  12. package/{cjs/methods → methods}/download-node-stream.d.ts +1 -1
  13. package/methods.cjs +17 -0
  14. package/methods.d.ts +3 -1
  15. package/methods.js +6 -1
  16. package/package.json +49 -33
  17. package/{esm/sqlite → sqlite}/driver.d.ts +1 -1
  18. package/sqlite/sqlite.test.d.ts +1 -0
  19. package/{cjs/utils → utils}/crypto.d.ts +1 -1
  20. package/utils/crypto.test.d.ts +1 -0
  21. package/utils/normalize-file.d.ts +6 -0
  22. package/{cjs/utils → utils}/stream-utils.d.ts +1 -1
  23. package/utils/stream-utils.test.d.ts +1 -0
  24. package/{cjs/utils → utils}/tcp.d.ts +3 -3
  25. package/utils/tcp.test.d.ts +1 -0
  26. package/utils.cjs +20 -0
  27. package/utils.d.ts +3 -1
  28. package/utils.js +9 -1
  29. package/{cjs/worker.d.ts → worker.d.ts} +1 -1
  30. package/cjs/client.js +0 -113
  31. package/cjs/client.js.map +0 -1
  32. package/cjs/common-internals-node/exit-hook.js +0 -47
  33. package/cjs/common-internals-node/exit-hook.js.map +0 -1
  34. package/cjs/common-internals-node/logging.js +0 -34
  35. package/cjs/common-internals-node/logging.js.map +0 -1
  36. package/cjs/common-internals-node/platform.js +0 -85
  37. package/cjs/common-internals-node/platform.js.map +0 -1
  38. package/cjs/index.js +0 -31
  39. package/cjs/index.js.map +0 -1
  40. package/cjs/methods/download-file.js +0 -34
  41. package/cjs/methods/download-file.js.map +0 -1
  42. package/cjs/methods/download-node-stream.js +0 -15
  43. package/cjs/methods/download-node-stream.js.map +0 -1
  44. package/cjs/methods.js +0 -29
  45. package/cjs/methods.js.map +0 -1
  46. package/cjs/package.json +0 -3
  47. package/cjs/sqlite/driver.d.ts +0 -25
  48. package/cjs/sqlite/driver.js +0 -29
  49. package/cjs/sqlite/driver.js.map +0 -1
  50. package/cjs/sqlite/index.js +0 -18
  51. package/cjs/sqlite/index.js.map +0 -1
  52. package/cjs/utils/crypto.js +0 -73
  53. package/cjs/utils/crypto.js.map +0 -1
  54. package/cjs/utils/normalize-file.d.ts +0 -10
  55. package/cjs/utils/normalize-file.js +0 -31
  56. package/cjs/utils/normalize-file.js.map +0 -1
  57. package/cjs/utils/stream-utils.js +0 -77
  58. package/cjs/utils/stream-utils.js.map +0 -1
  59. package/cjs/utils/tcp.js +0 -126
  60. package/cjs/utils/tcp.js.map +0 -1
  61. package/cjs/utils/version.js +0 -21
  62. package/cjs/utils/version.js.map +0 -1
  63. package/cjs/utils.js +0 -26
  64. package/cjs/utils.js.map +0 -1
  65. package/cjs/worker.js +0 -48
  66. package/cjs/worker.js.map +0 -1
  67. package/esm/client.d.ts +0 -46
  68. package/esm/client.js +0 -108
  69. package/esm/client.js.map +0 -1
  70. package/esm/common-internals-node/exit-hook.d.ts +0 -1
  71. package/esm/common-internals-node/exit-hook.js +0 -43
  72. package/esm/common-internals-node/exit-hook.js.map +0 -1
  73. package/esm/common-internals-node/logging.d.ts +0 -2
  74. package/esm/common-internals-node/logging.js +0 -31
  75. package/esm/common-internals-node/logging.js.map +0 -1
  76. package/esm/common-internals-node/platform.d.ts +0 -18
  77. package/esm/common-internals-node/platform.js +0 -58
  78. package/esm/common-internals-node/platform.js.map +0 -1
  79. package/esm/index.d.ts +0 -8
  80. package/esm/index.js +0 -9
  81. package/esm/index.js.map +0 -1
  82. package/esm/methods/download-file.d.ts +0 -9
  83. package/esm/methods/download-file.js +0 -30
  84. package/esm/methods/download-file.js.map +0 -1
  85. package/esm/methods/download-node-stream.d.ts +0 -8
  86. package/esm/methods/download-node-stream.js +0 -11
  87. package/esm/methods/download-node-stream.js.map +0 -1
  88. package/esm/methods.d.ts +0 -3
  89. package/esm/methods.js +0 -4
  90. package/esm/methods.js.map +0 -1
  91. package/esm/sqlite/driver.js +0 -22
  92. package/esm/sqlite/driver.js.map +0 -1
  93. package/esm/sqlite/index.d.ts +0 -8
  94. package/esm/sqlite/index.js +0 -13
  95. package/esm/sqlite/index.js.map +0 -1
  96. package/esm/utils/crypto.d.ts +0 -15
  97. package/esm/utils/crypto.js +0 -71
  98. package/esm/utils/crypto.js.map +0 -1
  99. package/esm/utils/normalize-file.d.ts +0 -10
  100. package/esm/utils/normalize-file.js +0 -27
  101. package/esm/utils/normalize-file.js.map +0 -1
  102. package/esm/utils/stream-utils.d.ts +0 -3
  103. package/esm/utils/stream-utils.js +0 -72
  104. package/esm/utils/stream-utils.js.map +0 -1
  105. package/esm/utils/tcp.d.ts +0 -29
  106. package/esm/utils/tcp.js +0 -118
  107. package/esm/utils/tcp.js.map +0 -1
  108. package/esm/utils/version.d.ts +0 -3
  109. package/esm/utils/version.js +0 -17
  110. package/esm/utils/version.js.map +0 -1
  111. package/esm/utils.d.ts +0 -3
  112. package/esm/utils.js +0 -4
  113. package/esm/utils.js.map +0 -1
  114. package/esm/worker.d.ts +0 -10
  115. package/esm/worker.js +0 -43
  116. package/esm/worker.js.map +0 -1
  117. /package/{cjs/common-internals-node → common-internals-node}/exit-hook.d.ts +0 -0
  118. /package/{cjs/common-internals-node → common-internals-node}/logging.d.ts +0 -0
  119. /package/{cjs/common-internals-node → common-internals-node}/platform.d.ts +0 -0
  120. /package/{cjs/index.d.ts → index.d.cts} +0 -0
  121. /package/{cjs/methods → methods}/download-file.d.ts +0 -0
  122. /package/{cjs/methods.d.ts → methods.d.cts} +0 -0
  123. /package/{cjs/sqlite → sqlite}/index.d.ts +0 -0
  124. /package/{cjs/utils → utils}/version.d.ts +0 -0
  125. /package/{cjs/utils.d.ts → utils.d.cts} +0 -0
@@ -0,0 +1,33 @@
1
+ if (typeof globalThis !== "undefined" && !globalThis._MTCUTE_CJS_DEPRECATION_WARNED) {
2
+ globalThis._MTCUTE_CJS_DEPRECATION_WARNED = true;
3
+ console.warn("[@mtcute/node] CommonJS support is deprecated and will be removed soon. Please consider switching to ESM, it's " + (/* @__PURE__ */ new Date()).getFullYear() + " already.");
4
+ console.warn("[@mtcute/node] Learn more about switching to ESM: https://gist.github.com/sindresorhus/a39789f98801d908bbc7ff3ecc99d99c");
5
+ }
6
+ "use strict";
7
+ const node_fs = require("node:fs");
8
+ const promises = require("node:fs/promises");
9
+ const core = require("@mtcute/core");
10
+ const methods_js = require("@mtcute/core/methods.js");
11
+ const streamUtils = require("./HP2yqAk_.js");
12
+ async function downloadToFile(client, filename, location, params) {
13
+ if (location instanceof core.FileLocation && ArrayBuffer.isView(location.location)) {
14
+ await promises.writeFile(filename, location.location);
15
+ }
16
+ const output = node_fs.createWriteStream(filename);
17
+ if (params?.abortSignal) {
18
+ params.abortSignal.addEventListener("abort", () => {
19
+ client.log.debug("aborting file download %s - cleaning up", filename);
20
+ output.destroy();
21
+ node_fs.rmSync(filename);
22
+ });
23
+ }
24
+ for await (const chunk of methods_js.downloadAsIterable(client, location, params)) {
25
+ output.write(chunk);
26
+ }
27
+ output.end();
28
+ }
29
+ function downloadAsNodeStream(client, location, params) {
30
+ return streamUtils.webStreamToNode(methods_js.downloadAsStream(client, location, params));
31
+ }
32
+ exports.downloadAsNodeStream = downloadAsNodeStream;
33
+ exports.downloadToFile = downloadToFile;
@@ -0,0 +1,74 @@
1
+ if (typeof globalThis !== "undefined" && !globalThis._MTCUTE_CJS_DEPRECATION_WARNED) {
2
+ globalThis._MTCUTE_CJS_DEPRECATION_WARNED = true;
3
+ console.warn("[@mtcute/node] CommonJS support is deprecated and will be removed soon. Please consider switching to ESM, it's " + (/* @__PURE__ */ new Date()).getFullYear() + " already.");
4
+ console.warn("[@mtcute/node] Learn more about switching to ESM: https://gist.github.com/sindresorhus/a39789f98801d908bbc7ff3ecc99d99c");
5
+ }
6
+ "use strict";
7
+ const node_crypto = require("node:crypto");
8
+ const promises = require("node:fs/promises");
9
+ const node_module = require("node:module");
10
+ const node_zlib = require("node:zlib");
11
+ const utils_js = require("@mtcute/core/utils.js");
12
+ const wasm = require("@mtcute/wasm");
13
+ var _documentCurrentScript = typeof document !== "undefined" ? document.currentScript : null;
14
+ class BaseNodeCryptoProvider extends utils_js.BaseCryptoProvider {
15
+ createAesCtr(key, iv) {
16
+ const cipher = node_crypto.createCipheriv(`aes-${key.length * 8}-ctr`, key, iv);
17
+ const update = (data) => cipher.update(data);
18
+ return {
19
+ process: update
20
+ };
21
+ }
22
+ pbkdf2(password, salt, iterations, keylen = 64, algo = "sha512") {
23
+ return new Promise(
24
+ (resolve, reject) => node_crypto.pbkdf2(password, salt, iterations, keylen, algo, (err, buf) => err !== null ? reject(err) : resolve(buf))
25
+ );
26
+ }
27
+ sha1(data) {
28
+ return node_crypto.createHash("sha1").update(data).digest();
29
+ }
30
+ sha256(data) {
31
+ return node_crypto.createHash("sha256").update(data).digest();
32
+ }
33
+ hmacSha256(data, key) {
34
+ return node_crypto.createHmac("sha256", key).update(data).digest();
35
+ }
36
+ gzip(data, maxSize) {
37
+ try {
38
+ return node_zlib.deflateSync(data, {
39
+ maxOutputLength: maxSize
40
+ });
41
+ } catch (e) {
42
+ if (e.code === "ERR_BUFFER_TOO_LARGE") {
43
+ return null;
44
+ }
45
+ throw e;
46
+ }
47
+ }
48
+ gunzip(data) {
49
+ return node_zlib.gunzipSync(data);
50
+ }
51
+ randomFill(buf) {
52
+ node_crypto.randomFillSync(buf);
53
+ }
54
+ }
55
+ class NodeCryptoProvider extends BaseNodeCryptoProvider {
56
+ async initialize() {
57
+ const require$1 = node_module.createRequire(typeof document === "undefined" ? require("url").pathToFileURL(__filename).href : _documentCurrentScript && _documentCurrentScript.src || new URL("chunks/cjs/D7i-4e0W.js", document.baseURI).href);
58
+ const wasmFile = require$1.resolve("@mtcute/wasm/mtcute.wasm");
59
+ const wasm$1 = await promises.readFile(wasmFile);
60
+ wasm.initSync(wasm$1);
61
+ }
62
+ createAesIge(key, iv) {
63
+ return {
64
+ encrypt(data) {
65
+ return wasm.ige256Encrypt(data, key, iv);
66
+ },
67
+ decrypt(data) {
68
+ return wasm.ige256Decrypt(data, key, iv);
69
+ }
70
+ };
71
+ }
72
+ }
73
+ exports.BaseNodeCryptoProvider = BaseNodeCryptoProvider;
74
+ exports.NodeCryptoProvider = NodeCryptoProvider;
@@ -0,0 +1,79 @@
1
+ if (typeof globalThis !== "undefined" && !globalThis._MTCUTE_CJS_DEPRECATION_WARNED) {
2
+ globalThis._MTCUTE_CJS_DEPRECATION_WARNED = true;
3
+ console.warn("[@mtcute/node] CommonJS support is deprecated and will be removed soon. Please consider switching to ESM, it's " + (/* @__PURE__ */ new Date()).getFullYear() + " already.");
4
+ console.warn("[@mtcute/node] Learn more about switching to ESM: https://gist.github.com/sindresorhus/a39789f98801d908bbc7ff3ecc99d99c");
5
+ }
6
+ "use strict";
7
+ const node_stream = require("node:stream");
8
+ const NODE_VERSION = typeof process !== "undefined" && "node" in process.versions ? process.versions.node : null;
9
+ const NODE_VERSION_TUPLE = NODE_VERSION ? /* @__PURE__ */ NODE_VERSION.split(".").map(Number) : null;
10
+ function isNodeVersionAfter(major, minor, patch) {
11
+ if (!NODE_VERSION_TUPLE) return true;
12
+ const [a, b, c] = NODE_VERSION_TUPLE;
13
+ if (a > major) return true;
14
+ if (a < major) return false;
15
+ if (b > minor) return true;
16
+ if (b < minor) return false;
17
+ return c >= patch;
18
+ }
19
+ function nodeStreamToWeb(stream) {
20
+ if (typeof node_stream.Readable.toWeb === "function") {
21
+ return node_stream.Readable.toWeb(stream);
22
+ }
23
+ stream.pause();
24
+ return new ReadableStream({
25
+ start(c) {
26
+ stream.on("data", (chunk) => {
27
+ c.enqueue(chunk);
28
+ });
29
+ stream.on("end", () => {
30
+ c.close();
31
+ });
32
+ stream.on("error", (err) => {
33
+ c.error(err);
34
+ });
35
+ },
36
+ pull() {
37
+ stream.resume();
38
+ }
39
+ });
40
+ }
41
+ function webStreamToNode(stream) {
42
+ if (typeof node_stream.Readable.fromWeb === "function" && isNodeVersionAfter(18, 13, 0)) {
43
+ return node_stream.Readable.fromWeb(stream);
44
+ }
45
+ const reader = stream.getReader();
46
+ let ended = false;
47
+ const readable = new node_stream.Readable({
48
+ async read() {
49
+ try {
50
+ const { done, value } = await reader.read();
51
+ if (done) {
52
+ this.push(null);
53
+ } else {
54
+ this.push(Buffer.from(value.buffer, value.byteOffset, value.byteLength));
55
+ }
56
+ } catch (err) {
57
+ this.destroy(err);
58
+ }
59
+ },
60
+ destroy(error, cb) {
61
+ if (!ended) {
62
+ void reader.cancel(error).catch(() => {
63
+ }).then(() => {
64
+ cb(error);
65
+ });
66
+ return;
67
+ }
68
+ cb(error);
69
+ }
70
+ });
71
+ reader.closed.then(() => {
72
+ ended = true;
73
+ }).catch((err) => {
74
+ readable.destroy(err);
75
+ });
76
+ return readable;
77
+ }
78
+ exports.nodeStreamToWeb = nodeStreamToWeb;
79
+ exports.webStreamToNode = webStreamToNode;
@@ -0,0 +1,3 @@
1
+ {
2
+ "type": "commonjs"
3
+ }
@@ -0,0 +1,75 @@
1
+ import { Readable } from "node:stream";
2
+ const NODE_VERSION = typeof process !== "undefined" && "node" in process.versions ? process.versions.node : null;
3
+ const NODE_VERSION_TUPLE = NODE_VERSION ? /* @__PURE__ */ NODE_VERSION.split(".").map(Number) : null;
4
+ function isNodeVersionAfter(major, minor, patch) {
5
+ if (!NODE_VERSION_TUPLE) return true;
6
+ const [a, b, c] = NODE_VERSION_TUPLE;
7
+ if (a > major) return true;
8
+ if (a < major) return false;
9
+ if (b > minor) return true;
10
+ if (b < minor) return false;
11
+ return c >= patch;
12
+ }
13
+ function nodeStreamToWeb(stream) {
14
+ if (typeof Readable.toWeb === "function") {
15
+ return Readable.toWeb(stream);
16
+ }
17
+ stream.pause();
18
+ return new ReadableStream({
19
+ start(c) {
20
+ stream.on("data", (chunk) => {
21
+ c.enqueue(chunk);
22
+ });
23
+ stream.on("end", () => {
24
+ c.close();
25
+ });
26
+ stream.on("error", (err) => {
27
+ c.error(err);
28
+ });
29
+ },
30
+ pull() {
31
+ stream.resume();
32
+ }
33
+ });
34
+ }
35
+ function webStreamToNode(stream) {
36
+ if (typeof Readable.fromWeb === "function" && isNodeVersionAfter(18, 13, 0)) {
37
+ return Readable.fromWeb(stream);
38
+ }
39
+ const reader = stream.getReader();
40
+ let ended = false;
41
+ const readable = new Readable({
42
+ async read() {
43
+ try {
44
+ const { done, value } = await reader.read();
45
+ if (done) {
46
+ this.push(null);
47
+ } else {
48
+ this.push(Buffer.from(value.buffer, value.byteOffset, value.byteLength));
49
+ }
50
+ } catch (err) {
51
+ this.destroy(err);
52
+ }
53
+ },
54
+ destroy(error, cb) {
55
+ if (!ended) {
56
+ void reader.cancel(error).catch(() => {
57
+ }).then(() => {
58
+ cb(error);
59
+ });
60
+ return;
61
+ }
62
+ cb(error);
63
+ }
64
+ });
65
+ reader.closed.then(() => {
66
+ ended = true;
67
+ }).catch((err) => {
68
+ readable.destroy(err);
69
+ });
70
+ return readable;
71
+ }
72
+ export {
73
+ nodeStreamToWeb,
74
+ webStreamToNode
75
+ };
@@ -0,0 +1,29 @@
1
+ import { createWriteStream, rmSync } from "node:fs";
2
+ import { writeFile } from "node:fs/promises";
3
+ import { FileLocation } from "@mtcute/core";
4
+ import { downloadAsIterable, downloadAsStream } from "@mtcute/core/methods.js";
5
+ import { webStreamToNode } from "./CKso6cAV.js";
6
+ async function downloadToFile(client, filename, location, params) {
7
+ if (location instanceof FileLocation && ArrayBuffer.isView(location.location)) {
8
+ await writeFile(filename, location.location);
9
+ }
10
+ const output = createWriteStream(filename);
11
+ if (params?.abortSignal) {
12
+ params.abortSignal.addEventListener("abort", () => {
13
+ client.log.debug("aborting file download %s - cleaning up", filename);
14
+ output.destroy();
15
+ rmSync(filename);
16
+ });
17
+ }
18
+ for await (const chunk of downloadAsIterable(client, location, params)) {
19
+ output.write(chunk);
20
+ }
21
+ output.end();
22
+ }
23
+ function downloadAsNodeStream(client, location, params) {
24
+ return webStreamToNode(downloadAsStream(client, location, params));
25
+ }
26
+ export {
27
+ downloadAsNodeStream,
28
+ downloadToFile
29
+ };
@@ -0,0 +1,69 @@
1
+ import { createCipheriv, pbkdf2, createHash, createHmac, randomFillSync } from "node:crypto";
2
+ import { readFile } from "node:fs/promises";
3
+ import { createRequire } from "node:module";
4
+ import { deflateSync, gunzipSync } from "node:zlib";
5
+ import { BaseCryptoProvider } from "@mtcute/core/utils.js";
6
+ import { initSync, ige256Encrypt, ige256Decrypt } from "@mtcute/wasm";
7
+ class BaseNodeCryptoProvider extends BaseCryptoProvider {
8
+ createAesCtr(key, iv) {
9
+ const cipher = createCipheriv(`aes-${key.length * 8}-ctr`, key, iv);
10
+ const update = (data) => cipher.update(data);
11
+ return {
12
+ process: update
13
+ };
14
+ }
15
+ pbkdf2(password, salt, iterations, keylen = 64, algo = "sha512") {
16
+ return new Promise(
17
+ (resolve, reject) => pbkdf2(password, salt, iterations, keylen, algo, (err, buf) => err !== null ? reject(err) : resolve(buf))
18
+ );
19
+ }
20
+ sha1(data) {
21
+ return createHash("sha1").update(data).digest();
22
+ }
23
+ sha256(data) {
24
+ return createHash("sha256").update(data).digest();
25
+ }
26
+ hmacSha256(data, key) {
27
+ return createHmac("sha256", key).update(data).digest();
28
+ }
29
+ gzip(data, maxSize) {
30
+ try {
31
+ return deflateSync(data, {
32
+ maxOutputLength: maxSize
33
+ });
34
+ } catch (e) {
35
+ if (e.code === "ERR_BUFFER_TOO_LARGE") {
36
+ return null;
37
+ }
38
+ throw e;
39
+ }
40
+ }
41
+ gunzip(data) {
42
+ return gunzipSync(data);
43
+ }
44
+ randomFill(buf) {
45
+ randomFillSync(buf);
46
+ }
47
+ }
48
+ class NodeCryptoProvider extends BaseNodeCryptoProvider {
49
+ async initialize() {
50
+ const require2 = createRequire(import.meta.url);
51
+ const wasmFile = require2.resolve("@mtcute/wasm/mtcute.wasm");
52
+ const wasm = await readFile(wasmFile);
53
+ initSync(wasm);
54
+ }
55
+ createAesIge(key, iv) {
56
+ return {
57
+ encrypt(data) {
58
+ return ige256Encrypt(data, key, iv);
59
+ },
60
+ decrypt(data) {
61
+ return ige256Decrypt(data, key, iv);
62
+ }
63
+ };
64
+ }
65
+ }
66
+ export {
67
+ BaseNodeCryptoProvider,
68
+ NodeCryptoProvider
69
+ };
@@ -1,5 +1,6 @@
1
+ import { Readable } from 'node:stream';
1
2
  import { FileDownloadLocation, FileDownloadParameters, ITelegramStorageProvider, PartialOnly, User } from '@mtcute/core';
2
- import { BaseTelegramClient as BaseTelegramClientBase, BaseTelegramClientOptions as BaseTelegramClientOptionsBase, TelegramClient as TelegramClientBase, TelegramClientOptions } from '@mtcute/core/client.js';
3
+ import { BaseTelegramClientOptions as BaseTelegramClientOptionsBase, TelegramClientOptions, BaseTelegramClient as BaseTelegramClientBase, TelegramClient as TelegramClientBase } from '@mtcute/core/client.js';
3
4
  export type { TelegramClientOptions };
4
5
  export interface BaseTelegramClientOptions extends PartialOnly<Omit<BaseTelegramClientOptionsBase, 'storage'>, 'transport' | 'crypto'> {
5
6
  /**
@@ -42,5 +43,5 @@ export declare class TelegramClient extends TelegramClientBase {
42
43
  start(params?: Parameters<TelegramClientBase['start']>[0]): Promise<User>;
43
44
  run(params: Parameters<TelegramClient['start']>[0] | ((user: User) => void | Promise<void>), then?: (user: User) => void | Promise<void>): void;
44
45
  downloadToFile(filename: string, location: FileDownloadLocation, params?: FileDownloadParameters | undefined): Promise<void>;
45
- downloadAsNodeStream(location: FileDownloadLocation, params?: FileDownloadParameters | undefined): import("stream").Readable;
46
+ downloadAsNodeStream(location: FileDownloadLocation, params?: FileDownloadParameters | undefined): Readable;
46
47
  }