#node.js #react-native #expo #multer
#node.js #react-native #expo #multer
Вопрос:
Как я могу загрузить файл (pdf, документы и т. Д.) С React Native с помощью expo На сервер с помощью node. Я видел много примеров изображений с использованием API-интерфейса expo image-picker, но я не встречал ни одного, который использовал бы API-интерфейсы document-picker или filesystem из expo. Документация файловой системы expo была немного сложной для интерпретации для такого новичка, как я.
Ответ №1:
Спасибо за помощь. Я смог придумать решение, и я опубликую его ниже, чтобы оно могло пригодиться тем, кто придет сюда в будущем.
React Native
import React, { useState } from 'react';
import { Button, View } from 'react-native';
import * as DocumentPicker from 'expo-document-picker';
import * as FileSystem from 'expo-file-system';
const DocPicker = () => {
const [ doc, setDoc ] = useState();
const pickDocument = async () => {
let result = await DocumentPicker.getDocumentAsync({ type: "*/*", copyToCacheDirectory: true }).then(response => {
if (response.type == 'success') {
let { name, size, uri } = response;
let nameParts = name.split('.');
let fileType = nameParts[nameParts.length - 1];
var fileToUpload = {
name: name,
size: size,
uri: uri,
type: "application/" fileType
};
console.log(fileToUpload, '...............file')
setDoc(fileToUpload);
}
});
// console.log(result);
console.log("Doc: " doc.uri);
}
const postDocument = () => {
const url = "http://192.168.10.107:8000/upload";
const fileUri = doc.uri;
const formData = new FormData();
formData.append('document', doc);
const options = {
method: 'POST',
body: formData,
headers: {
Accept: 'application/json',
'Content-Type': 'multipart/form-data',
},
};
console.log(formData);
fetch(url, options).catch((error) => console.log(error));
}
return (
<View>
<Button title="Select Document" onPress={pickDocument} />
<Button title="Upload" onPress={postDocument} />
</View>
)
};
export default DocPicker;
Node.js
const express = require('express')
const bodyParser = require('body-parser')
var multer = require('multer')
var upload = multer({ dest: 'uploads/' })
const app = express()
const fs = require('fs')
const http = require('http')
const port = 8000
app.use(bodyParser.json());
app.use(bodyParser.urlencoded({ extended: true }));
app.get('/', (req,res) => {
res.json({
success: true
})
})
app.post('/', (req, res) => {
console.log(req.body)
res.status(200)
})
app.post('/upload', upload.single('document'),(req , res) => {
console.log(req.file, req.body)
});
app.listen(port, () => {
console.log(`Example app listening at http://localhost:${port}`)
})
Приветствия!!!
Комментарии:
1. Это решение было правильным для меня, большое вам спасибо !! 🚀
Ответ №2:
Комментарии:
1. Эй, я попробовал это, и это выдало мне пару ошибок
Ответ №3:
Вот пример, который также использует multer и express на серверной части: https://github.com/expo/examples/tree/master/with-formdata-image-upload
Тем не менее, я бы рекомендовал использовать FileSystem.uploadAsync
вместо fetch
и в фоновом sessionType
режиме, чтобы поддерживать загрузку, пока приложение работает в фоновом режиме на iOS.
Комментарии:
1. Привет, в следующем примере используется средство выбора изображений, и, как вы уже упоминали, я хочу использовать FileSystem или DocumentPicker. Есть ли какие-либо примеры с любым из двух? Если да, это поможет мне решить! Спасибо!
Ответ №4:
Если решение, предложенное @Anandhu, не работает, попробуйте приведенный выше код следующим образом.
import React, { useState } from 'react';
import { Button, View } from 'react-native';
import * as DocumentPicker from 'expo-document-picker';
import * as FileSystem from 'expo-file-system';
const DocPicker = () => {
const [ doc, setDoc ] = useState();
const pickDocument = async () => {
let result = await DocumentPicker.getDocumentAsync({
type: "*/*",
copyToCacheDirectory: true })
.then(response => {
if (response.type == 'success') {
let { name, size, uri } = response;
/ ------------------------/
if (Platform.OS === "android" amp;amp; uri[0] === "/") {
uri = `file://${uri}`;
uri = uri.replace(/%/g, "%");
}
/ ------------------------/
let nameParts = name.split('.');
let fileType = nameParts[nameParts.length - 1];
var fileToUpload = {
name: name,
size: size,
uri: uri,
type: "application/" fileType
};
console.log(fileToUpload, '...............file')
setDoc(fileToUpload);
}
});
// console.log(result);
console.log("Doc: " doc.uri);
}
const postDocument = () => {
const url = "http://192.168.10.107:8000/upload";
const fileUri = doc.uri;
const formData = new FormData();
formData.append('document', doc);
const options = {
method: 'POST',
body: formData,
headers: {
Accept: 'application/json',
'Content-Type': 'multipart/form-data',
},
};
console.log(formData);
fetch(url, options).catch((error) => console.log(error));
}
return (
<View>
<Button title="Select Document" onPress={pickDocument} />
<Button title="Upload" onPress={postDocument} />
</View>
)
};
export default DocPicker;
Существует ошибка в способе кодирования пути, и file:// scheme отсутствует.
Эта ошибка может быть исправлена в следующем выпуске.