Skip to content

Commit 2c6f3f9

Browse files
committed
feat: support react-native asset file input
1 parent 36c4c25 commit 2c6f3f9

27 files changed

+360
-141
lines changed

packages/upload-client/src/UploadClient.ts

+5-5
Original file line numberDiff line numberDiff line change
@@ -25,7 +25,7 @@ import { UploadcareGroup } from './tools/UploadcareGroup'
2525
import { FileFromOptions, uploadFile } from './uploadFile'
2626

2727
import { FileInfo, GroupId, GroupInfo, Token, Url, Uuid } from './api/types'
28-
import { Settings, BrowserFile, NodeFile } from './types'
28+
import { SupportedFileInput, BrowserFile, NodeFile, Settings } from './types'
2929
import uploadFileGroup, { GroupFromOptions } from './uploadFileGroup'
3030

3131
/**
@@ -55,7 +55,7 @@ export default class UploadClient {
5555
}
5656

5757
base(
58-
file: NodeFile | BrowserFile,
58+
file: SupportedFileInput,
5959
options: Partial<BaseOptions> = {}
6060
): Promise<BaseResponse> {
6161
const settings = this.getSettings()
@@ -115,7 +115,7 @@ export default class UploadClient {
115115
}
116116

117117
multipartUpload(
118-
part: Buffer | Blob,
118+
part: BrowserFile | NodeFile,
119119
url: MultipartPart,
120120
options: Partial<MultipartUploadOptions> = {}
121121
): Promise<MultipartUploadResponse> {
@@ -141,7 +141,7 @@ export default class UploadClient {
141141
}
142142

143143
uploadFile(
144-
data: NodeFile | BrowserFile | Url | Uuid,
144+
data: SupportedFileInput | Url | Uuid,
145145
options: Partial<FileFromOptions> = {}
146146
): Promise<UploadcareFile> {
147147
const settings = this.getSettings()
@@ -150,7 +150,7 @@ export default class UploadClient {
150150
}
151151

152152
uploadFileGroup(
153-
data: (NodeFile | BrowserFile)[] | Url[] | Uuid[],
153+
data: SupportedFileInput[] | Url[] | Uuid[],
154154
options: Partial<FileFromOptions & GroupFromOptions> = {}
155155
): Promise<UploadcareGroup> {
156156
const settings = this.getSettings()

packages/upload-client/src/api/base.ts

+6-9
Original file line numberDiff line numberDiff line change
@@ -1,22 +1,19 @@
1+
import { camelizeKeys, CustomUserAgent } from '@uploadcare/api-client-utils'
2+
import { defaultSettings } from '../defaultSettings'
13
import request from '../request/request.node'
24
import buildFormData from '../tools/buildFormData'
5+
import { UploadClientError } from '../tools/errors'
36
import getUrl from '../tools/getUrl'
4-
import {
5-
defaultSettings,
6-
defaultFilename,
7-
defaultContentType
8-
} from '../defaultSettings'
97
import { getUserAgent } from '../tools/getUserAgent'
10-
import { camelizeKeys, CustomUserAgent } from '@uploadcare/api-client-utils'
11-
import { UploadClientError } from '../tools/errors'
128
import { retryIfFailed } from '../tools/retryIfFailed'
139

1410
/* Types */
15-
import { Uuid, ProgressCallback, Metadata } from './types'
1611
import { FailedResponse } from '../request/types'
1712
import { getContentType } from '../tools/getContentType'
1813
import { getFileName } from '../tools/getFileName'
1914
import { getStoreValue } from '../tools/getStoreValue'
15+
import { SupportedFileInput } from '../types'
16+
import { Metadata, ProgressCallback, Uuid } from './types'
2017

2118
export type BaseResponse = {
2219
file: Uuid
@@ -51,7 +48,7 @@ export type BaseOptions = {
5148
* Can be canceled and has progress.
5249
*/
5350
export default function base(
54-
file: NodeFile | BrowserFile,
51+
file: SupportedFileInput,
5552
{
5653
publicKey,
5754
fileName,

packages/upload-client/src/api/multipartUpload.ts

+4-4
Original file line numberDiff line numberDiff line change
@@ -2,10 +2,10 @@ import { MultipartPart } from './multipartStart'
22

33
import request from '../request/request.node'
44

5-
import { ComputableProgressInfo, ProgressCallback } from './types'
6-
import { NodeFile, BrowserFile } from '../types'
7-
import { retryIfFailed } from '../tools/retryIfFailed'
85
import defaultSettings from '../defaultSettings'
6+
import { retryIfFailed } from '../tools/retryIfFailed'
7+
import { SupportedFileInput } from '../types'
8+
import { ComputableProgressInfo, ProgressCallback } from './types'
99

1010
export type MultipartUploadOptions = {
1111
publicKey?: string
@@ -25,7 +25,7 @@ export type MultipartUploadResponse = {
2525
*/
2626

2727
export default function multipartUpload(
28-
part: NodeFile | BrowserFile,
28+
part: SupportedFileInput,
2929
url: MultipartPart,
3030
{
3131
signal,

packages/upload-client/src/index.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -32,7 +32,7 @@ export { UploadcareFile } from './tools/UploadcareFile'
3232
export { UploadcareGroup } from './tools/UploadcareGroup'
3333
export { UploadClientError } from './tools/errors'
3434
export { Settings } from './types'
35-
export { NodeFile, BrowserFile } from './types'
35+
export { NodeFile, BrowserFile, ReactNativeAsset } from './types'
3636
export { BaseOptions, BaseResponse } from './api/base'
3737
export { FileInfo, GroupId, GroupInfo, Token, Url, Uuid } from './api/types'
3838
export { InfoOptions } from './api/info'

packages/upload-client/src/request/request.node.ts

+6-5
Original file line numberDiff line numberDiff line change
@@ -2,12 +2,13 @@ import NodeFormData from 'form-data'
22

33
import http from 'http'
44
import https from 'https'
5-
import { parse } from 'url'
65
import { Readable, Transform } from 'stream'
6+
import { parse } from 'url'
77

8-
import { onCancel, CancelError } from '@uploadcare/api-client-utils'
9-
import { RequestOptions, RequestResponse } from './types'
8+
import { CancelError, onCancel } from '@uploadcare/api-client-utils'
109
import { ProgressCallback } from '../api/types'
10+
import { SupportedFileInput } from '../types'
11+
import { RequestOptions, RequestResponse } from './types'
1112

1213
// ProgressEmitter is a simple PassThrough-style transform stream which keeps
1314
// track of the number of bytes which have been piped through it and will
@@ -44,7 +45,7 @@ const getLength = (formData: NodeFormData): Promise<number> =>
4445
})
4546

4647
function isFormData(
47-
formData?: NodeFormData | FormData | Buffer | Blob
48+
formData?: NodeFormData | FormData | SupportedFileInput
4849
): formData is NodeFormData {
4950
if (formData && formData.toString() === '[object FormData]') {
5051
return true
@@ -54,7 +55,7 @@ function isFormData(
5455
}
5556

5657
function isReadable(
57-
data?: Readable | NodeFormData | FormData | Buffer | Blob,
58+
data?: Readable | NodeFormData | FormData | SupportedFileInput,
5859
isFormData?: boolean
5960
): data is Readable {
6061
if (data && (data instanceof Readable || isFormData)) {

packages/upload-client/src/request/types.ts

+3-3
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,7 @@ import {
44
ProgressCallback,
55
UnknownProgressInfo
66
} from '../api/types'
7-
import { BrowserFile, NodeFile } from '../types'
7+
import { SupportedFileInput } from '../types'
88

99
export type Headers = {
1010
[key: string]: string | string[] | undefined
@@ -14,7 +14,7 @@ export type RequestOptions = {
1414
method?: string
1515
url: string
1616
query?: string
17-
data?: NodeFormData | FormData | BrowserFile | NodeFile
17+
data?: NodeFormData | FormData | SupportedFileInput
1818
headers?: Headers
1919
signal?: AbortSignal
2020
onProgress?: ProgressCallback<ComputableProgressInfo | UnknownProgressInfo>
@@ -24,7 +24,7 @@ export type ErrorRequestInfo = {
2424
method?: string
2525
url: string
2626
query?: string
27-
data?: NodeFormData | FormData | BrowserFile | NodeFile
27+
data?: NodeFormData | FormData | SupportedFileInput
2828
headers?: Headers
2929
}
3030

packages/upload-client/src/tools/buildFormData.ts

+4-4
Original file line numberDiff line numberDiff line change
@@ -1,8 +1,8 @@
11
import getFormData, { getFileOptions, transformFile } from './getFormData.node'
22

33
import NodeFormData from 'form-data'
4-
import { BrowserFile, NodeFile } from '../types'
5-
import { isFileData } from '../uploadFile/types'
4+
import { SupportedFileInput } from '../types'
5+
import { isFileData } from './isFileData'
66

77
/**
88
* Constructs FormData instance.
@@ -21,7 +21,7 @@ interface FileOptions {
2121
}
2222

2323
interface FileType extends FileOptions {
24-
data: BrowserFile | NodeFile
24+
data: SupportedFileInput
2525
}
2626

2727
type InputValue = FileType | SimpleType | ObjectType
@@ -33,7 +33,7 @@ type FormDataOptions = {
3333
type Params = Array<
3434
[
3535
string,
36-
string | BrowserFile | NodeFile,
36+
string | SupportedFileInput,
3737
...(string | { [key: string]: string | undefined })[]
3838
]
3939
>
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
import { ReactNativeAsset } from '../types'
2+
3+
const memo: WeakMap<ReactNativeAsset, Blob> = new WeakMap()
4+
5+
export const getBlobFromReactNativeAsset = async (
6+
asset: ReactNativeAsset
7+
): Promise<Blob> => {
8+
if (memo.has(asset)) {
9+
return memo.get(asset) as Blob
10+
}
11+
const blob = await fetch(asset.uri).then((res) => res.blob())
12+
memo.set(asset, blob)
13+
return blob
14+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,17 @@
1+
import { SupportedFileInput } from '../types'
2+
import { getBlobFromReactNativeAsset } from './getBlobFromReactNativeAsset'
3+
import { isBlob, isBuffer, isFile, isReactNativeAsset } from './isFileData'
4+
5+
export const getFileSize = async (file: SupportedFileInput) => {
6+
if (isBuffer(file)) {
7+
return file.length
8+
}
9+
if (isFile(file) || isBlob(file)) {
10+
return file.size
11+
}
12+
if (isReactNativeAsset(file)) {
13+
const blob = await getBlobFromReactNativeAsset(file)
14+
return blob.size
15+
}
16+
throw new Error(`Unknown file type. Cannot determine file size.`)
17+
}
Original file line numberDiff line numberDiff line change
@@ -1,19 +1,26 @@
1-
import { BrowserFile, NodeFile } from '../types'
2-
import {
3-
FileTransformer,
4-
GetFormDataFileAppendOptions,
5-
ReactNativeAsset
6-
} from './types'
1+
import { SupportedFileInput, ReactNativeAsset } from '../types'
2+
import { isBlob, isReactNativeAsset } from './isFileData'
3+
import { FileTransformer, GetFormDataFileAppendOptions } from './types'
74

85
export const getFileOptions: GetFormDataFileAppendOptions = () => []
96

107
export const transformFile: FileTransformer = (
11-
file: BrowserFile | NodeFile,
8+
file: SupportedFileInput,
129
name: string,
1310
contentType: string
1411
): ReactNativeAsset => {
15-
const uri = URL.createObjectURL(file as BrowserFile)
16-
return { uri, name, type: contentType }
12+
if (isReactNativeAsset(file)) {
13+
return {
14+
uri: file.uri,
15+
name: file.name || name,
16+
type: file.type || contentType
17+
}
18+
}
19+
if (isBlob(file)) {
20+
const uri = URL.createObjectURL(file)
21+
return { uri, name: name, type: file.type || contentType }
22+
}
23+
throw new Error(`Unsupported file type.`)
1724
}
1825

1926
export default (): FormData => new FormData()
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,29 @@
1+
import { SupportedFileInput, ReactNativeAsset } from '../types'
2+
3+
export const isBlob = (data: unknown): data is Blob => {
4+
return typeof Blob !== 'undefined' && data instanceof Blob
5+
}
6+
7+
export const isFile = (data: unknown): data is File => {
8+
return typeof File !== 'undefined' && data instanceof File
9+
}
10+
11+
export const isBuffer = (data: unknown): data is Buffer => {
12+
return typeof Buffer !== 'undefined' && data instanceof Buffer
13+
}
14+
15+
export const isReactNativeAsset = (data: unknown): data is ReactNativeAsset => {
16+
return (
17+
!!data &&
18+
typeof data === 'object' &&
19+
!Array.isArray(data) &&
20+
'uri' in data &&
21+
typeof (data as Record<'uri', unknown>).uri === 'string'
22+
)
23+
}
24+
25+
export const isFileData = (data: unknown): data is SupportedFileInput => {
26+
return (
27+
isBlob(data) || isFile(data) || isBuffer(data) || isReactNativeAsset(data)
28+
)
29+
}

packages/upload-client/src/tools/isMultipart.ts

-10
Original file line numberDiff line numberDiff line change
@@ -1,15 +1,5 @@
11
import defaultSettings from '../defaultSettings'
22

3-
/* Types */
4-
import { NodeFile, BrowserFile } from '../types'
5-
6-
/**
7-
* Get file size.
8-
*/
9-
export const getFileSize = (file: NodeFile | BrowserFile): number => {
10-
return (file as Buffer).length || (file as Blob).size
11-
}
12-
133
/**
144
* Check if FileData is multipart data.
155
*/

packages/upload-client/src/tools/types.ts

+3-5
Original file line numberDiff line numberDiff line change
@@ -1,12 +1,10 @@
1-
import { BrowserFile, NodeFile } from '../types'
2-
3-
export type ReactNativeAsset = { type: string; uri: string; name?: string }
1+
import { SupportedFileInput } from '../types'
42

53
export type FileTransformer = (
6-
file: NodeFile | BrowserFile,
4+
file: SupportedFileInput,
75
name: string,
86
contentType: string
9-
) => NodeFile | BrowserFile | ReactNativeAsset
7+
) => SupportedFileInput
108

119
export type GetFormDataFileAppendOptions = (options: {
1210
[key: string]: string | undefined

packages/upload-client/src/types.ts

+10-1
Original file line numberDiff line numberDiff line change
@@ -30,4 +30,13 @@ export interface Settings extends Partial<DefaultSettings> {
3030
}
3131

3232
export type BrowserFile = Blob | File
33-
export type NodeFile = Buffer // | NodeJS.ReadableStream
33+
export type NodeFile = Buffer
34+
export type ReactNativeAsset = {
35+
type: string
36+
uri: string
37+
name?: string
38+
}
39+
export type ReactNativeFile = ReactNativeAsset | Blob
40+
41+
export type SupportedFileInput = BrowserFile | NodeFile | ReactNativeFile
42+
export type AnySlicable = BrowserFile | NodeFile

0 commit comments

Comments
 (0)