@smythos/sre 1.5.53 → 1.5.54
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/CHANGELOG +98 -98
- package/LICENSE +18 -18
- package/README.md +135 -135
- package/dist/bundle-analysis-lazy.html +4949 -0
- package/dist/bundle-analysis.html +4949 -0
- package/dist/index.js +3 -3
- package/dist/index.js.map +1 -1
- package/dist/types/subsystems/LLMManager/LLM.service/connectors/openai/OpenAIConnector.class.d.ts +1 -6
- package/dist/types/utils/package-manager.utils.d.ts +26 -0
- package/package.json +1 -1
- package/src/Components/APICall/APICall.class.ts +157 -157
- package/src/Components/APICall/AccessTokenManager.ts +166 -166
- package/src/Components/APICall/ArrayBufferResponse.helper.ts +58 -58
- package/src/Components/APICall/OAuth.helper.ts +447 -447
- package/src/Components/APICall/mimeTypeCategories.ts +46 -46
- package/src/Components/APICall/parseData.ts +167 -167
- package/src/Components/APICall/parseHeaders.ts +41 -41
- package/src/Components/APICall/parseProxy.ts +68 -68
- package/src/Components/APICall/parseUrl.ts +91 -91
- package/src/Components/APIEndpoint.class.ts +234 -234
- package/src/Components/APIOutput.class.ts +58 -58
- package/src/Components/AgentPlugin.class.ts +102 -102
- package/src/Components/Async.class.ts +155 -155
- package/src/Components/Await.class.ts +90 -90
- package/src/Components/Classifier.class.ts +158 -158
- package/src/Components/Component.class.ts +132 -132
- package/src/Components/ComponentHost.class.ts +38 -38
- package/src/Components/DataSourceCleaner.class.ts +92 -92
- package/src/Components/DataSourceIndexer.class.ts +181 -181
- package/src/Components/DataSourceLookup.class.ts +161 -161
- package/src/Components/ECMASandbox.class.ts +71 -71
- package/src/Components/FEncDec.class.ts +29 -29
- package/src/Components/FHash.class.ts +33 -33
- package/src/Components/FSign.class.ts +80 -80
- package/src/Components/FSleep.class.ts +25 -25
- package/src/Components/FTimestamp.class.ts +25 -25
- package/src/Components/FileStore.class.ts +78 -78
- package/src/Components/ForEach.class.ts +97 -97
- package/src/Components/GPTPlugin.class.ts +70 -70
- package/src/Components/GenAILLM.class.ts +586 -586
- package/src/Components/HuggingFace.class.ts +314 -314
- package/src/Components/Image/imageSettings.config.ts +70 -70
- package/src/Components/ImageGenerator.class.ts +502 -502
- package/src/Components/JSONFilter.class.ts +54 -54
- package/src/Components/LLMAssistant.class.ts +213 -213
- package/src/Components/LogicAND.class.ts +28 -28
- package/src/Components/LogicAtLeast.class.ts +85 -85
- package/src/Components/LogicAtMost.class.ts +86 -86
- package/src/Components/LogicOR.class.ts +29 -29
- package/src/Components/LogicXOR.class.ts +34 -34
- package/src/Components/MCPClient.class.ts +138 -138
- package/src/Components/MemoryDeleteKeyVal.class.ts +70 -70
- package/src/Components/MemoryReadKeyVal.class.ts +66 -66
- package/src/Components/MemoryWriteKeyVal.class.ts +62 -62
- package/src/Components/MemoryWriteObject.class.ts +97 -97
- package/src/Components/MultimodalLLM.class.ts +128 -128
- package/src/Components/OpenAPI.class.ts +72 -72
- package/src/Components/PromptGenerator.class.ts +122 -122
- package/src/Components/ScrapflyWebScrape.class.ts +159 -159
- package/src/Components/ServerlessCode.class.ts +123 -123
- package/src/Components/TavilyWebSearch.class.ts +98 -98
- package/src/Components/VisionLLM.class.ts +104 -104
- package/src/Components/ZapierAction.class.ts +127 -127
- package/src/Components/index.ts +97 -97
- package/src/Core/AgentProcess.helper.ts +240 -240
- package/src/Core/Connector.class.ts +123 -123
- package/src/Core/ConnectorsService.ts +197 -197
- package/src/Core/DummyConnector.ts +49 -49
- package/src/Core/HookService.ts +105 -105
- package/src/Core/SmythRuntime.class.ts +235 -235
- package/src/Core/SystemEvents.ts +16 -16
- package/src/Core/boot.ts +56 -56
- package/src/config.ts +15 -15
- package/src/constants.ts +126 -126
- package/src/data/hugging-face.params.json +579 -579
- package/src/helpers/AWSLambdaCode.helper.ts +590 -590
- package/src/helpers/BinaryInput.helper.ts +331 -331
- package/src/helpers/Conversation.helper.ts +1119 -1119
- package/src/helpers/ECMASandbox.helper.ts +54 -54
- package/src/helpers/JsonContent.helper.ts +97 -97
- package/src/helpers/LocalCache.helper.ts +97 -97
- package/src/helpers/Log.helper.ts +274 -274
- package/src/helpers/OpenApiParser.helper.ts +150 -150
- package/src/helpers/S3Cache.helper.ts +147 -147
- package/src/helpers/SmythURI.helper.ts +5 -5
- package/src/helpers/Sysconfig.helper.ts +77 -77
- package/src/helpers/TemplateString.helper.ts +243 -243
- package/src/helpers/TypeChecker.helper.ts +329 -329
- package/src/index.ts +3 -3
- package/src/index.ts.bak +3 -3
- package/src/subsystems/AgentManager/Agent.class.ts +1114 -1114
- package/src/subsystems/AgentManager/Agent.helper.ts +3 -3
- package/src/subsystems/AgentManager/AgentData.service/AgentDataConnector.ts +230 -230
- package/src/subsystems/AgentManager/AgentData.service/connectors/CLIAgentDataConnector.class.ts +66 -66
- package/src/subsystems/AgentManager/AgentData.service/connectors/LocalAgentDataConnector.class.ts +142 -142
- package/src/subsystems/AgentManager/AgentData.service/connectors/NullAgentData.class.ts +39 -39
- package/src/subsystems/AgentManager/AgentData.service/index.ts +18 -18
- package/src/subsystems/AgentManager/AgentLogger.class.ts +301 -297
- package/src/subsystems/AgentManager/AgentRequest.class.ts +51 -51
- package/src/subsystems/AgentManager/AgentRuntime.class.ts +559 -559
- package/src/subsystems/AgentManager/AgentSSE.class.ts +101 -101
- package/src/subsystems/AgentManager/AgentSettings.class.ts +52 -52
- package/src/subsystems/AgentManager/Component.service/ComponentConnector.ts +32 -32
- package/src/subsystems/AgentManager/Component.service/connectors/LocalComponentConnector.class.ts +60 -60
- package/src/subsystems/AgentManager/Component.service/index.ts +11 -11
- package/src/subsystems/AgentManager/EmbodimentSettings.class.ts +47 -47
- package/src/subsystems/AgentManager/ForkedAgent.class.ts +154 -154
- package/src/subsystems/AgentManager/OSResourceMonitor.ts +77 -77
- package/src/subsystems/ComputeManager/Code.service/CodeConnector.ts +98 -98
- package/src/subsystems/ComputeManager/Code.service/connectors/AWSLambdaCode.class.ts +172 -172
- package/src/subsystems/ComputeManager/Code.service/connectors/ECMASandbox.class.ts +131 -131
- package/src/subsystems/ComputeManager/Code.service/index.ts +13 -13
- package/src/subsystems/IO/CLI.service/CLIConnector.ts +47 -47
- package/src/subsystems/IO/CLI.service/index.ts +9 -9
- package/src/subsystems/IO/Log.service/LogConnector.ts +32 -32
- package/src/subsystems/IO/Log.service/connectors/ConsoleLog.class.ts +28 -28
- package/src/subsystems/IO/Log.service/index.ts +13 -13
- package/src/subsystems/IO/NKV.service/NKVConnector.ts +43 -43
- package/src/subsystems/IO/NKV.service/connectors/NKVLocalStorage.class.ts +234 -234
- package/src/subsystems/IO/NKV.service/connectors/NKVRAM.class.ts +204 -204
- package/src/subsystems/IO/NKV.service/connectors/NKVRedis.class.ts +182 -182
- package/src/subsystems/IO/NKV.service/index.ts +14 -14
- package/src/subsystems/IO/Router.service/RouterConnector.ts +21 -21
- package/src/subsystems/IO/Router.service/connectors/ExpressRouter.class.ts +48 -48
- package/src/subsystems/IO/Router.service/connectors/NullRouter.class.ts +40 -40
- package/src/subsystems/IO/Router.service/index.ts +11 -11
- package/src/subsystems/IO/Storage.service/SmythFS.class.ts +489 -489
- package/src/subsystems/IO/Storage.service/StorageConnector.ts +66 -66
- package/src/subsystems/IO/Storage.service/connectors/LocalStorage.class.ts +327 -327
- package/src/subsystems/IO/Storage.service/connectors/S3Storage.class.ts +482 -482
- package/src/subsystems/IO/Storage.service/index.ts +13 -13
- package/src/subsystems/IO/VectorDB.service/VectorDBConnector.ts +108 -108
- package/src/subsystems/IO/VectorDB.service/connectors/MilvusVectorDB.class.ts +454 -454
- package/src/subsystems/IO/VectorDB.service/connectors/PineconeVectorDB.class.ts +384 -384
- package/src/subsystems/IO/VectorDB.service/connectors/RAMVecrtorDB.class.ts +421 -421
- package/src/subsystems/IO/VectorDB.service/embed/BaseEmbedding.ts +107 -107
- package/src/subsystems/IO/VectorDB.service/embed/OpenAIEmbedding.ts +109 -109
- package/src/subsystems/IO/VectorDB.service/embed/index.ts +21 -21
- package/src/subsystems/IO/VectorDB.service/index.ts +14 -14
- package/src/subsystems/LLMManager/LLM.helper.ts +251 -251
- package/src/subsystems/LLMManager/LLM.inference.ts +339 -339
- package/src/subsystems/LLMManager/LLM.service/LLMConnector.ts +489 -489
- package/src/subsystems/LLMManager/LLM.service/LLMCredentials.helper.ts +171 -171
- package/src/subsystems/LLMManager/LLM.service/connectors/Anthropic.class.ts +659 -659
- package/src/subsystems/LLMManager/LLM.service/connectors/Bedrock.class.ts +400 -400
- package/src/subsystems/LLMManager/LLM.service/connectors/Echo.class.ts +77 -77
- package/src/subsystems/LLMManager/LLM.service/connectors/GoogleAI.class.ts +757 -757
- package/src/subsystems/LLMManager/LLM.service/connectors/Groq.class.ts +304 -304
- package/src/subsystems/LLMManager/LLM.service/connectors/Perplexity.class.ts +250 -250
- package/src/subsystems/LLMManager/LLM.service/connectors/VertexAI.class.ts +423 -423
- package/src/subsystems/LLMManager/LLM.service/connectors/openai/OpenAIConnector.class.ts +488 -488
- package/src/subsystems/LLMManager/LLM.service/connectors/openai/apiInterfaces/ChatCompletionsApiInterface.ts +524 -524
- package/src/subsystems/LLMManager/LLM.service/connectors/openai/apiInterfaces/OpenAIApiInterface.ts +100 -100
- package/src/subsystems/LLMManager/LLM.service/connectors/openai/apiInterfaces/OpenAIApiInterfaceFactory.ts +81 -81
- package/src/subsystems/LLMManager/LLM.service/connectors/openai/apiInterfaces/ResponsesApiInterface.ts +1145 -1145
- package/src/subsystems/LLMManager/LLM.service/connectors/openai/apiInterfaces/constants.ts +13 -13
- package/src/subsystems/LLMManager/LLM.service/connectors/openai/apiInterfaces/index.ts +4 -4
- package/src/subsystems/LLMManager/LLM.service/connectors/openai/apiInterfaces/utils.ts +11 -11
- package/src/subsystems/LLMManager/LLM.service/connectors/openai/types.ts +32 -32
- package/src/subsystems/LLMManager/LLM.service/connectors/xAI.class.ts +471 -471
- package/src/subsystems/LLMManager/LLM.service/index.ts +44 -44
- package/src/subsystems/LLMManager/ModelsProvider.service/ModelsProviderConnector.ts +300 -300
- package/src/subsystems/LLMManager/ModelsProvider.service/connectors/JSONModelsProvider.class.ts +252 -252
- package/src/subsystems/LLMManager/ModelsProvider.service/index.ts +11 -11
- package/src/subsystems/LLMManager/custom-models.ts +854 -854
- package/src/subsystems/LLMManager/models.ts +2540 -2540
- package/src/subsystems/LLMManager/paramMappings.ts +69 -69
- package/src/subsystems/MemoryManager/Cache.service/CacheConnector.ts +86 -86
- package/src/subsystems/MemoryManager/Cache.service/connectors/LocalStorageCache.class.ts +297 -297
- package/src/subsystems/MemoryManager/Cache.service/connectors/RAMCache.class.ts +201 -201
- package/src/subsystems/MemoryManager/Cache.service/connectors/RedisCache.class.ts +252 -252
- package/src/subsystems/MemoryManager/Cache.service/connectors/S3Cache.class.ts +373 -373
- package/src/subsystems/MemoryManager/Cache.service/index.ts +15 -15
- package/src/subsystems/MemoryManager/LLMCache.ts +72 -72
- package/src/subsystems/MemoryManager/LLMContext.ts +124 -124
- package/src/subsystems/MemoryManager/LLMMemory.service/LLMMemoryConnector.ts +26 -26
- package/src/subsystems/MemoryManager/RuntimeContext.ts +266 -266
- package/src/subsystems/Security/AccessControl/ACL.class.ts +208 -208
- package/src/subsystems/Security/AccessControl/AccessCandidate.class.ts +82 -82
- package/src/subsystems/Security/AccessControl/AccessRequest.class.ts +52 -52
- package/src/subsystems/Security/Account.service/AccountConnector.ts +44 -44
- package/src/subsystems/Security/Account.service/connectors/AWSAccount.class.ts +76 -76
- package/src/subsystems/Security/Account.service/connectors/DummyAccount.class.ts +130 -130
- package/src/subsystems/Security/Account.service/connectors/JSONFileAccount.class.ts +159 -159
- package/src/subsystems/Security/Account.service/index.ts +14 -14
- package/src/subsystems/Security/Credentials.helper.ts +62 -62
- package/src/subsystems/Security/ManagedVault.service/ManagedVaultConnector.ts +38 -38
- package/src/subsystems/Security/ManagedVault.service/connectors/NullManagedVault.class.ts +53 -53
- package/src/subsystems/Security/ManagedVault.service/connectors/SecretManagerManagedVault.ts +154 -154
- package/src/subsystems/Security/ManagedVault.service/index.ts +12 -12
- package/src/subsystems/Security/SecureConnector.class.ts +110 -110
- package/src/subsystems/Security/Vault.service/Vault.helper.ts +30 -30
- package/src/subsystems/Security/Vault.service/VaultConnector.ts +29 -29
- package/src/subsystems/Security/Vault.service/connectors/HashicorpVault.class.ts +46 -46
- package/src/subsystems/Security/Vault.service/connectors/JSONFileVault.class.ts +221 -221
- package/src/subsystems/Security/Vault.service/connectors/NullVault.class.ts +54 -54
- package/src/subsystems/Security/Vault.service/connectors/SecretsManager.class.ts +140 -140
- package/src/subsystems/Security/Vault.service/index.ts +12 -12
- package/src/types/ACL.types.ts +104 -104
- package/src/types/AWS.types.ts +10 -10
- package/src/types/Agent.types.ts +61 -61
- package/src/types/AgentLogger.types.ts +17 -17
- package/src/types/Cache.types.ts +1 -1
- package/src/types/Common.types.ts +2 -2
- package/src/types/LLM.types.ts +496 -496
- package/src/types/Redis.types.ts +8 -8
- package/src/types/SRE.types.ts +64 -64
- package/src/types/Security.types.ts +14 -14
- package/src/types/Storage.types.ts +5 -5
- package/src/types/VectorDB.types.ts +86 -86
- package/src/utils/base64.utils.ts +275 -275
- package/src/utils/cli.utils.ts +68 -68
- package/src/utils/data.utils.ts +322 -322
- package/src/utils/date-time.utils.ts +22 -22
- package/src/utils/general.utils.ts +238 -238
- package/src/utils/index.ts +12 -12
- package/src/utils/lazy-client.ts +261 -261
- package/src/utils/numbers.utils.ts +13 -13
- package/src/utils/oauth.utils.ts +35 -35
- package/src/utils/string.utils.ts +414 -414
- package/src/utils/url.utils.ts +19 -19
- package/src/utils/validation.utils.ts +74 -74
- package/dist/types/subsystems/LLMManager/ModelsProvider.service/connectors/SmythModelsProvider.class.d.ts +0 -39
|
@@ -1,331 +1,331 @@
|
|
|
1
|
-
import axios from 'axios';
|
|
2
|
-
import mime from 'mime';
|
|
3
|
-
import { fileTypeFromBuffer } from 'file-type';
|
|
4
|
-
import { Readable } from 'stream';
|
|
5
|
-
import fs from 'fs';
|
|
6
|
-
import path from 'path';
|
|
7
|
-
import os from 'os';
|
|
8
|
-
|
|
9
|
-
import { ConnectorService } from '@sre/Core/ConnectorsService';
|
|
10
|
-
import { SmythFS } from '@sre/IO/Storage.service/SmythFS.class';
|
|
11
|
-
import { IAccessCandidate } from '@sre/types/ACL.types';
|
|
12
|
-
import { getSizeFromBinary, isUrl, uid, getBase64FileInfo, getMimeType } from '@sre/utils';
|
|
13
|
-
import config from '@sre/config';
|
|
14
|
-
export class BinaryInput {
|
|
15
|
-
private size: number;
|
|
16
|
-
private url: string;
|
|
17
|
-
private _ready;
|
|
18
|
-
private _readyPromise;
|
|
19
|
-
private _source: Buffer;
|
|
20
|
-
private _uploading: boolean = false;
|
|
21
|
-
|
|
22
|
-
constructor(
|
|
23
|
-
data: BinaryInput | Buffer | ArrayBuffer | Blob | string | Record<string, any>,
|
|
24
|
-
private _name?: string,
|
|
25
|
-
public mimetype?: string,
|
|
26
|
-
private candidate?: IAccessCandidate,
|
|
27
|
-
) {
|
|
28
|
-
if (!_name) _name = uid();
|
|
29
|
-
this._name = _name;
|
|
30
|
-
//this._source = data;
|
|
31
|
-
|
|
32
|
-
this.load(data, _name, mimetype, candidate);
|
|
33
|
-
}
|
|
34
|
-
|
|
35
|
-
public async ready() {
|
|
36
|
-
if (this._ready) return true;
|
|
37
|
-
|
|
38
|
-
if (!this._readyPromise) {
|
|
39
|
-
this._readyPromise = new Promise((resolve) => {
|
|
40
|
-
const maxWait = 10000;
|
|
41
|
-
const interval = setInterval(() => {
|
|
42
|
-
if (this._ready) {
|
|
43
|
-
clearInterval(interval);
|
|
44
|
-
resolve(true);
|
|
45
|
-
}
|
|
46
|
-
if (maxWait <= 0) {
|
|
47
|
-
clearInterval(interval);
|
|
48
|
-
resolve(false);
|
|
49
|
-
}
|
|
50
|
-
}, 100);
|
|
51
|
-
});
|
|
52
|
-
}
|
|
53
|
-
|
|
54
|
-
return this._readyPromise;
|
|
55
|
-
}
|
|
56
|
-
|
|
57
|
-
private async load(data, name: string, mimetype?: string, candidate?: IAccessCandidate) {
|
|
58
|
-
//assume the mimetype from the provided name
|
|
59
|
-
const ext: any = name.split('.')?.length > 1 ? name.split('.').pop() : '';
|
|
60
|
-
// Need to set mimetype empty string if no extension is found, setting default mimetype to 'application/octet-stream' lead wrong direction when try to get the mimetype from base64 or buffer data (as it's not accurate all the time)
|
|
61
|
-
this.mimetype = mimetype || mime.getType(ext) || '';
|
|
62
|
-
this.url = ``;
|
|
63
|
-
|
|
64
|
-
if (typeof data === 'object' && data.url && data.mimetype && data.size) {
|
|
65
|
-
this.mimetype = data.mimetype;
|
|
66
|
-
this.size = data.size;
|
|
67
|
-
this.url = data.url;
|
|
68
|
-
|
|
69
|
-
const ext = mime.getExtension(this.mimetype);
|
|
70
|
-
if (!this._name.endsWith(`.${ext}`)) this._name += `.${ext}`;
|
|
71
|
-
|
|
72
|
-
if (candidate) {
|
|
73
|
-
this._source = await SmythFS.Instance.read(this.url, candidate).finally(() => {
|
|
74
|
-
this._ready = true;
|
|
75
|
-
});
|
|
76
|
-
} else {
|
|
77
|
-
this._ready = true;
|
|
78
|
-
}
|
|
79
|
-
return;
|
|
80
|
-
}
|
|
81
|
-
|
|
82
|
-
if (typeof data === 'string' && data.startsWith('smythfs://')) {
|
|
83
|
-
this.url = data;
|
|
84
|
-
if (candidate) {
|
|
85
|
-
try {
|
|
86
|
-
this._source = await SmythFS.Instance.read(this.url, candidate);
|
|
87
|
-
this.mimetype = await getMimeType(this._source);
|
|
88
|
-
this.size = this._source.byteLength;
|
|
89
|
-
|
|
90
|
-
if (!this.mimetype) {
|
|
91
|
-
this.mimetype = mime.getType(this.url) || mime.getType(this._name) || '';
|
|
92
|
-
}
|
|
93
|
-
|
|
94
|
-
const ext = mime.getExtension(this.mimetype);
|
|
95
|
-
if (!this._name.endsWith(`.${ext}`)) this._name += `.${ext}`;
|
|
96
|
-
} finally {
|
|
97
|
-
this._ready = true;
|
|
98
|
-
}
|
|
99
|
-
} else {
|
|
100
|
-
this._ready = true;
|
|
101
|
-
}
|
|
102
|
-
return;
|
|
103
|
-
}
|
|
104
|
-
|
|
105
|
-
if (isUrl(data)) {
|
|
106
|
-
try {
|
|
107
|
-
const response = await axios({
|
|
108
|
-
method: 'get',
|
|
109
|
-
url: data,
|
|
110
|
-
responseType: 'arraybuffer', // Important for handling binary data
|
|
111
|
-
});
|
|
112
|
-
|
|
113
|
-
this.size = response.data.byteLength;
|
|
114
|
-
this._source = Buffer.from(response.data, 'binary');
|
|
115
|
-
|
|
116
|
-
let mimetype = response.headers?.['content-type'] || '';
|
|
117
|
-
|
|
118
|
-
// Try to get the file extension from the URL
|
|
119
|
-
const urlPath = new URL(data).pathname;
|
|
120
|
-
let extension = urlPath.split('.')?.length > 1 ? urlPath.split('.').pop() : '';
|
|
121
|
-
|
|
122
|
-
// When content-type header is missing or invalid, try to determine mime type from file extension
|
|
123
|
-
if (!mimetype || mimetype.startsWith('binary/octet-stream')) {
|
|
124
|
-
mimetype = extension ? mime.getType(extension) : '';
|
|
125
|
-
}
|
|
126
|
-
|
|
127
|
-
// If we couldn't determine the mime type from headers or file extension, try detecting it from the actual file content
|
|
128
|
-
if (!mimetype) {
|
|
129
|
-
const type = await fileTypeFromBuffer(this._source);
|
|
130
|
-
mimetype = type?.mime || '';
|
|
131
|
-
}
|
|
132
|
-
|
|
133
|
-
this.mimetype = mimetype;
|
|
134
|
-
|
|
135
|
-
extension = extension || mime.getExtension(this.mimetype);
|
|
136
|
-
if (!this._name.endsWith(`.${extension}`)) this._name += `.${extension}`;
|
|
137
|
-
} catch (error) {
|
|
138
|
-
console.error('Error loading binary data from url:', data.url, error);
|
|
139
|
-
}
|
|
140
|
-
|
|
141
|
-
//this._source = data.url;
|
|
142
|
-
|
|
143
|
-
this._ready = true;
|
|
144
|
-
return;
|
|
145
|
-
}
|
|
146
|
-
|
|
147
|
-
// console.log('>>>>>>>>>>>>>>>>>>> is base64 file ?', isBase64DataUrl(data));
|
|
148
|
-
const base64FileInfo = await getBase64FileInfo(data);
|
|
149
|
-
if (base64FileInfo) {
|
|
150
|
-
// If the MIME type is already set, it's safe to use it,
|
|
151
|
-
// as determining the MIME type from the base64 string is not always accurate, specially when it's not a base64 URL
|
|
152
|
-
if (!this.mimetype) {
|
|
153
|
-
this.mimetype = base64FileInfo.mimetype;
|
|
154
|
-
}
|
|
155
|
-
this.size = base64FileInfo.size;
|
|
156
|
-
this._source = Buffer.from(base64FileInfo.data, 'base64');
|
|
157
|
-
|
|
158
|
-
const ext = mime.getExtension(this.mimetype);
|
|
159
|
-
if (!this._name.endsWith(`.${ext}`)) this._name += `.${ext}`;
|
|
160
|
-
|
|
161
|
-
this._ready = true;
|
|
162
|
-
return;
|
|
163
|
-
}
|
|
164
|
-
|
|
165
|
-
if (typeof data === 'string') {
|
|
166
|
-
this._source = Buffer.from(data);
|
|
167
|
-
this.size = data.length;
|
|
168
|
-
this.mimetype = 'text/plain';
|
|
169
|
-
if (!this._name.endsWith(`.txt`)) this._name += `.txt`;
|
|
170
|
-
|
|
171
|
-
this._ready = true;
|
|
172
|
-
return;
|
|
173
|
-
}
|
|
174
|
-
|
|
175
|
-
//this.size = getSizeFromBinary(data);
|
|
176
|
-
// //try to enforce the mimetype from the provided data
|
|
177
|
-
// if (Buffer.isBuffer(data)) {
|
|
178
|
-
// const fileType = await FileType.fileTypeFromBuffer(data);
|
|
179
|
-
// this.mimetype = fileType.mime;
|
|
180
|
-
// const ext = mime.getExtension(this.mimetype);
|
|
181
|
-
// if (!this._name.endsWith(`.${ext}`)) this._name += `.${ext}`;
|
|
182
|
-
|
|
183
|
-
// this._ready = true;
|
|
184
|
-
// return;
|
|
185
|
-
// }
|
|
186
|
-
//try to enforce the mimetype from the provided data
|
|
187
|
-
if (Buffer.isBuffer(data)) {
|
|
188
|
-
this._source = data;
|
|
189
|
-
this.size = getSizeFromBinary(data);
|
|
190
|
-
// If the MIME type is already set, it's safe to use it,
|
|
191
|
-
// as determining the MIME type from the buffer is not always accurate.
|
|
192
|
-
if (!this.mimetype) {
|
|
193
|
-
this.mimetype = await getMimeType(data);
|
|
194
|
-
}
|
|
195
|
-
const ext = mime.getExtension(this.mimetype);
|
|
196
|
-
if (!this._name.endsWith(`.${ext}`)) this._name += `.${ext}`;
|
|
197
|
-
this._ready = true;
|
|
198
|
-
return;
|
|
199
|
-
}
|
|
200
|
-
|
|
201
|
-
if (data instanceof Blob) {
|
|
202
|
-
this._source = Buffer.from(await data.arrayBuffer());
|
|
203
|
-
this.size = data.size;
|
|
204
|
-
this.mimetype = data.type;
|
|
205
|
-
this._ready = true;
|
|
206
|
-
return;
|
|
207
|
-
}
|
|
208
|
-
|
|
209
|
-
|
|
210
|
-
}
|
|
211
|
-
|
|
212
|
-
private async getUrlInfo(url) {
|
|
213
|
-
try {
|
|
214
|
-
// Before we had axios.head(), head method does not work for all URLs
|
|
215
|
-
const response = await axios.get(url);
|
|
216
|
-
const contentType = response.headers['content-type'];
|
|
217
|
-
const contentLength = response.headers['content-length'];
|
|
218
|
-
return { contentType, contentLength };
|
|
219
|
-
} catch (error) {
|
|
220
|
-
return { contentType: '', contentLength: 0 };
|
|
221
|
-
}
|
|
222
|
-
}
|
|
223
|
-
|
|
224
|
-
public static from(data, name?: string, mimetype?: string, candidate?: IAccessCandidate) {
|
|
225
|
-
if (data instanceof BinaryInput) return data;
|
|
226
|
-
return new BinaryInput(data, name, mimetype, candidate);
|
|
227
|
-
}
|
|
228
|
-
|
|
229
|
-
public async upload(candidate: IAccessCandidate, ttl?: number) {
|
|
230
|
-
await this.ready();
|
|
231
|
-
if (this._uploading) return;
|
|
232
|
-
|
|
233
|
-
try {
|
|
234
|
-
this._uploading = true;
|
|
235
|
-
if (!this.url) {
|
|
236
|
-
const accountConnector = ConnectorService.getAccountConnector();
|
|
237
|
-
const teamId = await accountConnector.getCandidateTeam(candidate);
|
|
238
|
-
|
|
239
|
-
this.url = `smythfs://${teamId}.team/${candidate.id}/_temp/${this._name}`;
|
|
240
|
-
await SmythFS.Instance.write(this.url, this._source, candidate, undefined, ttl);
|
|
241
|
-
this._uploading = false;
|
|
242
|
-
}
|
|
243
|
-
else {
|
|
244
|
-
this._uploading = false;
|
|
245
|
-
}
|
|
246
|
-
} catch (error) {
|
|
247
|
-
console.error('Error uploading binary data:', error);
|
|
248
|
-
this._uploading = false;
|
|
249
|
-
}
|
|
250
|
-
}
|
|
251
|
-
|
|
252
|
-
public async getJsonData(candidate: IAccessCandidate, ttl?: number) {
|
|
253
|
-
await this.upload(candidate, ttl);
|
|
254
|
-
return {
|
|
255
|
-
mimetype: this.mimetype,
|
|
256
|
-
size: this.size,
|
|
257
|
-
url: this.url,
|
|
258
|
-
name: this._name,
|
|
259
|
-
};
|
|
260
|
-
}
|
|
261
|
-
|
|
262
|
-
public async readData(candidate: IAccessCandidate) {
|
|
263
|
-
await this.ready();
|
|
264
|
-
if (!this.url) {
|
|
265
|
-
throw new Error('Binary data not ready');
|
|
266
|
-
}
|
|
267
|
-
const data = await SmythFS.Instance.read(this.url, candidate);
|
|
268
|
-
return data;
|
|
269
|
-
}
|
|
270
|
-
|
|
271
|
-
public async getName() {
|
|
272
|
-
await this.ready();
|
|
273
|
-
return this._name;
|
|
274
|
-
}
|
|
275
|
-
|
|
276
|
-
public async getBuffer() {
|
|
277
|
-
await this.ready();
|
|
278
|
-
|
|
279
|
-
return this._source;
|
|
280
|
-
}
|
|
281
|
-
|
|
282
|
-
/**
|
|
283
|
-
* Creates a read stream from the binary data.
|
|
284
|
-
* Uses temporary files to reduce memory usage for large files.
|
|
285
|
-
*/
|
|
286
|
-
public async getReadStream(): Promise<Readable> {
|
|
287
|
-
await this.ready();
|
|
288
|
-
|
|
289
|
-
// * Direct in-memory stream; faster but uses memory for the entire file.
|
|
290
|
-
return Readable.from(this._source);
|
|
291
|
-
|
|
292
|
-
// * Uses temp files (slower but memory-efficient). Disabled in favor of simpler in-memory approach.
|
|
293
|
-
// let tempDir: string;
|
|
294
|
-
|
|
295
|
-
// try {
|
|
296
|
-
// tempDir = path.join(config.env.DATA_PATH, 'tmp');
|
|
297
|
-
// // Test if we can access/write to this directory
|
|
298
|
-
// if (!fs.existsSync(tempDir)) {
|
|
299
|
-
// fs.mkdirSync(tempDir, { recursive: true });
|
|
300
|
-
// }
|
|
301
|
-
// } catch (error) {
|
|
302
|
-
// // Last resort: try to use an in-memory stream instead of a temp file
|
|
303
|
-
// console.warn('Failed to access temporary directories:', error, error);
|
|
304
|
-
// return Readable.from(this._source);
|
|
305
|
-
// }
|
|
306
|
-
|
|
307
|
-
// // Generate a unique temp file name
|
|
308
|
-
// const tempFilePath = path.join(tempDir, `${Date.now()}-${this._name || uid()}`);
|
|
309
|
-
|
|
310
|
-
// try {
|
|
311
|
-
// // Write the buffer to a temp file
|
|
312
|
-
// fs.writeFileSync(tempFilePath, this._source);
|
|
313
|
-
|
|
314
|
-
// // Create cleanup handler to remove temp file when stream ends
|
|
315
|
-
// const stream = fs.createReadStream(tempFilePath);
|
|
316
|
-
// stream.on('close', () => {
|
|
317
|
-
// try {
|
|
318
|
-
// fs.unlinkSync(tempFilePath);
|
|
319
|
-
// } catch (e) {
|
|
320
|
-
// console.warn('Failed to clean up temporary file:', tempFilePath, e);
|
|
321
|
-
// }
|
|
322
|
-
// });
|
|
323
|
-
|
|
324
|
-
// return stream;
|
|
325
|
-
// } catch (writeError) {
|
|
326
|
-
// // If we can't write to the temp file, fall back to an in-memory stream
|
|
327
|
-
// console.warn('Failed to write temporary file:', writeError);
|
|
328
|
-
// return Readable.from(this._source);
|
|
329
|
-
// }
|
|
330
|
-
}
|
|
331
|
-
}
|
|
1
|
+
import axios from 'axios';
|
|
2
|
+
import mime from 'mime';
|
|
3
|
+
import { fileTypeFromBuffer } from 'file-type';
|
|
4
|
+
import { Readable } from 'stream';
|
|
5
|
+
import fs from 'fs';
|
|
6
|
+
import path from 'path';
|
|
7
|
+
import os from 'os';
|
|
8
|
+
|
|
9
|
+
import { ConnectorService } from '@sre/Core/ConnectorsService';
|
|
10
|
+
import { SmythFS } from '@sre/IO/Storage.service/SmythFS.class';
|
|
11
|
+
import { IAccessCandidate } from '@sre/types/ACL.types';
|
|
12
|
+
import { getSizeFromBinary, isUrl, uid, getBase64FileInfo, getMimeType } from '@sre/utils';
|
|
13
|
+
import config from '@sre/config';
|
|
14
|
+
export class BinaryInput {
|
|
15
|
+
private size: number;
|
|
16
|
+
private url: string;
|
|
17
|
+
private _ready;
|
|
18
|
+
private _readyPromise;
|
|
19
|
+
private _source: Buffer;
|
|
20
|
+
private _uploading: boolean = false;
|
|
21
|
+
|
|
22
|
+
constructor(
|
|
23
|
+
data: BinaryInput | Buffer | ArrayBuffer | Blob | string | Record<string, any>,
|
|
24
|
+
private _name?: string,
|
|
25
|
+
public mimetype?: string,
|
|
26
|
+
private candidate?: IAccessCandidate,
|
|
27
|
+
) {
|
|
28
|
+
if (!_name) _name = uid();
|
|
29
|
+
this._name = _name;
|
|
30
|
+
//this._source = data;
|
|
31
|
+
|
|
32
|
+
this.load(data, _name, mimetype, candidate);
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
public async ready() {
|
|
36
|
+
if (this._ready) return true;
|
|
37
|
+
|
|
38
|
+
if (!this._readyPromise) {
|
|
39
|
+
this._readyPromise = new Promise((resolve) => {
|
|
40
|
+
const maxWait = 10000;
|
|
41
|
+
const interval = setInterval(() => {
|
|
42
|
+
if (this._ready) {
|
|
43
|
+
clearInterval(interval);
|
|
44
|
+
resolve(true);
|
|
45
|
+
}
|
|
46
|
+
if (maxWait <= 0) {
|
|
47
|
+
clearInterval(interval);
|
|
48
|
+
resolve(false);
|
|
49
|
+
}
|
|
50
|
+
}, 100);
|
|
51
|
+
});
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
return this._readyPromise;
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
private async load(data, name: string, mimetype?: string, candidate?: IAccessCandidate) {
|
|
58
|
+
//assume the mimetype from the provided name
|
|
59
|
+
const ext: any = name.split('.')?.length > 1 ? name.split('.').pop() : '';
|
|
60
|
+
// Need to set mimetype empty string if no extension is found, setting default mimetype to 'application/octet-stream' lead wrong direction when try to get the mimetype from base64 or buffer data (as it's not accurate all the time)
|
|
61
|
+
this.mimetype = mimetype || mime.getType(ext) || '';
|
|
62
|
+
this.url = ``;
|
|
63
|
+
|
|
64
|
+
if (typeof data === 'object' && data.url && data.mimetype && data.size) {
|
|
65
|
+
this.mimetype = data.mimetype;
|
|
66
|
+
this.size = data.size;
|
|
67
|
+
this.url = data.url;
|
|
68
|
+
|
|
69
|
+
const ext = mime.getExtension(this.mimetype);
|
|
70
|
+
if (!this._name.endsWith(`.${ext}`)) this._name += `.${ext}`;
|
|
71
|
+
|
|
72
|
+
if (candidate) {
|
|
73
|
+
this._source = await SmythFS.Instance.read(this.url, candidate).finally(() => {
|
|
74
|
+
this._ready = true;
|
|
75
|
+
});
|
|
76
|
+
} else {
|
|
77
|
+
this._ready = true;
|
|
78
|
+
}
|
|
79
|
+
return;
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
if (typeof data === 'string' && data.startsWith('smythfs://')) {
|
|
83
|
+
this.url = data;
|
|
84
|
+
if (candidate) {
|
|
85
|
+
try {
|
|
86
|
+
this._source = await SmythFS.Instance.read(this.url, candidate);
|
|
87
|
+
this.mimetype = await getMimeType(this._source);
|
|
88
|
+
this.size = this._source.byteLength;
|
|
89
|
+
|
|
90
|
+
if (!this.mimetype) {
|
|
91
|
+
this.mimetype = mime.getType(this.url) || mime.getType(this._name) || '';
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
const ext = mime.getExtension(this.mimetype);
|
|
95
|
+
if (!this._name.endsWith(`.${ext}`)) this._name += `.${ext}`;
|
|
96
|
+
} finally {
|
|
97
|
+
this._ready = true;
|
|
98
|
+
}
|
|
99
|
+
} else {
|
|
100
|
+
this._ready = true;
|
|
101
|
+
}
|
|
102
|
+
return;
|
|
103
|
+
}
|
|
104
|
+
|
|
105
|
+
if (isUrl(data)) {
|
|
106
|
+
try {
|
|
107
|
+
const response = await axios({
|
|
108
|
+
method: 'get',
|
|
109
|
+
url: data,
|
|
110
|
+
responseType: 'arraybuffer', // Important for handling binary data
|
|
111
|
+
});
|
|
112
|
+
|
|
113
|
+
this.size = response.data.byteLength;
|
|
114
|
+
this._source = Buffer.from(response.data, 'binary');
|
|
115
|
+
|
|
116
|
+
let mimetype = response.headers?.['content-type'] || '';
|
|
117
|
+
|
|
118
|
+
// Try to get the file extension from the URL
|
|
119
|
+
const urlPath = new URL(data).pathname;
|
|
120
|
+
let extension = urlPath.split('.')?.length > 1 ? urlPath.split('.').pop() : '';
|
|
121
|
+
|
|
122
|
+
// When content-type header is missing or invalid, try to determine mime type from file extension
|
|
123
|
+
if (!mimetype || mimetype.startsWith('binary/octet-stream')) {
|
|
124
|
+
mimetype = extension ? mime.getType(extension) : '';
|
|
125
|
+
}
|
|
126
|
+
|
|
127
|
+
// If we couldn't determine the mime type from headers or file extension, try detecting it from the actual file content
|
|
128
|
+
if (!mimetype) {
|
|
129
|
+
const type = await fileTypeFromBuffer(this._source);
|
|
130
|
+
mimetype = type?.mime || '';
|
|
131
|
+
}
|
|
132
|
+
|
|
133
|
+
this.mimetype = mimetype;
|
|
134
|
+
|
|
135
|
+
extension = extension || mime.getExtension(this.mimetype);
|
|
136
|
+
if (!this._name.endsWith(`.${extension}`)) this._name += `.${extension}`;
|
|
137
|
+
} catch (error) {
|
|
138
|
+
console.error('Error loading binary data from url:', data.url, error);
|
|
139
|
+
}
|
|
140
|
+
|
|
141
|
+
//this._source = data.url;
|
|
142
|
+
|
|
143
|
+
this._ready = true;
|
|
144
|
+
return;
|
|
145
|
+
}
|
|
146
|
+
|
|
147
|
+
// console.log('>>>>>>>>>>>>>>>>>>> is base64 file ?', isBase64DataUrl(data));
|
|
148
|
+
const base64FileInfo = await getBase64FileInfo(data);
|
|
149
|
+
if (base64FileInfo) {
|
|
150
|
+
// If the MIME type is already set, it's safe to use it,
|
|
151
|
+
// as determining the MIME type from the base64 string is not always accurate, specially when it's not a base64 URL
|
|
152
|
+
if (!this.mimetype) {
|
|
153
|
+
this.mimetype = base64FileInfo.mimetype;
|
|
154
|
+
}
|
|
155
|
+
this.size = base64FileInfo.size;
|
|
156
|
+
this._source = Buffer.from(base64FileInfo.data, 'base64');
|
|
157
|
+
|
|
158
|
+
const ext = mime.getExtension(this.mimetype);
|
|
159
|
+
if (!this._name.endsWith(`.${ext}`)) this._name += `.${ext}`;
|
|
160
|
+
|
|
161
|
+
this._ready = true;
|
|
162
|
+
return;
|
|
163
|
+
}
|
|
164
|
+
|
|
165
|
+
if (typeof data === 'string') {
|
|
166
|
+
this._source = Buffer.from(data);
|
|
167
|
+
this.size = data.length;
|
|
168
|
+
this.mimetype = 'text/plain';
|
|
169
|
+
if (!this._name.endsWith(`.txt`)) this._name += `.txt`;
|
|
170
|
+
|
|
171
|
+
this._ready = true;
|
|
172
|
+
return;
|
|
173
|
+
}
|
|
174
|
+
|
|
175
|
+
//this.size = getSizeFromBinary(data);
|
|
176
|
+
// //try to enforce the mimetype from the provided data
|
|
177
|
+
// if (Buffer.isBuffer(data)) {
|
|
178
|
+
// const fileType = await FileType.fileTypeFromBuffer(data);
|
|
179
|
+
// this.mimetype = fileType.mime;
|
|
180
|
+
// const ext = mime.getExtension(this.mimetype);
|
|
181
|
+
// if (!this._name.endsWith(`.${ext}`)) this._name += `.${ext}`;
|
|
182
|
+
|
|
183
|
+
// this._ready = true;
|
|
184
|
+
// return;
|
|
185
|
+
// }
|
|
186
|
+
//try to enforce the mimetype from the provided data
|
|
187
|
+
if (Buffer.isBuffer(data)) {
|
|
188
|
+
this._source = data;
|
|
189
|
+
this.size = getSizeFromBinary(data);
|
|
190
|
+
// If the MIME type is already set, it's safe to use it,
|
|
191
|
+
// as determining the MIME type from the buffer is not always accurate.
|
|
192
|
+
if (!this.mimetype) {
|
|
193
|
+
this.mimetype = await getMimeType(data);
|
|
194
|
+
}
|
|
195
|
+
const ext = mime.getExtension(this.mimetype);
|
|
196
|
+
if (!this._name.endsWith(`.${ext}`)) this._name += `.${ext}`;
|
|
197
|
+
this._ready = true;
|
|
198
|
+
return;
|
|
199
|
+
}
|
|
200
|
+
|
|
201
|
+
if (data instanceof Blob) {
|
|
202
|
+
this._source = Buffer.from(await data.arrayBuffer());
|
|
203
|
+
this.size = data.size;
|
|
204
|
+
this.mimetype = data.type;
|
|
205
|
+
this._ready = true;
|
|
206
|
+
return;
|
|
207
|
+
}
|
|
208
|
+
|
|
209
|
+
|
|
210
|
+
}
|
|
211
|
+
|
|
212
|
+
private async getUrlInfo(url) {
|
|
213
|
+
try {
|
|
214
|
+
// Before we had axios.head(), head method does not work for all URLs
|
|
215
|
+
const response = await axios.get(url);
|
|
216
|
+
const contentType = response.headers['content-type'];
|
|
217
|
+
const contentLength = response.headers['content-length'];
|
|
218
|
+
return { contentType, contentLength };
|
|
219
|
+
} catch (error) {
|
|
220
|
+
return { contentType: '', contentLength: 0 };
|
|
221
|
+
}
|
|
222
|
+
}
|
|
223
|
+
|
|
224
|
+
public static from(data, name?: string, mimetype?: string, candidate?: IAccessCandidate) {
|
|
225
|
+
if (data instanceof BinaryInput) return data;
|
|
226
|
+
return new BinaryInput(data, name, mimetype, candidate);
|
|
227
|
+
}
|
|
228
|
+
|
|
229
|
+
public async upload(candidate: IAccessCandidate, ttl?: number) {
|
|
230
|
+
await this.ready();
|
|
231
|
+
if (this._uploading) return;
|
|
232
|
+
|
|
233
|
+
try {
|
|
234
|
+
this._uploading = true;
|
|
235
|
+
if (!this.url) {
|
|
236
|
+
const accountConnector = ConnectorService.getAccountConnector();
|
|
237
|
+
const teamId = await accountConnector.getCandidateTeam(candidate);
|
|
238
|
+
|
|
239
|
+
this.url = `smythfs://${teamId}.team/${candidate.id}/_temp/${this._name}`;
|
|
240
|
+
await SmythFS.Instance.write(this.url, this._source, candidate, undefined, ttl);
|
|
241
|
+
this._uploading = false;
|
|
242
|
+
}
|
|
243
|
+
else {
|
|
244
|
+
this._uploading = false;
|
|
245
|
+
}
|
|
246
|
+
} catch (error) {
|
|
247
|
+
console.error('Error uploading binary data:', error);
|
|
248
|
+
this._uploading = false;
|
|
249
|
+
}
|
|
250
|
+
}
|
|
251
|
+
|
|
252
|
+
public async getJsonData(candidate: IAccessCandidate, ttl?: number) {
|
|
253
|
+
await this.upload(candidate, ttl);
|
|
254
|
+
return {
|
|
255
|
+
mimetype: this.mimetype,
|
|
256
|
+
size: this.size,
|
|
257
|
+
url: this.url,
|
|
258
|
+
name: this._name,
|
|
259
|
+
};
|
|
260
|
+
}
|
|
261
|
+
|
|
262
|
+
public async readData(candidate: IAccessCandidate) {
|
|
263
|
+
await this.ready();
|
|
264
|
+
if (!this.url) {
|
|
265
|
+
throw new Error('Binary data not ready');
|
|
266
|
+
}
|
|
267
|
+
const data = await SmythFS.Instance.read(this.url, candidate);
|
|
268
|
+
return data;
|
|
269
|
+
}
|
|
270
|
+
|
|
271
|
+
public async getName() {
|
|
272
|
+
await this.ready();
|
|
273
|
+
return this._name;
|
|
274
|
+
}
|
|
275
|
+
|
|
276
|
+
public async getBuffer() {
|
|
277
|
+
await this.ready();
|
|
278
|
+
|
|
279
|
+
return this._source;
|
|
280
|
+
}
|
|
281
|
+
|
|
282
|
+
/**
|
|
283
|
+
* Creates a read stream from the binary data.
|
|
284
|
+
* Uses temporary files to reduce memory usage for large files.
|
|
285
|
+
*/
|
|
286
|
+
public async getReadStream(): Promise<Readable> {
|
|
287
|
+
await this.ready();
|
|
288
|
+
|
|
289
|
+
// * Direct in-memory stream; faster but uses memory for the entire file.
|
|
290
|
+
return Readable.from(this._source);
|
|
291
|
+
|
|
292
|
+
// * Uses temp files (slower but memory-efficient). Disabled in favor of simpler in-memory approach.
|
|
293
|
+
// let tempDir: string;
|
|
294
|
+
|
|
295
|
+
// try {
|
|
296
|
+
// tempDir = path.join(config.env.DATA_PATH, 'tmp');
|
|
297
|
+
// // Test if we can access/write to this directory
|
|
298
|
+
// if (!fs.existsSync(tempDir)) {
|
|
299
|
+
// fs.mkdirSync(tempDir, { recursive: true });
|
|
300
|
+
// }
|
|
301
|
+
// } catch (error) {
|
|
302
|
+
// // Last resort: try to use an in-memory stream instead of a temp file
|
|
303
|
+
// console.warn('Failed to access temporary directories:', error, error);
|
|
304
|
+
// return Readable.from(this._source);
|
|
305
|
+
// }
|
|
306
|
+
|
|
307
|
+
// // Generate a unique temp file name
|
|
308
|
+
// const tempFilePath = path.join(tempDir, `${Date.now()}-${this._name || uid()}`);
|
|
309
|
+
|
|
310
|
+
// try {
|
|
311
|
+
// // Write the buffer to a temp file
|
|
312
|
+
// fs.writeFileSync(tempFilePath, this._source);
|
|
313
|
+
|
|
314
|
+
// // Create cleanup handler to remove temp file when stream ends
|
|
315
|
+
// const stream = fs.createReadStream(tempFilePath);
|
|
316
|
+
// stream.on('close', () => {
|
|
317
|
+
// try {
|
|
318
|
+
// fs.unlinkSync(tempFilePath);
|
|
319
|
+
// } catch (e) {
|
|
320
|
+
// console.warn('Failed to clean up temporary file:', tempFilePath, e);
|
|
321
|
+
// }
|
|
322
|
+
// });
|
|
323
|
+
|
|
324
|
+
// return stream;
|
|
325
|
+
// } catch (writeError) {
|
|
326
|
+
// // If we can't write to the temp file, fall back to an in-memory stream
|
|
327
|
+
// console.warn('Failed to write temporary file:', writeError);
|
|
328
|
+
// return Readable.from(this._source);
|
|
329
|
+
// }
|
|
330
|
+
}
|
|
331
|
+
}
|