Skip to main content

zip photos and download from react app -


  • library: 

jszip - const zip = jszip()

file-saver - saveAs(content, `${fetchListingId}.zip`)

issue: cors

the original plan was download photos from the front end and zip with jszip, but got cors error when download from firebase storage

solutions:

1. on graphql, add query: getPhotosBuffer(listingId: String): [PhotoBuffer] to return stringified buffer of photos

fetchPhotosBuffer = async (listingId: string): Promise<PhotoBuffer[]> =>
this.cacheProvider.processor<PhotoBuffer[]>(
async () => {
const photos = await this.collection
.where('listingId', '==', listingId)
.orderBy('orders')
.get()

const result: PhotoBuffer[] = []
await Promise.all(
photos.docs.map(async (photo: any) => {
const data = photo.data()
const { photoUrl } = data
try {
const buffer = await axios
.get(photoUrl, {
responseType: 'arraybuffer',
})
.then(response => Buffer.from(response.data, 'binary'))

result.push({ id: photo.id, data: buffer.toString() })
} catch (err) {
console.log(err)
}
})
)

return result
},
{ key: `${this.cachePrefix}-fetchPhotos-${listingId}` }
)

2. React - parse the stringified buffer to get data

useEffect(() => {
const fetchPhotos = async () => {
const result = await client.query({
query: QUERY_LISTING_PHOTOS_BUFFER,
variables: { listingId: fetchListingId },
})

const {
data: { getPhotosBuffer },
} = result

getPhotosBuffer.forEach((photo: any) => {
zip.file(`${photo.id}.jpg`, JSON.parse(photo.data).data, {
binary: true,
})
})

zip
.generateAsync({
type: 'blob',
})
.then(content => {
saveAs(content, `${fetchListingId}.zip`)
})
.finally(() => setFetchListingId(''))
}
if (fetchListingId) {
fetchPhotos()
}
}, [fetchListingId, client, zip])

Comments

Popular posts from this blog

for loop in javascript - promise - .eslintrc for "for of"

the vast majority of cases  map ,  forEach ,  find  etc. can be used.  async function printFiles () { const files = await getFilePaths(); await Promise.all(files. map (async (file) => { const contents = await fs.readFile(file, 'utf8') console.log(contents) })); } const inventory = [ { name : 'apples' , quantity : 2 } , { name : 'bananas' , quantity : 0 } , { name : 'cherries' , quantity : 5 } ] ; const result = inventory . find ( ( { name } ) => name === 'cherries' ) ;   function getFirstMatching(array) { for (let item of array) { const result = heavyTransform(item); if (result) { return result; } } } Specifically this shuts down the whole no-restricted-syntax. If you want to cherry-pick, here is the current definition: 'no-restricted-syntax' : [ 'error' , { selector : 'ForInStatement' , message...

fs.writeFile - permission issue on GCP (Google Cloud Run)

error on this: fs . writeFile ( fileName , pdfBytes , err => { if ( err ) { console . log ( err . message ) reject ( err . message ) } else resolve ( fileName ) }) #1 It looks like when deployed into Cloud Run it also requires the extra permission "Service Account Token Creator" to run  getSignedUrl . Locally for some reason this role is not required. #2 Only the directory  /tmp  is writable in Cloud Run. So, change the default write location to write into this directory. However, you have to be aware of 2 things: Cloud Run is stateless, that means when a new instance is created, the container start from scratch, with an empty  /tmp  directory /tmp  directory is an in-memory file system. The maximum allowed memory on Cloud Run is 2Gb, your app memory footprint included. In addition of your file and Airflow, not sure that you will have a lot of space. A final remark. Cloud Run is active only when it...

add new site to Vagrant Homestead

Homestead.yaml: folders : - map : ~/ freshinup to : / home / vagrant / code sites : - map : smartmotors . local . com to : / home / vagrant / code / smartmotors / public php : "7.3" - map : arbor . test to : / home / vagrant / code / arbor / public php : "7.3" /etc/hosts: 192.168.10.10 arbor.test 192.168.10.10 smartmotors.local.com vagrant up --provision php artisan smartmotors:seed --quickstart