forked from firebase/functions-samples
-
Notifications
You must be signed in to change notification settings - Fork 0
/
index.js
99 lines (91 loc) · 3.72 KB
/
index.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
/**
* Copyright 2016 Google Inc. All Rights Reserved.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for t`he specific language governing permissions and
* limitations under the License.
*/
'use strict';
const functions = require('firebase-functions');
const fs = require('fs');
const crypto = require('crypto');
const path = require('path');
const os = require('os');
const admin = require('firebase-admin');
admin.initializeApp();
const gcs = require('@google-cloud/storage')();
const spawn = require('child-process-promise').spawn;
/**
* When an image is uploaded in the Storage bucket the information and metadata of the image (the
* output of ImageMagick's `identify -verbose`) is saved in the Realtime Database.
*/
exports.metadata = functions.storage.object().onFinalize(async (object) => {
const filePath = object.name;
// Create random filename with same extension as uploaded file.
const randomFileName = crypto.randomBytes(20).toString('hex') + path.extname(filePath);
const tempLocalFile = path.join(os.tmpdir(), randomFileName);
// Exit if this is triggered on a file that is not an image.
if (!object.contentType.startsWith('image/')) {
console.log('This is not an image.');
return null;
}
let metadata;
// Download file from bucket.
const bucket = gcs.bucket(object.bucket);
await bucket.file(filePath).download({destination: tempLocalFile});
// Get Metadata from image.
const result = await spawn('identify', ['-verbose', tempLocalFile], {capture: ['stdout', 'stderr']});
// Save metadata to realtime datastore.
metadata = imageMagickOutputToObject(result.stdout);
const safeKey = makeKeyFirebaseCompatible(filePath);
await admin.database().ref(safeKey).set(metadata);
console.log('Wrote to:', filePath, 'data:', metadata);
// Cleanup temp directory after metadata is extracted
// Remove the file from temp directory
await fs.unlinkSync(tempLocalFile)
return console.log('cleanup successful!');
});
/**
* Convert the output of ImageMagick's `identify -verbose` command to a JavaScript Object.
*/
function imageMagickOutputToObject(output) {
let previousLineIndent = 0;
const lines = output.match(/[^\r\n]+/g);
lines.shift(); // Remove First line
lines.forEach((line, index) => {
const currentIdent = line.search(/\S/);
line = line.trim();
if (line.endsWith(':')) {
lines[index] = makeKeyFirebaseCompatible(`"${line.replace(':', '":{')}`);
} else {
const split = line.replace('"', '\\"').split(': ');
split[0] = makeKeyFirebaseCompatible(split[0]);
lines[index] = `"${split.join('":"')}",`;
}
if (currentIdent < previousLineIndent) {
lines[index - 1] = lines[index - 1].substring(0, lines[index - 1].length - 1);
lines[index] = new Array(1 + (previousLineIndent - currentIdent) / 2).join('}') + ',' + lines[index];
}
previousLineIndent = currentIdent;
});
output = lines.join('');
output = '{' + output.substring(0, output.length - 1) + '}'; // remove trailing comma.
output = JSON.parse(output);
console.log('Metadata extracted from image', output);
return output;
}
/**
* Makes sure the given string does not contain characters that can't be used as Firebase
* Realtime Database keys such as '.' and replaces them by '*'.
*/
function makeKeyFirebaseCompatible(key) {
return key.replace(/\./g, '*');
}