├── .deployment ├── .github ├── ISSUE_TEMPLATE.md └── PULL_REQUEST_TEMPLATE.md ├── CHANGELOG.md ├── CONTRIBUTING.md ├── LICENSE.md ├── README.md ├── v10 ├── .env.example ├── .gitignore ├── app.js ├── bin │ └── www ├── config.js ├── package-lock.json ├── package.json ├── public │ ├── favicon.ico │ └── stylesheets │ │ └── style.css ├── routes │ ├── index.js │ └── upload.js └── views │ ├── error.hbs │ ├── index.hbs │ ├── layout.hbs │ └── success.hbs └── v12 ├── .env.example ├── .gitignore ├── app.js ├── bin └── www ├── package-lock.json ├── package.json ├── public ├── favicon.ico └── stylesheets │ └── style.css ├── routes └── index.js └── views ├── error.hbs ├── index.hbs ├── layout.hbs └── success.hbs /.deployment: -------------------------------------------------------------------------------- 1 | [config] 2 | project = v12 -------------------------------------------------------------------------------- /.github/ISSUE_TEMPLATE.md: -------------------------------------------------------------------------------- 1 | 4 | > Please provide us with the following information: 5 | > --------------------------------------------------------------- 6 | 7 | ### This issue is for a: (mark with an `x`) 8 | ``` 9 | - [ ] bug report -> please search issues before submitting 10 | - [ ] feature request 11 | - [ ] documentation issue or request 12 | - [ ] regression (a behavior that used to work and stopped in a new release) 13 | ``` 14 | 15 | ### Minimal steps to reproduce 16 | > 17 | 18 | ### Any log messages given by the failure 19 | > 20 | 21 | ### Expected/desired behavior 22 | > 23 | 24 | ### OS and Version? 25 | > Windows 7, 8 or 10. Linux (which distribution). macOS (Yosemite? El Capitan? Sierra?) 26 | 27 | ### Versions 28 | > 29 | 30 | ### Mention any other details that might be useful 31 | 32 | > --------------------------------------------------------------- 33 | > Thanks! We'll be in touch soon. 34 | -------------------------------------------------------------------------------- /.github/PULL_REQUEST_TEMPLATE.md: -------------------------------------------------------------------------------- 1 | ## Purpose 2 | 3 | * ... 4 | 5 | ## Does this introduce a breaking change? 6 | 7 | ``` 8 | [ ] Yes 9 | [ ] No 10 | ``` 11 | 12 | ## Pull Request Type 13 | What kind of change does this Pull Request introduce? 14 | 15 | 16 | ``` 17 | [ ] Bugfix 18 | [ ] Feature 19 | [ ] Code style update (formatting, local variables) 20 | [ ] Refactoring (no functional changes, no api changes) 21 | [ ] Documentation content changes 22 | [ ] Other... Please describe: 23 | ``` 24 | 25 | ## How to Test 26 | * Get the code 27 | 28 | ``` 29 | git clone [repo-address] 30 | cd [repo-name] 31 | git checkout [branch-name] 32 | npm install 33 | ``` 34 | 35 | * Test the code 36 | 37 | ``` 38 | ``` 39 | 40 | ## What to Check 41 | Verify that the following are valid 42 | * ... 43 | 44 | ## Other Information 45 | -------------------------------------------------------------------------------- /CHANGELOG.md: -------------------------------------------------------------------------------- 1 | ## [project-title] Changelog 2 | 3 | 4 | # x.y.z (yyyy-mm-dd) 5 | 6 | *Features* 7 | * ... 8 | 9 | *Bug Fixes* 10 | * ... 11 | 12 | *Breaking Changes* 13 | * ... 14 | -------------------------------------------------------------------------------- /CONTRIBUTING.md: -------------------------------------------------------------------------------- 1 | # Contributing to this project 2 | 3 | This project welcomes contributions and suggestions. Most contributions require you to agree to a 4 | Contributor License Agreement (CLA) declaring that you have the right to, and actually do, grant us 5 | the rights to use your contribution. For details, visit https://cla.microsoft.com. 6 | 7 | When you submit a pull request, a CLA-bot will automatically determine whether you need to provide 8 | a CLA and decorate the PR appropriately (e.g., label, comment). Simply follow the instructions 9 | provided by the bot. You will only need to do this once across all repos using our CLA. 10 | 11 | This project has adopted the [Microsoft Open Source Code of Conduct](https://opensource.microsoft.com/codeofconduct/). 12 | For more information see the [Code of Conduct FAQ](https://opensource.microsoft.com/codeofconduct/faq/) or 13 | contact [opencode@microsoft.com](mailto:opencode@microsoft.com) with any additional questions or comments. 14 | 15 | - [Code of Conduct](#coc) 16 | - [Issues and Bugs](#issue) 17 | - [Feature Requests](#feature) 18 | - [Submission Guidelines](#submit) 19 | 20 | ## Code of Conduct 21 | Help us keep this project open and inclusive. Please read and follow our [Code of Conduct](https://opensource.microsoft.com/codeofconduct/). 22 | 23 | ## Found an Issue? 24 | If you find a bug in the source code or a mistake in the documentation, you can help us by 25 | [submitting an issue](#submit-issue) to the GitHub Repository. Even better, you can 26 | [submit a Pull Request](#submit-pr) with a fix. 27 | 28 | ## Want a Feature? 29 | You can *request* a new feature by [submitting an issue](#submit-issue) to the GitHub 30 | Repository. If you would like to *implement* a new feature, please submit an issue with 31 | a proposal for your work first, to be sure that we can use it. 32 | 33 | * **Small Features** can be crafted and directly [submitted as a Pull Request](#submit-pr). 34 | 35 | ## Submission Guidelines 36 | 37 | ### Submitting an Issue 38 | Before you submit an issue, search the archive, maybe your question was already answered. 39 | 40 | If your issue appears to be a bug, and hasn't been reported, open a new issue. 41 | Help us to maximize the effort we can spend fixing issues and adding new 42 | features, by not reporting duplicate issues. Providing the following information will increase the 43 | chances of your issue being dealt with quickly: 44 | 45 | * **Overview of the Issue** - if an error is being thrown a non-minified stack trace helps 46 | * **Version** - what version is affected (e.g. 0.1.2) 47 | * **Motivation for or Use Case** - explain what are you trying to do and why the current behavior is a bug for you 48 | * **Browsers and Operating System** - is this a problem with all browsers? 49 | * **Reproduce the Error** - provide a live example or a unambiguous set of steps 50 | * **Related Issues** - has a similar issue been reported before? 51 | * **Suggest a Fix** - if you can't fix the bug yourself, perhaps you can point to what might be 52 | causing the problem (line of code or commit) 53 | 54 | You can file new issues by providing the above information at the corresponding repository's issues link: https://github.com/[organization-name]/[repository-name]/issues/new]. 55 | 56 | ### Submitting a Pull Request (PR) 57 | Before you submit your Pull Request (PR) consider the following guidelines: 58 | 59 | * Search the repository (https://github.com/[organization-name]/[repository-name]/pulls) for an open or closed PR 60 | that relates to your submission. You don't want to duplicate effort. 61 | 62 | * Make your changes in a new git fork: 63 | 64 | * Commit your changes using a descriptive commit message 65 | * Push your fork to GitHub: 66 | * In GitHub, create a pull request 67 | * If we suggest changes then: 68 | * Make the required updates. 69 | * Rebase your fork and force push to your GitHub repository (this will update your Pull Request): 70 | 71 | ```shell 72 | git rebase master -i 73 | git push -f 74 | ``` 75 | 76 | That's it! Thank you for your contribution! 77 | -------------------------------------------------------------------------------- /LICENSE.md: -------------------------------------------------------------------------------- 1 | MIT License 2 | 3 | Copyright (c) Microsoft Corporation. All rights reserved. 4 | 5 | Permission is hereby granted, free of charge, to any person obtaining a copy 6 | of this software and associated documentation files (the "Software"), to deal 7 | in the Software without restriction, including without limitation the rights 8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 9 | copies of the Software, and to permit persons to whom the Software is 10 | furnished to do so, subject to the following conditions: 11 | 12 | The above copyright notice and this permission notice shall be included in all 13 | copies or substantial portions of the Software. 14 | 15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 21 | SOFTWARE -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | --- 2 | page_type: sample 3 | languages: 4 | - javascript 5 | - nodejs 6 | products: 7 | - azure 8 | - azure-storage 9 | description: "How to stream blobs to Azure Blob Storage with Node.js." 10 | urlFragment: stream-blobs-nodejs 11 | --- 12 | 13 | # How to stream blobs to Azure Blob Storage with Node.js 14 | 15 | ## SDK Versions 16 | In this sample, you will find the following folders: 17 | 18 | * **v10** - references Storage Blob SDK v10 19 | * **v12** - references Storage Blob SDK v12 20 | 21 | ## Prerequisites 22 | If you don't have a Microsoft Azure subscription, you can get a [free account] before you begin. 23 | 24 | Clone the repository to your machine: 25 | 26 | ```bash 27 | git clone https://github.com/Azure-Samples/azure-sdk-for-js-storage-blob-stream-nodejs.git 28 | ``` 29 | 30 | Then, switch to the appropriate folder: 31 | 32 | ```bash 33 | cd v10 34 | ``` 35 | 36 | or 37 | 38 | ```bash 39 | cd v12 40 | ``` 41 | 42 | Install dependencies via `npm`: 43 | 44 | ```bash 45 | npm install 46 | ``` 47 | 48 | ## This Sample shows how to do following operations of Storage Blobs 49 | * Create a storage account. 50 | * Create a container. 51 | * Upload a stream to [blockblob]. 52 | 53 | ## Adding your storage account name and key 54 | Navigate to your storage account in the [Azure Portal] and copy the account name and key (under **Settings** > **Access keys**) into the `.env.example` file. Save the file and then rename it from `.env.example` to `.env`. 55 | 56 | ## Running the sample 57 | Start the server: 58 | 59 | ```bash 60 | npm start 61 | ``` 62 | 63 | Navigate to [http://localhost:3000] and upload an image to blob storage. 64 | 65 | You can use the [Azure Storage Explorer] to view blob containers and verify your upload is successful. 66 | 67 | 68 | [blockblob]: https://docs.microsoft.com/en-us/rest/api/storageservices/understanding-block-blobs--append-blobs--and-page-blobs 69 | [Azure Portal]: https://portal.azure.com 70 | [http://localhost:3000]: http://localhost:3000 71 | [Azure Storage Explorer]: https://azure.microsoft.com/features/storage-explorer/ 72 | [free account]: http://go.microsoft.com/fwlink/?LinkId=330212 73 | 74 | -------------------------------------------------------------------------------- /v10/.env.example: -------------------------------------------------------------------------------- 1 | AZURE_STORAGE_ACCOUNT_NAME= 2 | AZURE_STORAGE_ACCOUNT_ACCESS_KEY= -------------------------------------------------------------------------------- /v10/.gitignore: -------------------------------------------------------------------------------- 1 | # Logs 2 | logs 3 | *.log 4 | npm-debug.log* 5 | yarn-debug.log* 6 | yarn-error.log* 7 | 8 | # Runtime data 9 | pids 10 | *.pid 11 | *.seed 12 | *.pid.lock 13 | 14 | # Directory for instrumented libs generated by jscoverage/JSCover 15 | lib-cov 16 | 17 | # Coverage directory used by tools like istanbul 18 | coverage 19 | 20 | # nyc test coverage 21 | .nyc_output 22 | 23 | # Grunt intermediate storage (http://gruntjs.com/creating-plugins#storing-task-files) 24 | .grunt 25 | 26 | # Bower dependency directory (https://bower.io/) 27 | bower_components 28 | 29 | # node-waf configuration 30 | .lock-wscript 31 | 32 | # Compiled binary addons (https://nodejs.org/api/addons.html) 33 | build/Release 34 | 35 | # Dependency directories 36 | node_modules/ 37 | jspm_packages/ 38 | 39 | # TypeScript v1 declaration files 40 | typings/ 41 | 42 | # Optional npm cache directory 43 | .npm 44 | 45 | # Optional eslint cache 46 | .eslintcache 47 | 48 | # Optional REPL history 49 | .node_repl_history 50 | 51 | # Output of 'npm pack' 52 | *.tgz 53 | 54 | # Yarn Integrity file 55 | .yarn-integrity 56 | 57 | # dotenv environment variables file 58 | .env 59 | 60 | # next.js build output 61 | .next 62 | 63 | -------------------------------------------------------------------------------- /v10/app.js: -------------------------------------------------------------------------------- 1 | const express = require('express'); 2 | const path = require('path'); 3 | const favicon = require('serve-favicon'); 4 | const logger = require('morgan'); 5 | const cookieParser = require('cookie-parser'); 6 | const bodyParser = require('body-parser'); 7 | 8 | const index = require('./routes/index'); 9 | const upload = require('./routes/upload'); 10 | 11 | const app = express(); 12 | 13 | // view engine setup 14 | app.set('views', path.join(__dirname, 'views')); 15 | app.set('view engine', 'hbs'); 16 | 17 | app.use(favicon(path.join(__dirname, 'public', 'favicon.ico'))); 18 | app.use(logger('dev')); 19 | app.use(bodyParser.json()); 20 | app.use(bodyParser.urlencoded({ extended: false })); 21 | app.use(cookieParser()); 22 | app.use(express.static(path.join(__dirname, 'public'))); 23 | 24 | app.use('/', index); 25 | app.use('/upload', upload); 26 | 27 | // catch 404 and forward to error handler 28 | app.use((req, res, next) => { 29 | const err = new Error('Not Found'); 30 | err.status = 404; 31 | next(err); 32 | }); 33 | 34 | // error handler 35 | app.use((err, req, res, next) => { 36 | // set locals, only providing error in development 37 | res.locals.message = err.message; 38 | res.locals.error = req.app.get('env') === 'development' ? err : {}; 39 | 40 | // render the error page 41 | res.status(err.status || 500); 42 | res.render('error'); 43 | }); 44 | 45 | module.exports = app; 46 | -------------------------------------------------------------------------------- /v10/bin/www: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env node 2 | 3 | /** 4 | * Module dependencies. 5 | */ 6 | 7 | var app = require('../app'); 8 | var debug = require('debug')('node-blobs-image-resize:server'); 9 | var http = require('http'); 10 | 11 | /** 12 | * Get port from environment and store in Express. 13 | */ 14 | 15 | var port = normalizePort(process.env.PORT || '3000'); 16 | app.set('port', port); 17 | 18 | /** 19 | * Create HTTP server. 20 | */ 21 | 22 | var server = http.createServer(app); 23 | 24 | /** 25 | * Listen on provided port, on all network interfaces. 26 | */ 27 | 28 | server.listen(port); 29 | server.on('error', onError); 30 | server.on('listening', onListening); 31 | 32 | /** 33 | * Normalize a port into a number, string, or false. 34 | */ 35 | 36 | function normalizePort(val) { 37 | var port = parseInt(val, 10); 38 | 39 | if (isNaN(port)) { 40 | // named pipe 41 | return val; 42 | } 43 | 44 | if (port >= 0) { 45 | // port number 46 | return port; 47 | } 48 | 49 | return false; 50 | } 51 | 52 | /** 53 | * Event listener for HTTP server "error" event. 54 | */ 55 | 56 | function onError(error) { 57 | if (error.syscall !== 'listen') { 58 | throw error; 59 | } 60 | 61 | var bind = typeof port === 'string' 62 | ? 'Pipe ' + port 63 | : 'Port ' + port; 64 | 65 | // handle specific listen errors with friendly messages 66 | switch (error.code) { 67 | case 'EACCES': 68 | console.error(bind + ' requires elevated privileges'); 69 | process.exit(1); 70 | break; 71 | case 'EADDRINUSE': 72 | console.error(bind + ' is already in use'); 73 | process.exit(1); 74 | break; 75 | default: 76 | throw error; 77 | } 78 | } 79 | 80 | /** 81 | * Event listener for HTTP server "listening" event. 82 | */ 83 | 84 | function onListening() { 85 | var addr = server.address(); 86 | var bind = typeof addr === 'string' 87 | ? 'pipe ' + addr 88 | : 'port ' + addr.port; 89 | debug('Listening on ' + bind); 90 | } 91 | -------------------------------------------------------------------------------- /v10/config.js: -------------------------------------------------------------------------------- 1 | 2 | const _module = { 3 | getStorageAccountName: () => { 4 | const matches = /AccountName=(.*?);/.exec(process.env.AZURE_STORAGE_CONNECTION_STRING); 5 | return matches[1]; 6 | } 7 | }; 8 | 9 | module.exports = _module; -------------------------------------------------------------------------------- /v10/package-lock.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "node-blobs-image-resize", 3 | "version": "0.0.0", 4 | "lockfileVersion": 1, 5 | "requires": true, 6 | "dependencies": { 7 | "@azure/ms-rest-js": { 8 | "version": "2.0.5", 9 | "resolved": "https://registry.npmjs.org/@azure/ms-rest-js/-/ms-rest-js-2.0.5.tgz", 10 | "integrity": "sha512-zj9/JOSsbaJleZz9k9RXKJMBPA2NBUgsus5iZxGlhK+pDeb8M2ps+lkKhJSwLnlTUj5CvrcN3PZDF2b2nSYCQQ==", 11 | "requires": { 12 | "@types/node-fetch": "^2.3.7", 13 | "@types/tunnel": "0.0.1", 14 | "abort-controller": "^3.0.0", 15 | "form-data": "^2.5.0", 16 | "node-fetch": "^2.6.0", 17 | "tough-cookie": "^3.0.1", 18 | "tslib": "^1.10.0", 19 | "tunnel": "0.0.6", 20 | "uuid": "^3.3.2", 21 | "xml2js": "^0.4.19" 22 | } 23 | }, 24 | "@azure/storage-blob": { 25 | "version": "10.5.0", 26 | "resolved": "https://registry.npmjs.org/@azure/storage-blob/-/storage-blob-10.5.0.tgz", 27 | "integrity": "sha512-67+0EP7STy9BQgzvN1RgmSvXhxRd044eDgepX7zBp7XslBxz8YGo2cSLm9w5o5Qf1FLCRlwuziRMikaPCLMpVw==", 28 | "requires": { 29 | "@azure/ms-rest-js": "^2.0.0", 30 | "events": "^3.0.0", 31 | "tslib": "^1.9.3" 32 | } 33 | }, 34 | "@types/node": { 35 | "version": "13.1.6", 36 | "resolved": "https://registry.npmjs.org/@types/node/-/node-13.1.6.tgz", 37 | "integrity": "sha512-Jg1F+bmxcpENHP23sVKkNuU3uaxPnsBMW0cLjleiikFKomJQbsn0Cqk2yDvQArqzZN6ABfBkZ0To7pQ8sLdWDg==" 38 | }, 39 | "@types/node-fetch": { 40 | "version": "2.5.4", 41 | "resolved": "https://registry.npmjs.org/@types/node-fetch/-/node-fetch-2.5.4.tgz", 42 | "integrity": "sha512-Oz6id++2qAOFuOlE1j0ouk1dzl3mmI1+qINPNBhi9nt/gVOz0G+13Ao6qjhdF0Ys+eOkhu6JnFmt38bR3H0POQ==", 43 | "requires": { 44 | "@types/node": "*" 45 | } 46 | }, 47 | "@types/tunnel": { 48 | "version": "0.0.1", 49 | "resolved": "https://registry.npmjs.org/@types/tunnel/-/tunnel-0.0.1.tgz", 50 | "integrity": "sha512-AOqu6bQu5MSWwYvehMXLukFHnupHrpZ8nvgae5Ggie9UwzDR1CCwoXgSSWNZJuyOlCdfdsWMA5F2LlmvyoTv8A==", 51 | "requires": { 52 | "@types/node": "*" 53 | } 54 | }, 55 | "abort-controller": { 56 | "version": "3.0.0", 57 | "resolved": "https://registry.npmjs.org/abort-controller/-/abort-controller-3.0.0.tgz", 58 | "integrity": "sha512-h8lQ8tacZYnR3vNQTgibj+tODHI5/+l06Au2Pcriv/Gmet0eaj4TwWH41sO9wnHDiQsEj19q0drzdWdeAHtweg==", 59 | "requires": { 60 | "event-target-shim": "^5.0.0" 61 | } 62 | }, 63 | "accepts": { 64 | "version": "1.3.7", 65 | "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.7.tgz", 66 | "integrity": "sha512-Il80Qs2WjYlJIBNzNkK6KYqlVMTbZLXgHx2oT0pU/fjRHyEp+PEfEPY0R3WCwAGVOtauxh1hOxNgIf5bv7dQpA==", 67 | "requires": { 68 | "mime-types": "~2.1.24", 69 | "negotiator": "0.6.2" 70 | } 71 | }, 72 | "append-field": { 73 | "version": "1.0.0", 74 | "resolved": "https://registry.npmjs.org/append-field/-/append-field-1.0.0.tgz", 75 | "integrity": "sha1-HjRA6RXwsSA9I3SOeO3XubW0PlY=" 76 | }, 77 | "array-flatten": { 78 | "version": "1.1.1", 79 | "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz", 80 | "integrity": "sha1-ml9pkFGx5wczKPKgCJaLZOopVdI=" 81 | }, 82 | "asynckit": { 83 | "version": "0.4.0", 84 | "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", 85 | "integrity": "sha1-x57Zf380y48robyXkLzDZkdLS3k=" 86 | }, 87 | "basic-auth": { 88 | "version": "2.0.1", 89 | "resolved": "https://registry.npmjs.org/basic-auth/-/basic-auth-2.0.1.tgz", 90 | "integrity": "sha512-NF+epuEdnUYVlGuhaxbbq+dvJttwLnGY+YixlXlME5KpQ5W3CnXA5cVTneY3SPbPDRkcjMbifrwmFYcClgOZeg==", 91 | "requires": { 92 | "safe-buffer": "5.1.2" 93 | } 94 | }, 95 | "body-parser": { 96 | "version": "1.19.0", 97 | "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.19.0.tgz", 98 | "integrity": "sha512-dhEPs72UPbDnAQJ9ZKMNTP6ptJaionhP5cBb541nXPlW60Jepo9RV/a4fX4XWW9CuFNK22krhrj1+rgzifNCsw==", 99 | "requires": { 100 | "bytes": "3.1.0", 101 | "content-type": "~1.0.4", 102 | "debug": "2.6.9", 103 | "depd": "~1.1.2", 104 | "http-errors": "1.7.2", 105 | "iconv-lite": "0.4.24", 106 | "on-finished": "~2.3.0", 107 | "qs": "6.7.0", 108 | "raw-body": "2.4.0", 109 | "type-is": "~1.6.17" 110 | } 111 | }, 112 | "buffer-from": { 113 | "version": "1.1.1", 114 | "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.1.tgz", 115 | "integrity": "sha512-MQcXEUbCKtEo7bhqEs6560Hyd4XaovZlO/k9V3hjVUF/zwW7KBVdSK4gIt/bzwS9MbR5qob+F5jusZsb0YQK2A==" 116 | }, 117 | "busboy": { 118 | "version": "0.2.14", 119 | "resolved": "https://registry.npmjs.org/busboy/-/busboy-0.2.14.tgz", 120 | "integrity": "sha1-bCpiLvz0fFe7vh4qnDetNseSVFM=", 121 | "requires": { 122 | "dicer": "0.2.5", 123 | "readable-stream": "1.1.x" 124 | }, 125 | "dependencies": { 126 | "isarray": { 127 | "version": "0.0.1", 128 | "resolved": "https://registry.npmjs.org/isarray/-/isarray-0.0.1.tgz", 129 | "integrity": "sha1-ihis/Kmo9Bd+Cav8YDiTmwXR7t8=" 130 | }, 131 | "readable-stream": { 132 | "version": "1.1.14", 133 | "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-1.1.14.tgz", 134 | "integrity": "sha1-fPTFTvZI44EwhMY23SB54WbAgdk=", 135 | "requires": { 136 | "core-util-is": "~1.0.0", 137 | "inherits": "~2.0.1", 138 | "isarray": "0.0.1", 139 | "string_decoder": "~0.10.x" 140 | } 141 | }, 142 | "string_decoder": { 143 | "version": "0.10.31", 144 | "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-0.10.31.tgz", 145 | "integrity": "sha1-YuIDvEF2bGwoyfyEMB2rHFMQ+pQ=" 146 | } 147 | } 148 | }, 149 | "bytes": { 150 | "version": "3.1.0", 151 | "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.0.tgz", 152 | "integrity": "sha512-zauLjrfCG+xvoyaqLoV8bLVXXNGC4JqlxFCutSDWA6fJrTo2ZuvLYTqZ7aHBLZSMOopbzwv8f+wZcVzfVTI2Dg==" 153 | }, 154 | "combined-stream": { 155 | "version": "1.0.8", 156 | "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz", 157 | "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==", 158 | "requires": { 159 | "delayed-stream": "~1.0.0" 160 | } 161 | }, 162 | "commander": { 163 | "version": "2.20.3", 164 | "resolved": "https://registry.npmjs.org/commander/-/commander-2.20.3.tgz", 165 | "integrity": "sha512-GpVkmM8vF2vQUkj2LvZmD35JxeJOLCwJ9cUkugyk2nuhbv3+mJvpLYYt+0+USMxE+oj+ey/lJEnhZw75x/OMcQ==", 166 | "optional": true 167 | }, 168 | "concat-stream": { 169 | "version": "1.6.2", 170 | "resolved": "https://registry.npmjs.org/concat-stream/-/concat-stream-1.6.2.tgz", 171 | "integrity": "sha512-27HBghJxjiZtIk3Ycvn/4kbJk/1uZuJFfuPEns6LaEvpvG1f0hTea8lilrouyo9mVc2GWdcEZ8OLoGmSADlrCw==", 172 | "requires": { 173 | "buffer-from": "^1.0.0", 174 | "inherits": "^2.0.3", 175 | "readable-stream": "^2.2.2", 176 | "typedarray": "^0.0.6" 177 | } 178 | }, 179 | "content-disposition": { 180 | "version": "0.5.3", 181 | "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.3.tgz", 182 | "integrity": "sha512-ExO0774ikEObIAEV9kDo50o+79VCUdEB6n6lzKgGwupcVeRlhrj3qGAfwq8G6uBJjkqLrhT0qEYFcWng8z1z0g==", 183 | "requires": { 184 | "safe-buffer": "5.1.2" 185 | } 186 | }, 187 | "content-type": { 188 | "version": "1.0.4", 189 | "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.4.tgz", 190 | "integrity": "sha512-hIP3EEPs8tB9AT1L+NUqtwOAps4mk2Zob89MWXMHjHWg9milF/j4osnnQLXBCBFBk/tvIG/tUc9mOUJiPBhPXA==" 191 | }, 192 | "cookie": { 193 | "version": "0.3.1", 194 | "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.3.1.tgz", 195 | "integrity": "sha1-5+Ch+e9DtMi6klxcWpboBtFoc7s=" 196 | }, 197 | "cookie-parser": { 198 | "version": "1.4.4", 199 | "resolved": "https://registry.npmjs.org/cookie-parser/-/cookie-parser-1.4.4.tgz", 200 | "integrity": "sha512-lo13tqF3JEtFO7FyA49CqbhaFkskRJ0u/UAiINgrIXeRCY41c88/zxtrECl8AKH3B0hj9q10+h3Kt8I7KlW4tw==", 201 | "requires": { 202 | "cookie": "0.3.1", 203 | "cookie-signature": "1.0.6" 204 | } 205 | }, 206 | "cookie-signature": { 207 | "version": "1.0.6", 208 | "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz", 209 | "integrity": "sha1-4wOogrNCzD7oylE6eZmXNNqzriw=" 210 | }, 211 | "core-util-is": { 212 | "version": "1.0.2", 213 | "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.2.tgz", 214 | "integrity": "sha1-tf1UIgqivFq1eqtxQMlAdUUDwac=" 215 | }, 216 | "debug": { 217 | "version": "2.6.9", 218 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", 219 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", 220 | "requires": { 221 | "ms": "2.0.0" 222 | } 223 | }, 224 | "delayed-stream": { 225 | "version": "1.0.0", 226 | "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", 227 | "integrity": "sha1-3zrhmayt+31ECqrgsp4icrJOxhk=" 228 | }, 229 | "depd": { 230 | "version": "1.1.2", 231 | "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.2.tgz", 232 | "integrity": "sha1-m81S4UwJd2PnSbJ0xDRu0uVgtak=" 233 | }, 234 | "destroy": { 235 | "version": "1.0.4", 236 | "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.0.4.tgz", 237 | "integrity": "sha1-l4hXRCxEdJ5CBmE+N5RiBYJqvYA=" 238 | }, 239 | "dicer": { 240 | "version": "0.2.5", 241 | "resolved": "https://registry.npmjs.org/dicer/-/dicer-0.2.5.tgz", 242 | "integrity": "sha1-WZbAhrszIYyBLAkL3cCc0S+stw8=", 243 | "requires": { 244 | "readable-stream": "1.1.x", 245 | "streamsearch": "0.1.2" 246 | }, 247 | "dependencies": { 248 | "isarray": { 249 | "version": "0.0.1", 250 | "resolved": "https://registry.npmjs.org/isarray/-/isarray-0.0.1.tgz", 251 | "integrity": "sha1-ihis/Kmo9Bd+Cav8YDiTmwXR7t8=" 252 | }, 253 | "readable-stream": { 254 | "version": "1.1.14", 255 | "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-1.1.14.tgz", 256 | "integrity": "sha1-fPTFTvZI44EwhMY23SB54WbAgdk=", 257 | "requires": { 258 | "core-util-is": "~1.0.0", 259 | "inherits": "~2.0.1", 260 | "isarray": "0.0.1", 261 | "string_decoder": "~0.10.x" 262 | } 263 | }, 264 | "string_decoder": { 265 | "version": "0.10.31", 266 | "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-0.10.31.tgz", 267 | "integrity": "sha1-YuIDvEF2bGwoyfyEMB2rHFMQ+pQ=" 268 | } 269 | } 270 | }, 271 | "dotenv": { 272 | "version": "5.0.1", 273 | "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-5.0.1.tgz", 274 | "integrity": "sha512-4As8uPrjfwb7VXC+WnLCbXK7y+Ueb2B3zgNCePYfhxS1PYeaO1YTeplffTEcbfLhvFNGLAz90VvJs9yomG7bow==" 275 | }, 276 | "ee-first": { 277 | "version": "1.1.1", 278 | "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", 279 | "integrity": "sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0=" 280 | }, 281 | "encodeurl": { 282 | "version": "1.0.2", 283 | "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz", 284 | "integrity": "sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k=" 285 | }, 286 | "escape-html": { 287 | "version": "1.0.3", 288 | "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz", 289 | "integrity": "sha1-Aljq5NPQwJdN4cFpGI7wBR0dGYg=" 290 | }, 291 | "etag": { 292 | "version": "1.8.1", 293 | "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz", 294 | "integrity": "sha1-Qa4u62XvpiJorr/qg6x9eSmbCIc=" 295 | }, 296 | "event-target-shim": { 297 | "version": "5.0.1", 298 | "resolved": "https://registry.npmjs.org/event-target-shim/-/event-target-shim-5.0.1.tgz", 299 | "integrity": "sha512-i/2XbnSz/uxRCU6+NdVJgKWDTM427+MqYbkQzD321DuCQJUqOuJKIA0IM2+W2xtYHdKOmZ4dR6fExsd4SXL+WQ==" 300 | }, 301 | "events": { 302 | "version": "3.1.0", 303 | "resolved": "https://registry.npmjs.org/events/-/events-3.1.0.tgz", 304 | "integrity": "sha512-Rv+u8MLHNOdMjTAFeT3nCjHn2aGlx435FP/sDHNaRhDEMwyI/aB22Kj2qIN8R0cw3z28psEQLYwxVKLsKrMgWg==" 305 | }, 306 | "express": { 307 | "version": "4.17.1", 308 | "resolved": "https://registry.npmjs.org/express/-/express-4.17.1.tgz", 309 | "integrity": "sha512-mHJ9O79RqluphRrcw2X/GTh3k9tVv8YcoyY4Kkh4WDMUYKRZUq0h1o0w2rrrxBqM7VoeUVqgb27xlEMXTnYt4g==", 310 | "requires": { 311 | "accepts": "~1.3.7", 312 | "array-flatten": "1.1.1", 313 | "body-parser": "1.19.0", 314 | "content-disposition": "0.5.3", 315 | "content-type": "~1.0.4", 316 | "cookie": "0.4.0", 317 | "cookie-signature": "1.0.6", 318 | "debug": "2.6.9", 319 | "depd": "~1.1.2", 320 | "encodeurl": "~1.0.2", 321 | "escape-html": "~1.0.3", 322 | "etag": "~1.8.1", 323 | "finalhandler": "~1.1.2", 324 | "fresh": "0.5.2", 325 | "merge-descriptors": "1.0.1", 326 | "methods": "~1.1.2", 327 | "on-finished": "~2.3.0", 328 | "parseurl": "~1.3.3", 329 | "path-to-regexp": "0.1.7", 330 | "proxy-addr": "~2.0.5", 331 | "qs": "6.7.0", 332 | "range-parser": "~1.2.1", 333 | "safe-buffer": "5.1.2", 334 | "send": "0.17.1", 335 | "serve-static": "1.14.1", 336 | "setprototypeof": "1.1.1", 337 | "statuses": "~1.5.0", 338 | "type-is": "~1.6.18", 339 | "utils-merge": "1.0.1", 340 | "vary": "~1.1.2" 341 | }, 342 | "dependencies": { 343 | "cookie": { 344 | "version": "0.4.0", 345 | "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.4.0.tgz", 346 | "integrity": "sha512-+Hp8fLp57wnUSt0tY0tHEXh4voZRDnoIrZPqlo3DPiI4y9lwg/jqx+1Om94/W6ZaPDOUbnjOt/99w66zk+l1Xg==" 347 | } 348 | } 349 | }, 350 | "finalhandler": { 351 | "version": "1.1.2", 352 | "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.1.2.tgz", 353 | "integrity": "sha512-aAWcW57uxVNrQZqFXjITpW3sIUQmHGG3qSb9mUah9MgMC4NeWhNOlNjXEYq3HjRAvL6arUviZGGJsBg6z0zsWA==", 354 | "requires": { 355 | "debug": "2.6.9", 356 | "encodeurl": "~1.0.2", 357 | "escape-html": "~1.0.3", 358 | "on-finished": "~2.3.0", 359 | "parseurl": "~1.3.3", 360 | "statuses": "~1.5.0", 361 | "unpipe": "~1.0.0" 362 | } 363 | }, 364 | "foreachasync": { 365 | "version": "3.0.0", 366 | "resolved": "https://registry.npmjs.org/foreachasync/-/foreachasync-3.0.0.tgz", 367 | "integrity": "sha1-VQKYfchxS+M5IJfzLgBxyd7gfPY=" 368 | }, 369 | "form-data": { 370 | "version": "2.5.1", 371 | "resolved": "https://registry.npmjs.org/form-data/-/form-data-2.5.1.tgz", 372 | "integrity": "sha512-m21N3WOmEEURgk6B9GLOE4RuWOFf28Lhh9qGYeNlGq4VDXUlJy2th2slBNU8Gp8EzloYZOibZJ7t5ecIrFSjVA==", 373 | "requires": { 374 | "asynckit": "^0.4.0", 375 | "combined-stream": "^1.0.6", 376 | "mime-types": "^2.1.12" 377 | } 378 | }, 379 | "forwarded": { 380 | "version": "0.1.2", 381 | "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.1.2.tgz", 382 | "integrity": "sha1-mMI9qxF1ZXuMBXPozszZGw/xjIQ=" 383 | }, 384 | "fresh": { 385 | "version": "0.5.2", 386 | "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz", 387 | "integrity": "sha1-PYyt2Q2XZWn6g1qx+OSyOhBWBac=" 388 | }, 389 | "from2": { 390 | "version": "2.3.0", 391 | "resolved": "https://registry.npmjs.org/from2/-/from2-2.3.0.tgz", 392 | "integrity": "sha1-i/tVAr3kpNNs/e6gB/zKIdfjgq8=", 393 | "requires": { 394 | "inherits": "^2.0.1", 395 | "readable-stream": "^2.0.0" 396 | } 397 | }, 398 | "handlebars": { 399 | "version": "4.3.5", 400 | "resolved": "https://registry.npmjs.org/handlebars/-/handlebars-4.3.5.tgz", 401 | "integrity": "sha512-I16T/l8X9DV3sEkY9sK9lsPRgDsj82ayBY/4pAZyP2BcX5WeRM3O06bw9kIs2GLrHvFB/DNzWWJyFvof8wQGqw==", 402 | "requires": { 403 | "neo-async": "^2.6.0", 404 | "optimist": "^0.6.1", 405 | "source-map": "^0.6.1", 406 | "uglify-js": "^3.1.4" 407 | } 408 | }, 409 | "hbs": { 410 | "version": "4.0.6", 411 | "resolved": "https://registry.npmjs.org/hbs/-/hbs-4.0.6.tgz", 412 | "integrity": "sha512-KFt3Y4zOvVQOp84TmqVaFTpBTYO1sVenBoBY712MI3vPkKxVoO6AsuEyDayIRPRAHRYZHHWnmc4spFa8fhQpLw==", 413 | "requires": { 414 | "handlebars": "4.3.5", 415 | "walk": "2.3.14" 416 | } 417 | }, 418 | "http-errors": { 419 | "version": "1.7.2", 420 | "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.7.2.tgz", 421 | "integrity": "sha512-uUQBt3H/cSIVfch6i1EuPNy/YsRSOUBXTVfZ+yR7Zjez3qjBz6i9+i4zjNaoqcoFVI4lQJ5plg63TvGfRSDCRg==", 422 | "requires": { 423 | "depd": "~1.1.2", 424 | "inherits": "2.0.3", 425 | "setprototypeof": "1.1.1", 426 | "statuses": ">= 1.5.0 < 2", 427 | "toidentifier": "1.0.0" 428 | } 429 | }, 430 | "iconv-lite": { 431 | "version": "0.4.24", 432 | "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", 433 | "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", 434 | "requires": { 435 | "safer-buffer": ">= 2.1.2 < 3" 436 | } 437 | }, 438 | "inherits": { 439 | "version": "2.0.3", 440 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", 441 | "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=" 442 | }, 443 | "into-stream": { 444 | "version": "3.1.0", 445 | "resolved": "https://registry.npmjs.org/into-stream/-/into-stream-3.1.0.tgz", 446 | "integrity": "sha1-lvsKk2wSur1v8XUqF9BWFqvQlMY=", 447 | "requires": { 448 | "from2": "^2.1.1", 449 | "p-is-promise": "^1.1.0" 450 | } 451 | }, 452 | "ip-regex": { 453 | "version": "2.1.0", 454 | "resolved": "https://registry.npmjs.org/ip-regex/-/ip-regex-2.1.0.tgz", 455 | "integrity": "sha1-+ni/XS5pE8kRzp+BnuUUa7bYROk=" 456 | }, 457 | "ipaddr.js": { 458 | "version": "1.9.0", 459 | "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.0.tgz", 460 | "integrity": "sha512-M4Sjn6N/+O6/IXSJseKqHoFc+5FdGJ22sXqnjTpdZweHK64MzEPAyQZyEU3R/KRv2GLoa7nNtg/C2Ev6m7z+eA==" 461 | }, 462 | "isarray": { 463 | "version": "1.0.0", 464 | "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", 465 | "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE=" 466 | }, 467 | "media-typer": { 468 | "version": "0.3.0", 469 | "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", 470 | "integrity": "sha1-hxDXrwqmJvj/+hzgAWhUUmMlV0g=" 471 | }, 472 | "merge-descriptors": { 473 | "version": "1.0.1", 474 | "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz", 475 | "integrity": "sha1-sAqqVW3YtEVoFQ7J0blT8/kMu2E=" 476 | }, 477 | "methods": { 478 | "version": "1.1.2", 479 | "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz", 480 | "integrity": "sha1-VSmk1nZUE07cxSZmVoNbD4Ua/O4=" 481 | }, 482 | "mime": { 483 | "version": "1.6.0", 484 | "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz", 485 | "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==" 486 | }, 487 | "mime-db": { 488 | "version": "1.43.0", 489 | "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.43.0.tgz", 490 | "integrity": "sha512-+5dsGEEovYbT8UY9yD7eE4XTc4UwJ1jBYlgaQQF38ENsKR3wj/8q8RFZrF9WIZpB2V1ArTVFUva8sAul1NzRzQ==" 491 | }, 492 | "mime-types": { 493 | "version": "2.1.26", 494 | "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.26.tgz", 495 | "integrity": "sha512-01paPWYgLrkqAyrlDorC1uDwl2p3qZT7yl806vW7DvDoxwXi46jsjFbg+WdwotBIk6/MbEhO/dh5aZ5sNj/dWQ==", 496 | "requires": { 497 | "mime-db": "1.43.0" 498 | } 499 | }, 500 | "minimist": { 501 | "version": "0.0.10", 502 | "resolved": "https://registry.npmjs.org/minimist/-/minimist-0.0.10.tgz", 503 | "integrity": "sha1-3j+YVD2/lggr5IrRoMfNqDYwHc8=" 504 | }, 505 | "mkdirp": { 506 | "version": "0.5.1", 507 | "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.1.tgz", 508 | "integrity": "sha1-MAV0OOrGz3+MR2fzhkjWaX11yQM=", 509 | "requires": { 510 | "minimist": "0.0.8" 511 | }, 512 | "dependencies": { 513 | "minimist": { 514 | "version": "0.0.8", 515 | "resolved": "https://registry.npmjs.org/minimist/-/minimist-0.0.8.tgz", 516 | "integrity": "sha1-hX/Kv8M5fSYluCKCYuhqp6ARsF0=" 517 | } 518 | } 519 | }, 520 | "morgan": { 521 | "version": "1.9.1", 522 | "resolved": "https://registry.npmjs.org/morgan/-/morgan-1.9.1.tgz", 523 | "integrity": "sha512-HQStPIV4y3afTiCYVxirakhlCfGkI161c76kKFca7Fk1JusM//Qeo1ej2XaMniiNeaZklMVrh3vTtIzpzwbpmA==", 524 | "requires": { 525 | "basic-auth": "~2.0.0", 526 | "debug": "2.6.9", 527 | "depd": "~1.1.2", 528 | "on-finished": "~2.3.0", 529 | "on-headers": "~1.0.1" 530 | } 531 | }, 532 | "ms": { 533 | "version": "2.0.0", 534 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", 535 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" 536 | }, 537 | "multer": { 538 | "version": "1.4.2", 539 | "resolved": "https://registry.npmjs.org/multer/-/multer-1.4.2.tgz", 540 | "integrity": "sha512-xY8pX7V+ybyUpbYMxtjM9KAiD9ixtg5/JkeKUTD6xilfDv0vzzOFcCp4Ljb1UU3tSOM3VTZtKo63OmzOrGi3Cg==", 541 | "requires": { 542 | "append-field": "^1.0.0", 543 | "busboy": "^0.2.11", 544 | "concat-stream": "^1.5.2", 545 | "mkdirp": "^0.5.1", 546 | "object-assign": "^4.1.1", 547 | "on-finished": "^2.3.0", 548 | "type-is": "^1.6.4", 549 | "xtend": "^4.0.0" 550 | } 551 | }, 552 | "negotiator": { 553 | "version": "0.6.2", 554 | "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.2.tgz", 555 | "integrity": "sha512-hZXc7K2e+PgeI1eDBe/10Ard4ekbfrrqG8Ep+8Jmf4JID2bNg7NvCPOZN+kfF574pFQI7mum2AUqDidoKqcTOw==" 556 | }, 557 | "neo-async": { 558 | "version": "2.6.1", 559 | "resolved": "https://registry.npmjs.org/neo-async/-/neo-async-2.6.1.tgz", 560 | "integrity": "sha512-iyam8fBuCUpWeKPGpaNMetEocMt364qkCsfL9JuhjXX6dRnguRVOfk2GZaDpPjcOKiiXCPINZC1GczQ7iTq3Zw==" 561 | }, 562 | "node-fetch": { 563 | "version": "2.6.0", 564 | "resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-2.6.0.tgz", 565 | "integrity": "sha512-8dG4H5ujfvFiqDmVu9fQ5bOHUC15JMjMY/Zumv26oOvvVJjM67KF8koCWIabKQ1GJIa9r2mMZscBq/TbdOcmNA==" 566 | }, 567 | "object-assign": { 568 | "version": "4.1.1", 569 | "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", 570 | "integrity": "sha1-IQmtx5ZYh8/AXLvUQsrIv7s2CGM=" 571 | }, 572 | "on-finished": { 573 | "version": "2.3.0", 574 | "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.3.0.tgz", 575 | "integrity": "sha1-IPEzZIGwg811M3mSoWlxqi2QaUc=", 576 | "requires": { 577 | "ee-first": "1.1.1" 578 | } 579 | }, 580 | "on-headers": { 581 | "version": "1.0.2", 582 | "resolved": "https://registry.npmjs.org/on-headers/-/on-headers-1.0.2.tgz", 583 | "integrity": "sha512-pZAE+FJLoyITytdqK0U5s+FIpjN0JP3OzFi/u8Rx+EV5/W+JTWGXG8xFzevE7AjBfDqHv/8vL8qQsIhHnqRkrA==" 584 | }, 585 | "optimist": { 586 | "version": "0.6.1", 587 | "resolved": "https://registry.npmjs.org/optimist/-/optimist-0.6.1.tgz", 588 | "integrity": "sha1-2j6nRob6IaGaERwybpDrFaAZZoY=", 589 | "requires": { 590 | "minimist": "~0.0.1", 591 | "wordwrap": "~0.0.2" 592 | } 593 | }, 594 | "p-is-promise": { 595 | "version": "1.1.0", 596 | "resolved": "https://registry.npmjs.org/p-is-promise/-/p-is-promise-1.1.0.tgz", 597 | "integrity": "sha1-nJRWmJ6fZYgBewQ01WCXZ1w9oF4=" 598 | }, 599 | "parseurl": { 600 | "version": "1.3.3", 601 | "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz", 602 | "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==" 603 | }, 604 | "path-to-regexp": { 605 | "version": "0.1.7", 606 | "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz", 607 | "integrity": "sha1-32BBeABfUi8V60SQ5yR6G/qmf4w=" 608 | }, 609 | "process-nextick-args": { 610 | "version": "2.0.1", 611 | "resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-2.0.1.tgz", 612 | "integrity": "sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag==" 613 | }, 614 | "proxy-addr": { 615 | "version": "2.0.5", 616 | "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.5.tgz", 617 | "integrity": "sha512-t/7RxHXPH6cJtP0pRG6smSr9QJidhB+3kXu0KgXnbGYMgzEnUxRQ4/LDdfOwZEMyIh3/xHb8PX3t+lfL9z+YVQ==", 618 | "requires": { 619 | "forwarded": "~0.1.2", 620 | "ipaddr.js": "1.9.0" 621 | } 622 | }, 623 | "psl": { 624 | "version": "1.7.0", 625 | "resolved": "https://registry.npmjs.org/psl/-/psl-1.7.0.tgz", 626 | "integrity": "sha512-5NsSEDv8zY70ScRnOTn7bK7eanl2MvFrOrS/R6x+dBt5g1ghnj9Zv90kO8GwT8gxcu2ANyFprnFYB85IogIJOQ==" 627 | }, 628 | "punycode": { 629 | "version": "2.1.1", 630 | "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.1.1.tgz", 631 | "integrity": "sha512-XRsRjdf+j5ml+y/6GKHPZbrF/8p2Yga0JPtdqTIY2Xe5ohJPD9saDJJLPvp9+NSBprVvevdXZybnj2cv8OEd0A==" 632 | }, 633 | "qs": { 634 | "version": "6.7.0", 635 | "resolved": "https://registry.npmjs.org/qs/-/qs-6.7.0.tgz", 636 | "integrity": "sha512-VCdBRNFTX1fyE7Nb6FYoURo/SPe62QCaAyzJvUjwRaIsc+NePBEniHlvxFmmX56+HZphIGtV0XeCirBtpDrTyQ==" 637 | }, 638 | "range-parser": { 639 | "version": "1.2.1", 640 | "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz", 641 | "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==" 642 | }, 643 | "raw-body": { 644 | "version": "2.4.0", 645 | "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.4.0.tgz", 646 | "integrity": "sha512-4Oz8DUIwdvoa5qMJelxipzi/iJIi40O5cGV1wNYp5hvZP8ZN0T+jiNkL0QepXs+EsQ9XJ8ipEDoiH70ySUJP3Q==", 647 | "requires": { 648 | "bytes": "3.1.0", 649 | "http-errors": "1.7.2", 650 | "iconv-lite": "0.4.24", 651 | "unpipe": "1.0.0" 652 | } 653 | }, 654 | "readable-stream": { 655 | "version": "2.3.7", 656 | "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.7.tgz", 657 | "integrity": "sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw==", 658 | "requires": { 659 | "core-util-is": "~1.0.0", 660 | "inherits": "~2.0.3", 661 | "isarray": "~1.0.0", 662 | "process-nextick-args": "~2.0.0", 663 | "safe-buffer": "~5.1.1", 664 | "string_decoder": "~1.1.1", 665 | "util-deprecate": "~1.0.1" 666 | } 667 | }, 668 | "safe-buffer": { 669 | "version": "5.1.2", 670 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", 671 | "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==" 672 | }, 673 | "safer-buffer": { 674 | "version": "2.1.2", 675 | "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", 676 | "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==" 677 | }, 678 | "sax": { 679 | "version": "1.2.4", 680 | "resolved": "https://registry.npmjs.org/sax/-/sax-1.2.4.tgz", 681 | "integrity": "sha512-NqVDv9TpANUjFm0N8uM5GxL36UgKi9/atZw+x7YFnQ8ckwFGKrl4xX4yWtrey3UJm5nP1kUbnYgLopqWNSRhWw==" 682 | }, 683 | "send": { 684 | "version": "0.17.1", 685 | "resolved": "https://registry.npmjs.org/send/-/send-0.17.1.tgz", 686 | "integrity": "sha512-BsVKsiGcQMFwT8UxypobUKyv7irCNRHk1T0G680vk88yf6LBByGcZJOTJCrTP2xVN6yI+XjPJcNuE3V4fT9sAg==", 687 | "requires": { 688 | "debug": "2.6.9", 689 | "depd": "~1.1.2", 690 | "destroy": "~1.0.4", 691 | "encodeurl": "~1.0.2", 692 | "escape-html": "~1.0.3", 693 | "etag": "~1.8.1", 694 | "fresh": "0.5.2", 695 | "http-errors": "~1.7.2", 696 | "mime": "1.6.0", 697 | "ms": "2.1.1", 698 | "on-finished": "~2.3.0", 699 | "range-parser": "~1.2.1", 700 | "statuses": "~1.5.0" 701 | }, 702 | "dependencies": { 703 | "ms": { 704 | "version": "2.1.1", 705 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.1.tgz", 706 | "integrity": "sha512-tgp+dl5cGk28utYktBsrFqA7HKgrhgPsg6Z/EfhWI4gl1Hwq8B/GmY/0oXZ6nF8hDVesS/FpnYaD/kOWhYQvyg==" 707 | } 708 | } 709 | }, 710 | "serve-favicon": { 711 | "version": "2.4.5", 712 | "resolved": "https://registry.npmjs.org/serve-favicon/-/serve-favicon-2.4.5.tgz", 713 | "integrity": "sha512-s7F8h2NrslMkG50KxvlGdj+ApSwaLex0vexuJ9iFf3GLTIp1ph/l1qZvRe9T9TJEYZgmq72ZwJ2VYiAEtChknw==", 714 | "requires": { 715 | "etag": "~1.8.1", 716 | "fresh": "0.5.2", 717 | "ms": "2.0.0", 718 | "parseurl": "~1.3.2", 719 | "safe-buffer": "5.1.1" 720 | }, 721 | "dependencies": { 722 | "safe-buffer": { 723 | "version": "5.1.1", 724 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.1.tgz", 725 | "integrity": "sha512-kKvNJn6Mm93gAczWVJg7wH+wGYWNrDHdWvpUmHyEsgCtIwwo3bqPtV4tR5tuPaUhTOo/kvhVwd8XwwOllGYkbg==" 726 | } 727 | } 728 | }, 729 | "serve-static": { 730 | "version": "1.14.1", 731 | "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.14.1.tgz", 732 | "integrity": "sha512-JMrvUwE54emCYWlTI+hGrGv5I8dEwmco/00EvkzIIsR7MqrHonbD9pO2MOfFnpFntl7ecpZs+3mW+XbQZu9QCg==", 733 | "requires": { 734 | "encodeurl": "~1.0.2", 735 | "escape-html": "~1.0.3", 736 | "parseurl": "~1.3.3", 737 | "send": "0.17.1" 738 | } 739 | }, 740 | "setprototypeof": { 741 | "version": "1.1.1", 742 | "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.1.tgz", 743 | "integrity": "sha512-JvdAWfbXeIGaZ9cILp38HntZSFSo3mWg6xGcJJsd+d4aRMOqauag1C63dJfDw7OaMYwEbHMOxEZ1lqVRYP2OAw==" 744 | }, 745 | "source-map": { 746 | "version": "0.6.1", 747 | "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", 748 | "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==" 749 | }, 750 | "statuses": { 751 | "version": "1.5.0", 752 | "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.5.0.tgz", 753 | "integrity": "sha1-Fhx9rBd2Wf2YEfQ3cfqZOBR4Yow=" 754 | }, 755 | "streamsearch": { 756 | "version": "0.1.2", 757 | "resolved": "https://registry.npmjs.org/streamsearch/-/streamsearch-0.1.2.tgz", 758 | "integrity": "sha1-gIudDlb8Jz2Am6VzOOkpkZoanxo=" 759 | }, 760 | "string_decoder": { 761 | "version": "1.1.1", 762 | "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz", 763 | "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==", 764 | "requires": { 765 | "safe-buffer": "~5.1.0" 766 | } 767 | }, 768 | "toidentifier": { 769 | "version": "1.0.0", 770 | "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.0.tgz", 771 | "integrity": "sha512-yaOH/Pk/VEhBWWTlhI+qXxDFXlejDGcQipMlyxda9nthulaxLZUNcUqFxokp0vcYnvteJln5FNQDRrxj3YcbVw==" 772 | }, 773 | "tough-cookie": { 774 | "version": "3.0.1", 775 | "resolved": "https://registry.npmjs.org/tough-cookie/-/tough-cookie-3.0.1.tgz", 776 | "integrity": "sha512-yQyJ0u4pZsv9D4clxO69OEjLWYw+jbgspjTue4lTQZLfV0c5l1VmK2y1JK8E9ahdpltPOaAThPcp5nKPUgSnsg==", 777 | "requires": { 778 | "ip-regex": "^2.1.0", 779 | "psl": "^1.1.28", 780 | "punycode": "^2.1.1" 781 | } 782 | }, 783 | "tslib": { 784 | "version": "1.10.0", 785 | "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.10.0.tgz", 786 | "integrity": "sha512-qOebF53frne81cf0S9B41ByenJ3/IuH8yJKngAX35CmiZySA0khhkovshKK+jGCaMnVomla7gVlIcc3EvKPbTQ==" 787 | }, 788 | "tunnel": { 789 | "version": "0.0.6", 790 | "resolved": "https://registry.npmjs.org/tunnel/-/tunnel-0.0.6.tgz", 791 | "integrity": "sha512-1h/Lnq9yajKY2PEbBadPXj3VxsDDu844OnaAo52UVmIzIvwwtBPIuNvkjuzBlTWpfJyUbG3ez0KSBibQkj4ojg==" 792 | }, 793 | "type-is": { 794 | "version": "1.6.18", 795 | "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz", 796 | "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==", 797 | "requires": { 798 | "media-typer": "0.3.0", 799 | "mime-types": "~2.1.24" 800 | } 801 | }, 802 | "typedarray": { 803 | "version": "0.0.6", 804 | "resolved": "https://registry.npmjs.org/typedarray/-/typedarray-0.0.6.tgz", 805 | "integrity": "sha1-hnrHTjhkGHsdPUfZlqeOxciDB3c=" 806 | }, 807 | "uglify-js": { 808 | "version": "3.7.5", 809 | "resolved": "https://registry.npmjs.org/uglify-js/-/uglify-js-3.7.5.tgz", 810 | "integrity": "sha512-GFZ3EXRptKGvb/C1Sq6nO1iI7AGcjyqmIyOw0DrD0675e+NNbGO72xmMM2iEBdFbxaTLo70NbjM/Wy54uZIlsg==", 811 | "optional": true, 812 | "requires": { 813 | "commander": "~2.20.3", 814 | "source-map": "~0.6.1" 815 | } 816 | }, 817 | "unpipe": { 818 | "version": "1.0.0", 819 | "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", 820 | "integrity": "sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw=" 821 | }, 822 | "util-deprecate": { 823 | "version": "1.0.2", 824 | "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", 825 | "integrity": "sha1-RQ1Nyfpw3nMnYvvS1KKJgUGaDM8=" 826 | }, 827 | "utils-merge": { 828 | "version": "1.0.1", 829 | "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz", 830 | "integrity": "sha1-n5VxD1CiZ5R7LMwSR0HBAoQn5xM=" 831 | }, 832 | "uuid": { 833 | "version": "3.3.3", 834 | "resolved": "https://registry.npmjs.org/uuid/-/uuid-3.3.3.tgz", 835 | "integrity": "sha512-pW0No1RGHgzlpHJO1nsVrHKpOEIxkGg1xB+v0ZmdNH5OAeAwzAVrCnI2/6Mtx+Uys6iaylxa+D3g4j63IKKjSQ==" 836 | }, 837 | "vary": { 838 | "version": "1.1.2", 839 | "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz", 840 | "integrity": "sha1-IpnwLG3tMNSllhsLn3RSShj2NPw=" 841 | }, 842 | "walk": { 843 | "version": "2.3.14", 844 | "resolved": "https://registry.npmjs.org/walk/-/walk-2.3.14.tgz", 845 | "integrity": "sha512-5skcWAUmySj6hkBdH6B6+3ddMjVQYH5Qy9QGbPmN8kVmLteXk+yVXg+yfk1nbX30EYakahLrr8iPcCxJQSCBeg==", 846 | "requires": { 847 | "foreachasync": "^3.0.0" 848 | } 849 | }, 850 | "wordwrap": { 851 | "version": "0.0.3", 852 | "resolved": "https://registry.npmjs.org/wordwrap/-/wordwrap-0.0.3.tgz", 853 | "integrity": "sha1-o9XabNXAvAAI03I0u68b7WMFkQc=" 854 | }, 855 | "xml2js": { 856 | "version": "0.4.23", 857 | "resolved": "https://registry.npmjs.org/xml2js/-/xml2js-0.4.23.tgz", 858 | "integrity": "sha512-ySPiMjM0+pLDftHgXY4By0uswI3SPKLDw/i3UXbnO8M/p28zqexCUoPmQFrYD+/1BzhGJSs2i1ERWKJAtiLrug==", 859 | "requires": { 860 | "sax": ">=0.6.0", 861 | "xmlbuilder": "~11.0.0" 862 | } 863 | }, 864 | "xmlbuilder": { 865 | "version": "11.0.1", 866 | "resolved": "https://registry.npmjs.org/xmlbuilder/-/xmlbuilder-11.0.1.tgz", 867 | "integrity": "sha512-fDlsI/kFEx7gLvbecc0/ohLG50fugQp8ryHzMTuW9vSa1GJ0XYWKnhsUx7oie3G98+r56aTQIUB4kht42R3JvA==" 868 | }, 869 | "xtend": { 870 | "version": "4.0.2", 871 | "resolved": "https://registry.npmjs.org/xtend/-/xtend-4.0.2.tgz", 872 | "integrity": "sha512-LKYU1iAXJXUgAXn9URjiu+MWhyUXHsvfp7mcuYm9dSUKK0/CjtrUwFAxD82/mCWbtLsGjFIad0wIsod4zrTAEQ==" 873 | } 874 | } 875 | } 876 | -------------------------------------------------------------------------------- /v10/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "node-blobs-image-resize", 3 | "version": "0.0.0", 4 | "private": true, 5 | "scripts": { 6 | "start": "node ./bin/www" 7 | }, 8 | "dependencies": { 9 | "@azure/storage-blob": "^10.3.0", 10 | "body-parser": "^1.18.3", 11 | "cookie-parser": "~1.4.3", 12 | "debug": "~2.6.9", 13 | "dotenv": "^5.0.1", 14 | "express": "^4.16.4", 15 | "hbs": "~4.0.1", 16 | "into-stream": "^3.1.0", 17 | "morgan": "^1.9.1", 18 | "multer": "^1.4.1", 19 | "serve-favicon": "~2.4.5" 20 | } 21 | } 22 | -------------------------------------------------------------------------------- /v10/public/favicon.ico: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Azure-Samples/azure-sdk-for-js-storage-blob-stream-nodejs/50eb069440e805489c5d8c7232b4735f130322c2/v10/public/favicon.ico -------------------------------------------------------------------------------- /v10/public/stylesheets/style.css: -------------------------------------------------------------------------------- 1 | body { 2 | padding: 50px; 3 | font: 14px "Lucida Grande", Helvetica, Arial, sans-serif; 4 | } 5 | 6 | a { 7 | color: #00B7FF; 8 | } 9 | 10 | .upload-container { 11 | border: solid 1px #ccc; 12 | padding: 10px; 13 | max-width: fit-content; 14 | margin-bottom: 10px 15 | } -------------------------------------------------------------------------------- /v10/routes/index.js: -------------------------------------------------------------------------------- 1 | if (process.env.NODE_ENV !== 'production') { 2 | require('dotenv').load(); 3 | } 4 | 5 | const { 6 | Aborter, 7 | ContainerURL, 8 | ServiceURL, 9 | StorageURL, 10 | SharedKeyCredential 11 | } = require('@azure/storage-blob'); 12 | 13 | const express = require('express'); 14 | const router = express.Router(); 15 | const containerName = 'thumbnails'; 16 | 17 | const sharedKeyCredential = new SharedKeyCredential( 18 | process.env.AZURE_STORAGE_ACCOUNT_NAME, 19 | process.env.AZURE_STORAGE_ACCOUNT_ACCESS_KEY); 20 | const pipeline = StorageURL.newPipeline(sharedKeyCredential); 21 | 22 | const serviceURL = new ServiceURL( 23 | `https://${process.env.AZURE_STORAGE_ACCOUNT_NAME}.blob.core.windows.net`, 24 | pipeline 25 | ); 26 | 27 | const containerURL = ContainerURL.fromServiceURL(serviceURL, containerName); 28 | 29 | router.get('/', async (req, res, next) => { 30 | 31 | let viewData; 32 | 33 | try { 34 | 35 | const listBlobsResponse = await containerURL.listBlobFlatSegment(Aborter.none); 36 | 37 | for (const blob of listBlobsResponse.segment.blobItems) { 38 | console.log(`Blob: ${blob.name}`); 39 | } 40 | 41 | viewData = { 42 | title: 'Home', 43 | viewName: 'index', 44 | accountName: process.env.AZURE_STORAGE_ACCOUNT_NAME, 45 | containerName: containerName 46 | }; 47 | 48 | if (listBlobsResponse.segment.blobItems.length) { 49 | viewData.thumbnails = listBlobsResponse.segment.blobItems; 50 | } 51 | 52 | } catch (err) { 53 | 54 | viewData = { 55 | title: 'Error', 56 | viewName: 'error', 57 | message: 'There was an error contacting the blob storage container.', 58 | error: err 59 | }; 60 | 61 | res.status(500); 62 | 63 | } finally { 64 | 65 | res.render(viewData.viewName, viewData); 66 | 67 | } 68 | }); 69 | 70 | module.exports = router; -------------------------------------------------------------------------------- /v10/routes/upload.js: -------------------------------------------------------------------------------- 1 | if (process.env.NODE_ENV !== 'production') { 2 | require('dotenv').load(); 3 | } 4 | 5 | const { 6 | Aborter, 7 | BlobURL, 8 | BlockBlobURL, 9 | ContainerURL, 10 | ServiceURL, 11 | StorageURL, 12 | SharedKeyCredential, 13 | uploadStreamToBlockBlob 14 | } = require('@azure/storage-blob'); 15 | 16 | const express = require('express'); 17 | const router = express.Router(); 18 | const multer = require('multer'); 19 | const inMemoryStorage = multer.memoryStorage(); 20 | const uploadStrategy = multer({ storage: inMemoryStorage }).single('image'); 21 | const getStream = require('into-stream'); 22 | const containerName = 'images'; 23 | const ONE_MEGABYTE = 1024 * 1024; 24 | const uploadOptions = { bufferSize: 4 * ONE_MEGABYTE, maxBuffers: 20 }; 25 | const ONE_MINUTE = 60 * 1000; 26 | 27 | const sharedKeyCredential = new SharedKeyCredential( 28 | process.env.AZURE_STORAGE_ACCOUNT_NAME, 29 | process.env.AZURE_STORAGE_ACCOUNT_ACCESS_KEY); 30 | const pipeline = StorageURL.newPipeline(sharedKeyCredential); 31 | const serviceURL = new ServiceURL( 32 | `https://${process.env.AZURE_STORAGE_ACCOUNT_NAME}.blob.core.windows.net`, 33 | pipeline 34 | ); 35 | 36 | const getBlobName = originalName => { 37 | // Use a random number to generate a unique file name, 38 | // removing "0." from the start of the string. 39 | const identifier = Math.random().toString().replace(/0\./, ''); 40 | return `${identifier}-${originalName}`; 41 | }; 42 | 43 | router.post('/', uploadStrategy, async (req, res) => { 44 | 45 | const aborter = Aborter.timeout(30 * ONE_MINUTE); 46 | const blobName = getBlobName(req.file.originalname); 47 | const stream = getStream(req.file.buffer); 48 | const containerURL = ContainerURL.fromServiceURL(serviceURL, containerName); 49 | const blockBlobURL = BlockBlobURL.fromContainerURL(containerURL, blobName); 50 | 51 | try { 52 | 53 | await uploadStreamToBlockBlob(aborter, stream, 54 | blockBlobURL, uploadOptions.bufferSize, uploadOptions.maxBuffers, 55 | { blobHTTPHeaders: { blobContentType: "image/jpeg" } }); 56 | 57 | res.render('success', { message: 'File uploaded to Azure Blob storage.' }); 58 | 59 | } catch (err) { 60 | 61 | res.render('error', { message: err.message }); 62 | 63 | } 64 | }); 65 | 66 | module.exports = router; -------------------------------------------------------------------------------- /v10/views/error.hbs: -------------------------------------------------------------------------------- 1 |

{{message}}

2 |

{{error.status}}

3 |
{{error.stack}}
4 | -------------------------------------------------------------------------------- /v10/views/index.hbs: -------------------------------------------------------------------------------- 1 |

Image Resizer

2 | 3 |

Upload photos

4 |
5 |

Select an image to upload:

6 |
7 |
8 |
9 | 10 |

{{message}}

11 | 12 |

Generated Thumbnails

13 | {{#each this.thumbnails}} 14 | 15 | {{/each}} 16 | 17 |
18 |

This app has no official privacy policy. Your data will be uploaded to a service in order to produce a thumbnail. Your images will be public once you upload them and there is no automated way to remove them.

19 | 20 | -------------------------------------------------------------------------------- /v10/views/layout.hbs: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | {{title}} 5 | 6 | 7 | 8 | {{{body}}} 9 | 10 | 11 | -------------------------------------------------------------------------------- /v10/views/success.hbs: -------------------------------------------------------------------------------- 1 |

Success!

2 | 3 |

{{message}}

4 | 5 | Return home to view thumbnails -------------------------------------------------------------------------------- /v12/.env.example: -------------------------------------------------------------------------------- 1 | AZURE_STORAGE_ACCOUNT_NAME= 2 | AZURE_STORAGE_ACCOUNT_ACCESS_KEY= -------------------------------------------------------------------------------- /v12/.gitignore: -------------------------------------------------------------------------------- 1 | # Logs 2 | logs 3 | *.log 4 | npm-debug.log* 5 | yarn-debug.log* 6 | yarn-error.log* 7 | 8 | # Runtime data 9 | pids 10 | *.pid 11 | *.seed 12 | *.pid.lock 13 | 14 | # Directory for instrumented libs generated by jscoverage/JSCover 15 | lib-cov 16 | 17 | # Coverage directory used by tools like istanbul 18 | coverage 19 | 20 | # nyc test coverage 21 | .nyc_output 22 | 23 | # Grunt intermediate storage (http://gruntjs.com/creating-plugins#storing-task-files) 24 | .grunt 25 | 26 | # Bower dependency directory (https://bower.io/) 27 | bower_components 28 | 29 | # node-waf configuration 30 | .lock-wscript 31 | 32 | # Compiled binary addons (https://nodejs.org/api/addons.html) 33 | build/Release 34 | 35 | # Dependency directories 36 | node_modules/ 37 | jspm_packages/ 38 | 39 | # TypeScript v1 declaration files 40 | typings/ 41 | 42 | # Optional npm cache directory 43 | .npm 44 | 45 | # Optional eslint cache 46 | .eslintcache 47 | 48 | # Optional REPL history 49 | .node_repl_history 50 | 51 | # Output of 'npm pack' 52 | *.tgz 53 | 54 | # Yarn Integrity file 55 | .yarn-integrity 56 | 57 | # dotenv environment variables file 58 | .env 59 | 60 | # next.js build output 61 | .next 62 | -------------------------------------------------------------------------------- /v12/app.js: -------------------------------------------------------------------------------- 1 | const express = require('express'); 2 | const path = require('path'); 3 | const favicon = require('serve-favicon'); 4 | const logger = require('morgan'); 5 | const cookieParser = require('cookie-parser'); 6 | const bodyParser = require('body-parser'); 7 | const index = require('./routes/index'); 8 | const app = express(); 9 | 10 | // view engine setup 11 | app.set('views', path.join(__dirname, 'views')); 12 | app.set('view engine', 'hbs'); 13 | 14 | app.use(favicon(path.join(__dirname, 'public', 'favicon.ico'))); 15 | app.use(logger('dev')); 16 | app.use(bodyParser.json()); 17 | app.use(bodyParser.urlencoded({ extended: false })); 18 | app.use(cookieParser()); 19 | app.use(express.static(path.join(__dirname, 'public'))); 20 | 21 | app.use('/', index); 22 | 23 | // catch 404 and forward to error handler 24 | app.use((req, res, next) => { 25 | const err = new Error('Not Found'); 26 | err.status = 404; 27 | next(err); 28 | }); 29 | 30 | // error handler 31 | app.use((err, req, res, next) => { 32 | // set locals, only providing error in development 33 | res.locals.message = err.message; 34 | res.locals.error = req.app.get('env') === 'development' ? err : {}; 35 | 36 | // render the error page 37 | res.status(err.status || 500); 38 | res.render('error'); 39 | }); 40 | 41 | module.exports = app; 42 | -------------------------------------------------------------------------------- /v12/bin/www: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env node 2 | 3 | /** 4 | * Module dependencies. 5 | */ 6 | 7 | var app = require('../app'); 8 | var debug = require('debug')('node-blobs-image-resize:server'); 9 | var http = require('http'); 10 | 11 | /** 12 | * Get port from environment and store in Express. 13 | */ 14 | 15 | var port = normalizePort(process.env.PORT || '3000'); 16 | app.set('port', port); 17 | 18 | /** 19 | * Create HTTP server. 20 | */ 21 | 22 | var server = http.createServer(app); 23 | 24 | /** 25 | * Listen on provided port, on all network interfaces. 26 | */ 27 | 28 | server.listen(port); 29 | server.on('error', onError); 30 | server.on('listening', onListening); 31 | 32 | /** 33 | * Normalize a port into a number, string, or false. 34 | */ 35 | 36 | function normalizePort(val) { 37 | var port = parseInt(val, 10); 38 | 39 | if (isNaN(port)) { 40 | // named pipe 41 | return val; 42 | } 43 | 44 | if (port >= 0) { 45 | // port number 46 | return port; 47 | } 48 | 49 | return false; 50 | } 51 | 52 | /** 53 | * Event listener for HTTP server "error" event. 54 | */ 55 | 56 | function onError(error) { 57 | if (error.syscall !== 'listen') { 58 | throw error; 59 | } 60 | 61 | var bind = typeof port === 'string' 62 | ? 'Pipe ' + port 63 | : 'Port ' + port; 64 | 65 | // handle specific listen errors with friendly messages 66 | switch (error.code) { 67 | case 'EACCES': 68 | console.error(bind + ' requires elevated privileges'); 69 | process.exit(1); 70 | break; 71 | case 'EADDRINUSE': 72 | console.error(bind + ' is already in use'); 73 | process.exit(1); 74 | break; 75 | default: 76 | throw error; 77 | } 78 | } 79 | 80 | /** 81 | * Event listener for HTTP server "listening" event. 82 | */ 83 | 84 | function onListening() { 85 | var addr = server.address(); 86 | var bind = typeof addr === 'string' 87 | ? 'pipe ' + addr 88 | : 'port ' + addr.port; 89 | debug('Listening on ' + bind); 90 | } 91 | -------------------------------------------------------------------------------- /v12/package-lock.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "node-blobs-image-resize", 3 | "version": "0.0.12", 4 | "lockfileVersion": 1, 5 | "requires": true, 6 | "dependencies": { 7 | "@azure/abort-controller": { 8 | "version": "1.0.1", 9 | "resolved": "https://registry.npmjs.org/@azure/abort-controller/-/abort-controller-1.0.1.tgz", 10 | "integrity": "sha512-wP2Jw6uPp8DEDy0n4KNidvwzDjyVV2xnycEIq7nPzj1rHyb/r+t3OPeNT1INZePP2wy5ZqlwyuyOMTi0ePyY1A==", 11 | "requires": { 12 | "tslib": "^1.9.3" 13 | } 14 | }, 15 | "@azure/core-asynciterator-polyfill": { 16 | "version": "1.0.0", 17 | "resolved": "https://registry.npmjs.org/@azure/core-asynciterator-polyfill/-/core-asynciterator-polyfill-1.0.0.tgz", 18 | "integrity": "sha512-kmv8CGrPfN9SwMwrkiBK9VTQYxdFQEGe0BmQk+M8io56P9KNzpAxcWE/1fxJj7uouwN4kXF0BHW8DNlgx+wtCg==" 19 | }, 20 | "@azure/core-auth": { 21 | "version": "1.1.2", 22 | "resolved": "https://registry.npmjs.org/@azure/core-auth/-/core-auth-1.1.2.tgz", 23 | "integrity": "sha512-IUbP/f3v96dpHgXUwsAjUwDzjlUjawyUhWhGKKB6Qxy+iqppC/pVBPyc6kdpyTe7H30HN+4H3f0lar7Wp9Hx6A==", 24 | "requires": { 25 | "@azure/abort-controller": "^1.0.0", 26 | "@azure/core-tracing": "1.0.0-preview.8", 27 | "@opentelemetry/api": "^0.6.1", 28 | "tslib": "^1.10.0" 29 | } 30 | }, 31 | "@azure/core-http": { 32 | "version": "1.1.3", 33 | "resolved": "https://registry.npmjs.org/@azure/core-http/-/core-http-1.1.3.tgz", 34 | "integrity": "sha512-GysW3+BRVV4L9cs3GsuCbnlyibrQU6hh5mcJ7hlnk7tdUBzWybUvJ8/P/nHX49PgwRmi81pD5v1ht2jF0IzxAQ==", 35 | "requires": { 36 | "@azure/abort-controller": "^1.0.0", 37 | "@azure/core-auth": "^1.1.2", 38 | "@azure/core-tracing": "1.0.0-preview.8", 39 | "@azure/logger": "^1.0.0", 40 | "@opentelemetry/api": "^0.6.1", 41 | "@types/node-fetch": "^2.5.0", 42 | "@types/tunnel": "^0.0.1", 43 | "form-data": "^3.0.0", 44 | "node-fetch": "^2.6.0", 45 | "process": "^0.11.10", 46 | "tough-cookie": "^4.0.0", 47 | "tslib": "^1.10.0", 48 | "tunnel": "^0.0.6", 49 | "uuid": "^8.1.0", 50 | "xml2js": "^0.4.19" 51 | } 52 | }, 53 | "@azure/core-lro": { 54 | "version": "1.0.2", 55 | "resolved": "https://registry.npmjs.org/@azure/core-lro/-/core-lro-1.0.2.tgz", 56 | "integrity": "sha512-Yr0JD7GKryOmbcb5wHCQoQ4KCcH5QJWRNorofid+UvudLaxnbCfvKh/cUfQsGUqRjO9L/Bw4X7FP824DcHdMxw==", 57 | "requires": { 58 | "@azure/abort-controller": "^1.0.0", 59 | "@azure/core-http": "^1.1.1", 60 | "events": "^3.0.0", 61 | "tslib": "^1.10.0" 62 | } 63 | }, 64 | "@azure/core-paging": { 65 | "version": "1.1.1", 66 | "resolved": "https://registry.npmjs.org/@azure/core-paging/-/core-paging-1.1.1.tgz", 67 | "integrity": "sha512-hqEJBEGKan4YdOaL9ZG/GRG6PXaFd/Wb3SSjQW4LWotZzgl6xqG00h6wmkrpd2NNkbBkD1erLHBO3lPHApv+iQ==", 68 | "requires": { 69 | "@azure/core-asynciterator-polyfill": "^1.0.0" 70 | } 71 | }, 72 | "@azure/core-tracing": { 73 | "version": "1.0.0-preview.8", 74 | "resolved": "https://registry.npmjs.org/@azure/core-tracing/-/core-tracing-1.0.0-preview.8.tgz", 75 | "integrity": "sha512-ZKUpCd7Dlyfn7bdc+/zC/sf0aRIaNQMDuSj2RhYRFe3p70hVAnYGp3TX4cnG2yoEALp/LTj/XnZGQ8Xzf6Ja/Q==", 76 | "requires": { 77 | "@opencensus/web-types": "0.0.7", 78 | "@opentelemetry/api": "^0.6.1", 79 | "tslib": "^1.10.0" 80 | } 81 | }, 82 | "@azure/logger": { 83 | "version": "1.0.0", 84 | "resolved": "https://registry.npmjs.org/@azure/logger/-/logger-1.0.0.tgz", 85 | "integrity": "sha512-g2qLDgvmhyIxR3JVS8N67CyIOeFRKQlX/llxYJQr1OSGQqM3HTpVP8MjmjcEKbL/OIt2N9C9UFaNQuKOw1laOA==", 86 | "requires": { 87 | "tslib": "^1.9.3" 88 | } 89 | }, 90 | "@azure/storage-blob": { 91 | "version": "12.1.2", 92 | "resolved": "https://registry.npmjs.org/@azure/storage-blob/-/storage-blob-12.1.2.tgz", 93 | "integrity": "sha512-PCHgG4r3xLt5FaFj+uiMqrRpuzD3TD17cvxCeA1JKK2bJEf8b07H3QRLQVf0DM1MmvYY8FgQagkWZTp+jr9yew==", 94 | "requires": { 95 | "@azure/abort-controller": "^1.0.0", 96 | "@azure/core-http": "^1.1.1", 97 | "@azure/core-lro": "^1.0.2", 98 | "@azure/core-paging": "^1.1.1", 99 | "@azure/core-tracing": "1.0.0-preview.8", 100 | "@azure/logger": "^1.0.0", 101 | "@opentelemetry/api": "^0.6.1", 102 | "events": "^3.0.0", 103 | "tslib": "^1.10.0" 104 | } 105 | }, 106 | "@opencensus/web-types": { 107 | "version": "0.0.7", 108 | "resolved": "https://registry.npmjs.org/@opencensus/web-types/-/web-types-0.0.7.tgz", 109 | "integrity": "sha512-xB+w7ZDAu3YBzqH44rCmG9/RlrOmFuDPt/bpf17eJr8eZSrLt7nc7LnWdxM9Mmoj/YKMHpxRg28txu3TcpiL+g==" 110 | }, 111 | "@opentelemetry/api": { 112 | "version": "0.6.1", 113 | "resolved": "https://registry.npmjs.org/@opentelemetry/api/-/api-0.6.1.tgz", 114 | "integrity": "sha512-wpufGZa7tTxw7eAsjXJtiyIQ42IWQdX9iUQp7ACJcKo1hCtuhLU+K2Nv1U6oRwT1oAlZTE6m4CgWKZBhOiau3Q==", 115 | "requires": { 116 | "@opentelemetry/context-base": "^0.6.1" 117 | } 118 | }, 119 | "@opentelemetry/context-base": { 120 | "version": "0.6.1", 121 | "resolved": "https://registry.npmjs.org/@opentelemetry/context-base/-/context-base-0.6.1.tgz", 122 | "integrity": "sha512-5bHhlTBBq82ti3qPT15TRxkYTFPPQWbnkkQkmHPtqiS1XcTB69cEKd3Jm7Cfi/vkPoyxapmePE9tyA7EzLt8SQ==" 123 | }, 124 | "@types/node": { 125 | "version": "14.0.14", 126 | "resolved": "https://registry.npmjs.org/@types/node/-/node-14.0.14.tgz", 127 | "integrity": "sha512-syUgf67ZQpaJj01/tRTknkMNoBBLWJOBODF0Zm4NrXmiSuxjymFrxnTu1QVYRubhVkRcZLYZG8STTwJRdVm/WQ==" 128 | }, 129 | "@types/node-fetch": { 130 | "version": "2.5.7", 131 | "resolved": "https://registry.npmjs.org/@types/node-fetch/-/node-fetch-2.5.7.tgz", 132 | "integrity": "sha512-o2WVNf5UhWRkxlf6eq+jMZDu7kjgpgJfl4xVNlvryc95O/6F2ld8ztKX+qu+Rjyet93WAWm5LjeX9H5FGkODvw==", 133 | "requires": { 134 | "@types/node": "*", 135 | "form-data": "^3.0.0" 136 | } 137 | }, 138 | "@types/tunnel": { 139 | "version": "0.0.1", 140 | "resolved": "https://registry.npmjs.org/@types/tunnel/-/tunnel-0.0.1.tgz", 141 | "integrity": "sha512-AOqu6bQu5MSWwYvehMXLukFHnupHrpZ8nvgae5Ggie9UwzDR1CCwoXgSSWNZJuyOlCdfdsWMA5F2LlmvyoTv8A==", 142 | "requires": { 143 | "@types/node": "*" 144 | } 145 | }, 146 | "accepts": { 147 | "version": "1.3.7", 148 | "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.7.tgz", 149 | "integrity": "sha512-Il80Qs2WjYlJIBNzNkK6KYqlVMTbZLXgHx2oT0pU/fjRHyEp+PEfEPY0R3WCwAGVOtauxh1hOxNgIf5bv7dQpA==", 150 | "requires": { 151 | "mime-types": "~2.1.24", 152 | "negotiator": "0.6.2" 153 | } 154 | }, 155 | "append-field": { 156 | "version": "1.0.0", 157 | "resolved": "https://registry.npmjs.org/append-field/-/append-field-1.0.0.tgz", 158 | "integrity": "sha1-HjRA6RXwsSA9I3SOeO3XubW0PlY=" 159 | }, 160 | "array-flatten": { 161 | "version": "1.1.1", 162 | "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz", 163 | "integrity": "sha1-ml9pkFGx5wczKPKgCJaLZOopVdI=" 164 | }, 165 | "asynckit": { 166 | "version": "0.4.0", 167 | "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", 168 | "integrity": "sha1-x57Zf380y48robyXkLzDZkdLS3k=" 169 | }, 170 | "basic-auth": { 171 | "version": "2.0.1", 172 | "resolved": "https://registry.npmjs.org/basic-auth/-/basic-auth-2.0.1.tgz", 173 | "integrity": "sha512-NF+epuEdnUYVlGuhaxbbq+dvJttwLnGY+YixlXlME5KpQ5W3CnXA5cVTneY3SPbPDRkcjMbifrwmFYcClgOZeg==", 174 | "requires": { 175 | "safe-buffer": "5.1.2" 176 | } 177 | }, 178 | "body-parser": { 179 | "version": "1.19.0", 180 | "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.19.0.tgz", 181 | "integrity": "sha512-dhEPs72UPbDnAQJ9ZKMNTP6ptJaionhP5cBb541nXPlW60Jepo9RV/a4fX4XWW9CuFNK22krhrj1+rgzifNCsw==", 182 | "requires": { 183 | "bytes": "3.1.0", 184 | "content-type": "~1.0.4", 185 | "debug": "2.6.9", 186 | "depd": "~1.1.2", 187 | "http-errors": "1.7.2", 188 | "iconv-lite": "0.4.24", 189 | "on-finished": "~2.3.0", 190 | "qs": "6.7.0", 191 | "raw-body": "2.4.0", 192 | "type-is": "~1.6.17" 193 | }, 194 | "dependencies": { 195 | "debug": { 196 | "version": "2.6.9", 197 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", 198 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", 199 | "requires": { 200 | "ms": "2.0.0" 201 | } 202 | } 203 | } 204 | }, 205 | "buffer-from": { 206 | "version": "1.1.1", 207 | "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.1.tgz", 208 | "integrity": "sha512-MQcXEUbCKtEo7bhqEs6560Hyd4XaovZlO/k9V3hjVUF/zwW7KBVdSK4gIt/bzwS9MbR5qob+F5jusZsb0YQK2A==" 209 | }, 210 | "busboy": { 211 | "version": "0.2.14", 212 | "resolved": "https://registry.npmjs.org/busboy/-/busboy-0.2.14.tgz", 213 | "integrity": "sha1-bCpiLvz0fFe7vh4qnDetNseSVFM=", 214 | "requires": { 215 | "dicer": "0.2.5", 216 | "readable-stream": "1.1.x" 217 | }, 218 | "dependencies": { 219 | "isarray": { 220 | "version": "0.0.1", 221 | "resolved": "https://registry.npmjs.org/isarray/-/isarray-0.0.1.tgz", 222 | "integrity": "sha1-ihis/Kmo9Bd+Cav8YDiTmwXR7t8=" 223 | }, 224 | "readable-stream": { 225 | "version": "1.1.14", 226 | "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-1.1.14.tgz", 227 | "integrity": "sha1-fPTFTvZI44EwhMY23SB54WbAgdk=", 228 | "requires": { 229 | "core-util-is": "~1.0.0", 230 | "inherits": "~2.0.1", 231 | "isarray": "0.0.1", 232 | "string_decoder": "~0.10.x" 233 | } 234 | }, 235 | "string_decoder": { 236 | "version": "0.10.31", 237 | "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-0.10.31.tgz", 238 | "integrity": "sha1-YuIDvEF2bGwoyfyEMB2rHFMQ+pQ=" 239 | } 240 | } 241 | }, 242 | "bytes": { 243 | "version": "3.1.0", 244 | "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.0.tgz", 245 | "integrity": "sha512-zauLjrfCG+xvoyaqLoV8bLVXXNGC4JqlxFCutSDWA6fJrTo2ZuvLYTqZ7aHBLZSMOopbzwv8f+wZcVzfVTI2Dg==" 246 | }, 247 | "combined-stream": { 248 | "version": "1.0.8", 249 | "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz", 250 | "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==", 251 | "requires": { 252 | "delayed-stream": "~1.0.0" 253 | } 254 | }, 255 | "concat-stream": { 256 | "version": "1.6.2", 257 | "resolved": "https://registry.npmjs.org/concat-stream/-/concat-stream-1.6.2.tgz", 258 | "integrity": "sha512-27HBghJxjiZtIk3Ycvn/4kbJk/1uZuJFfuPEns6LaEvpvG1f0hTea8lilrouyo9mVc2GWdcEZ8OLoGmSADlrCw==", 259 | "requires": { 260 | "buffer-from": "^1.0.0", 261 | "inherits": "^2.0.3", 262 | "readable-stream": "^2.2.2", 263 | "typedarray": "^0.0.6" 264 | } 265 | }, 266 | "content-disposition": { 267 | "version": "0.5.3", 268 | "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.3.tgz", 269 | "integrity": "sha512-ExO0774ikEObIAEV9kDo50o+79VCUdEB6n6lzKgGwupcVeRlhrj3qGAfwq8G6uBJjkqLrhT0qEYFcWng8z1z0g==", 270 | "requires": { 271 | "safe-buffer": "5.1.2" 272 | } 273 | }, 274 | "content-type": { 275 | "version": "1.0.4", 276 | "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.4.tgz", 277 | "integrity": "sha512-hIP3EEPs8tB9AT1L+NUqtwOAps4mk2Zob89MWXMHjHWg9milF/j4osnnQLXBCBFBk/tvIG/tUc9mOUJiPBhPXA==" 278 | }, 279 | "cookie": { 280 | "version": "0.4.0", 281 | "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.4.0.tgz", 282 | "integrity": "sha512-+Hp8fLp57wnUSt0tY0tHEXh4voZRDnoIrZPqlo3DPiI4y9lwg/jqx+1Om94/W6ZaPDOUbnjOt/99w66zk+l1Xg==" 283 | }, 284 | "cookie-parser": { 285 | "version": "1.4.5", 286 | "resolved": "https://registry.npmjs.org/cookie-parser/-/cookie-parser-1.4.5.tgz", 287 | "integrity": "sha512-f13bPUj/gG/5mDr+xLmSxxDsB9DQiTIfhJS/sqjrmfAWiAN+x2O4i/XguTL9yDZ+/IFDanJ+5x7hC4CXT9Tdzw==", 288 | "requires": { 289 | "cookie": "0.4.0", 290 | "cookie-signature": "1.0.6" 291 | } 292 | }, 293 | "cookie-signature": { 294 | "version": "1.0.6", 295 | "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz", 296 | "integrity": "sha1-4wOogrNCzD7oylE6eZmXNNqzriw=" 297 | }, 298 | "core-util-is": { 299 | "version": "1.0.2", 300 | "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.2.tgz", 301 | "integrity": "sha1-tf1UIgqivFq1eqtxQMlAdUUDwac=" 302 | }, 303 | "debug": { 304 | "version": "4.1.1", 305 | "resolved": "https://registry.npmjs.org/debug/-/debug-4.1.1.tgz", 306 | "integrity": "sha512-pYAIzeRo8J6KPEaJ0VWOh5Pzkbw/RetuzehGM7QRRX5he4fPHx2rdKMB256ehJCkX+XRQm16eZLqLNS8RSZXZw==", 307 | "requires": { 308 | "ms": "^2.1.1" 309 | }, 310 | "dependencies": { 311 | "ms": { 312 | "version": "2.1.2", 313 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", 314 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==" 315 | } 316 | } 317 | }, 318 | "delayed-stream": { 319 | "version": "1.0.0", 320 | "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", 321 | "integrity": "sha1-3zrhmayt+31ECqrgsp4icrJOxhk=" 322 | }, 323 | "depd": { 324 | "version": "1.1.2", 325 | "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.2.tgz", 326 | "integrity": "sha1-m81S4UwJd2PnSbJ0xDRu0uVgtak=" 327 | }, 328 | "destroy": { 329 | "version": "1.0.4", 330 | "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.0.4.tgz", 331 | "integrity": "sha1-l4hXRCxEdJ5CBmE+N5RiBYJqvYA=" 332 | }, 333 | "dicer": { 334 | "version": "0.2.5", 335 | "resolved": "https://registry.npmjs.org/dicer/-/dicer-0.2.5.tgz", 336 | "integrity": "sha1-WZbAhrszIYyBLAkL3cCc0S+stw8=", 337 | "requires": { 338 | "readable-stream": "1.1.x", 339 | "streamsearch": "0.1.2" 340 | }, 341 | "dependencies": { 342 | "isarray": { 343 | "version": "0.0.1", 344 | "resolved": "https://registry.npmjs.org/isarray/-/isarray-0.0.1.tgz", 345 | "integrity": "sha1-ihis/Kmo9Bd+Cav8YDiTmwXR7t8=" 346 | }, 347 | "readable-stream": { 348 | "version": "1.1.14", 349 | "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-1.1.14.tgz", 350 | "integrity": "sha1-fPTFTvZI44EwhMY23SB54WbAgdk=", 351 | "requires": { 352 | "core-util-is": "~1.0.0", 353 | "inherits": "~2.0.1", 354 | "isarray": "0.0.1", 355 | "string_decoder": "~0.10.x" 356 | } 357 | }, 358 | "string_decoder": { 359 | "version": "0.10.31", 360 | "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-0.10.31.tgz", 361 | "integrity": "sha1-YuIDvEF2bGwoyfyEMB2rHFMQ+pQ=" 362 | } 363 | } 364 | }, 365 | "dotenv": { 366 | "version": "8.2.0", 367 | "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-8.2.0.tgz", 368 | "integrity": "sha512-8sJ78ElpbDJBHNeBzUbUVLsqKdccaa/BXF1uPTw3GrvQTBgrQrtObr2mUrE38vzYd8cEv+m/JBfDLioYcfXoaw==" 369 | }, 370 | "ee-first": { 371 | "version": "1.1.1", 372 | "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", 373 | "integrity": "sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0=" 374 | }, 375 | "encodeurl": { 376 | "version": "1.0.2", 377 | "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz", 378 | "integrity": "sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k=" 379 | }, 380 | "escape-html": { 381 | "version": "1.0.3", 382 | "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz", 383 | "integrity": "sha1-Aljq5NPQwJdN4cFpGI7wBR0dGYg=" 384 | }, 385 | "etag": { 386 | "version": "1.8.1", 387 | "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz", 388 | "integrity": "sha1-Qa4u62XvpiJorr/qg6x9eSmbCIc=" 389 | }, 390 | "events": { 391 | "version": "3.1.0", 392 | "resolved": "https://registry.npmjs.org/events/-/events-3.1.0.tgz", 393 | "integrity": "sha512-Rv+u8MLHNOdMjTAFeT3nCjHn2aGlx435FP/sDHNaRhDEMwyI/aB22Kj2qIN8R0cw3z28psEQLYwxVKLsKrMgWg==" 394 | }, 395 | "express": { 396 | "version": "4.17.1", 397 | "resolved": "https://registry.npmjs.org/express/-/express-4.17.1.tgz", 398 | "integrity": "sha512-mHJ9O79RqluphRrcw2X/GTh3k9tVv8YcoyY4Kkh4WDMUYKRZUq0h1o0w2rrrxBqM7VoeUVqgb27xlEMXTnYt4g==", 399 | "requires": { 400 | "accepts": "~1.3.7", 401 | "array-flatten": "1.1.1", 402 | "body-parser": "1.19.0", 403 | "content-disposition": "0.5.3", 404 | "content-type": "~1.0.4", 405 | "cookie": "0.4.0", 406 | "cookie-signature": "1.0.6", 407 | "debug": "2.6.9", 408 | "depd": "~1.1.2", 409 | "encodeurl": "~1.0.2", 410 | "escape-html": "~1.0.3", 411 | "etag": "~1.8.1", 412 | "finalhandler": "~1.1.2", 413 | "fresh": "0.5.2", 414 | "merge-descriptors": "1.0.1", 415 | "methods": "~1.1.2", 416 | "on-finished": "~2.3.0", 417 | "parseurl": "~1.3.3", 418 | "path-to-regexp": "0.1.7", 419 | "proxy-addr": "~2.0.5", 420 | "qs": "6.7.0", 421 | "range-parser": "~1.2.1", 422 | "safe-buffer": "5.1.2", 423 | "send": "0.17.1", 424 | "serve-static": "1.14.1", 425 | "setprototypeof": "1.1.1", 426 | "statuses": "~1.5.0", 427 | "type-is": "~1.6.18", 428 | "utils-merge": "1.0.1", 429 | "vary": "~1.1.2" 430 | }, 431 | "dependencies": { 432 | "debug": { 433 | "version": "2.6.9", 434 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", 435 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", 436 | "requires": { 437 | "ms": "2.0.0" 438 | } 439 | } 440 | } 441 | }, 442 | "finalhandler": { 443 | "version": "1.1.2", 444 | "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.1.2.tgz", 445 | "integrity": "sha512-aAWcW57uxVNrQZqFXjITpW3sIUQmHGG3qSb9mUah9MgMC4NeWhNOlNjXEYq3HjRAvL6arUviZGGJsBg6z0zsWA==", 446 | "requires": { 447 | "debug": "2.6.9", 448 | "encodeurl": "~1.0.2", 449 | "escape-html": "~1.0.3", 450 | "on-finished": "~2.3.0", 451 | "parseurl": "~1.3.3", 452 | "statuses": "~1.5.0", 453 | "unpipe": "~1.0.0" 454 | }, 455 | "dependencies": { 456 | "debug": { 457 | "version": "2.6.9", 458 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", 459 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", 460 | "requires": { 461 | "ms": "2.0.0" 462 | } 463 | } 464 | } 465 | }, 466 | "foreachasync": { 467 | "version": "3.0.0", 468 | "resolved": "https://registry.npmjs.org/foreachasync/-/foreachasync-3.0.0.tgz", 469 | "integrity": "sha1-VQKYfchxS+M5IJfzLgBxyd7gfPY=" 470 | }, 471 | "form-data": { 472 | "version": "3.0.0", 473 | "resolved": "https://registry.npmjs.org/form-data/-/form-data-3.0.0.tgz", 474 | "integrity": "sha512-CKMFDglpbMi6PyN+brwB9Q/GOw0eAnsrEZDgcsH5Krhz5Od/haKHAX0NmQfha2zPPz0JpWzA7GJHGSnvCRLWsg==", 475 | "requires": { 476 | "asynckit": "^0.4.0", 477 | "combined-stream": "^1.0.8", 478 | "mime-types": "^2.1.12" 479 | } 480 | }, 481 | "forwarded": { 482 | "version": "0.1.2", 483 | "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.1.2.tgz", 484 | "integrity": "sha1-mMI9qxF1ZXuMBXPozszZGw/xjIQ=" 485 | }, 486 | "fresh": { 487 | "version": "0.5.2", 488 | "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz", 489 | "integrity": "sha1-PYyt2Q2XZWn6g1qx+OSyOhBWBac=" 490 | }, 491 | "from2": { 492 | "version": "2.3.0", 493 | "resolved": "https://registry.npmjs.org/from2/-/from2-2.3.0.tgz", 494 | "integrity": "sha1-i/tVAr3kpNNs/e6gB/zKIdfjgq8=", 495 | "requires": { 496 | "inherits": "^2.0.1", 497 | "readable-stream": "^2.0.0" 498 | } 499 | }, 500 | "handlebars": { 501 | "version": "4.7.6", 502 | "resolved": "https://registry.npmjs.org/handlebars/-/handlebars-4.7.6.tgz", 503 | "integrity": "sha512-1f2BACcBfiwAfStCKZNrUCgqNZkGsAT7UM3kkYtXuLo0KnaVfjKOyf7PRzB6++aK9STyT1Pd2ZCPe3EGOXleXA==", 504 | "requires": { 505 | "minimist": "^1.2.5", 506 | "neo-async": "^2.6.0", 507 | "source-map": "^0.6.1", 508 | "uglify-js": "^3.1.4", 509 | "wordwrap": "^1.0.0" 510 | } 511 | }, 512 | "hbs": { 513 | "version": "4.1.1", 514 | "resolved": "https://registry.npmjs.org/hbs/-/hbs-4.1.1.tgz", 515 | "integrity": "sha512-6QsbB4RwbpL4cb4DNyjEEPF+suwp+3yZqFVlhILEn92ScC0U4cDCR+FDX53jkfKJPhutcqhAvs+rOLZw5sQrDA==", 516 | "requires": { 517 | "handlebars": "4.7.6", 518 | "walk": "2.3.14" 519 | } 520 | }, 521 | "http-errors": { 522 | "version": "1.7.2", 523 | "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.7.2.tgz", 524 | "integrity": "sha512-uUQBt3H/cSIVfch6i1EuPNy/YsRSOUBXTVfZ+yR7Zjez3qjBz6i9+i4zjNaoqcoFVI4lQJ5plg63TvGfRSDCRg==", 525 | "requires": { 526 | "depd": "~1.1.2", 527 | "inherits": "2.0.3", 528 | "setprototypeof": "1.1.1", 529 | "statuses": ">= 1.5.0 < 2", 530 | "toidentifier": "1.0.0" 531 | } 532 | }, 533 | "iconv-lite": { 534 | "version": "0.4.24", 535 | "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", 536 | "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", 537 | "requires": { 538 | "safer-buffer": ">= 2.1.2 < 3" 539 | } 540 | }, 541 | "inherits": { 542 | "version": "2.0.3", 543 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", 544 | "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=" 545 | }, 546 | "into-stream": { 547 | "version": "5.1.1", 548 | "resolved": "https://registry.npmjs.org/into-stream/-/into-stream-5.1.1.tgz", 549 | "integrity": "sha512-krrAJ7McQxGGmvaYbB7Q1mcA+cRwg9Ij2RfWIeVesNBgVDZmzY/Fa4IpZUT3bmdRzMzdf/mzltCG2Dq99IZGBA==", 550 | "requires": { 551 | "from2": "^2.3.0", 552 | "p-is-promise": "^3.0.0" 553 | } 554 | }, 555 | "ipaddr.js": { 556 | "version": "1.9.1", 557 | "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz", 558 | "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==" 559 | }, 560 | "isarray": { 561 | "version": "1.0.0", 562 | "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", 563 | "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE=" 564 | }, 565 | "media-typer": { 566 | "version": "0.3.0", 567 | "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", 568 | "integrity": "sha1-hxDXrwqmJvj/+hzgAWhUUmMlV0g=" 569 | }, 570 | "merge-descriptors": { 571 | "version": "1.0.1", 572 | "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz", 573 | "integrity": "sha1-sAqqVW3YtEVoFQ7J0blT8/kMu2E=" 574 | }, 575 | "methods": { 576 | "version": "1.1.2", 577 | "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz", 578 | "integrity": "sha1-VSmk1nZUE07cxSZmVoNbD4Ua/O4=" 579 | }, 580 | "mime": { 581 | "version": "1.6.0", 582 | "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz", 583 | "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==" 584 | }, 585 | "mime-db": { 586 | "version": "1.44.0", 587 | "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.44.0.tgz", 588 | "integrity": "sha512-/NOTfLrsPBVeH7YtFPgsVWveuL+4SjjYxaQ1xtM1KMFj7HdxlBlxeyNLzhyJVx7r4rZGJAZ/6lkKCitSc/Nmpg==" 589 | }, 590 | "mime-types": { 591 | "version": "2.1.27", 592 | "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.27.tgz", 593 | "integrity": "sha512-JIhqnCasI9yD+SsmkquHBxTSEuZdQX5BuQnS2Vc7puQQQ+8yiP5AY5uWhpdv4YL4VM5c6iliiYWPgJ/nJQLp7w==", 594 | "requires": { 595 | "mime-db": "1.44.0" 596 | } 597 | }, 598 | "minimist": { 599 | "version": "1.2.5", 600 | "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.5.tgz", 601 | "integrity": "sha512-FM9nNUYrRBAELZQT3xeZQ7fmMOBg6nWNmJKTcgsJeaLstP/UODVpGsr5OhXhhXg6f+qtJ8uiZ+PUxkDWcgIXLw==" 602 | }, 603 | "mkdirp": { 604 | "version": "0.5.5", 605 | "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.5.tgz", 606 | "integrity": "sha512-NKmAlESf6jMGym1++R0Ra7wvhV+wFW63FaSOFPwRahvea0gMUcGUhVeAg/0BC0wiv9ih5NYPB1Wn1UEI1/L+xQ==", 607 | "requires": { 608 | "minimist": "^1.2.5" 609 | } 610 | }, 611 | "morgan": { 612 | "version": "1.10.0", 613 | "resolved": "https://registry.npmjs.org/morgan/-/morgan-1.10.0.tgz", 614 | "integrity": "sha512-AbegBVI4sh6El+1gNwvD5YIck7nSA36weD7xvIxG4in80j/UoK8AEGaWnnz8v1GxonMCltmlNs5ZKbGvl9b1XQ==", 615 | "requires": { 616 | "basic-auth": "~2.0.1", 617 | "debug": "2.6.9", 618 | "depd": "~2.0.0", 619 | "on-finished": "~2.3.0", 620 | "on-headers": "~1.0.2" 621 | }, 622 | "dependencies": { 623 | "debug": { 624 | "version": "2.6.9", 625 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", 626 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", 627 | "requires": { 628 | "ms": "2.0.0" 629 | } 630 | }, 631 | "depd": { 632 | "version": "2.0.0", 633 | "resolved": "https://registry.npmjs.org/depd/-/depd-2.0.0.tgz", 634 | "integrity": "sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw==" 635 | } 636 | } 637 | }, 638 | "ms": { 639 | "version": "2.0.0", 640 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", 641 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" 642 | }, 643 | "multer": { 644 | "version": "1.4.2", 645 | "resolved": "https://registry.npmjs.org/multer/-/multer-1.4.2.tgz", 646 | "integrity": "sha512-xY8pX7V+ybyUpbYMxtjM9KAiD9ixtg5/JkeKUTD6xilfDv0vzzOFcCp4Ljb1UU3tSOM3VTZtKo63OmzOrGi3Cg==", 647 | "requires": { 648 | "append-field": "^1.0.0", 649 | "busboy": "^0.2.11", 650 | "concat-stream": "^1.5.2", 651 | "mkdirp": "^0.5.1", 652 | "object-assign": "^4.1.1", 653 | "on-finished": "^2.3.0", 654 | "type-is": "^1.6.4", 655 | "xtend": "^4.0.0" 656 | } 657 | }, 658 | "negotiator": { 659 | "version": "0.6.2", 660 | "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.2.tgz", 661 | "integrity": "sha512-hZXc7K2e+PgeI1eDBe/10Ard4ekbfrrqG8Ep+8Jmf4JID2bNg7NvCPOZN+kfF574pFQI7mum2AUqDidoKqcTOw==" 662 | }, 663 | "neo-async": { 664 | "version": "2.6.1", 665 | "resolved": "https://registry.npmjs.org/neo-async/-/neo-async-2.6.1.tgz", 666 | "integrity": "sha512-iyam8fBuCUpWeKPGpaNMetEocMt364qkCsfL9JuhjXX6dRnguRVOfk2GZaDpPjcOKiiXCPINZC1GczQ7iTq3Zw==" 667 | }, 668 | "node-fetch": { 669 | "version": "2.6.0", 670 | "resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-2.6.0.tgz", 671 | "integrity": "sha512-8dG4H5ujfvFiqDmVu9fQ5bOHUC15JMjMY/Zumv26oOvvVJjM67KF8koCWIabKQ1GJIa9r2mMZscBq/TbdOcmNA==" 672 | }, 673 | "object-assign": { 674 | "version": "4.1.1", 675 | "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", 676 | "integrity": "sha1-IQmtx5ZYh8/AXLvUQsrIv7s2CGM=" 677 | }, 678 | "on-finished": { 679 | "version": "2.3.0", 680 | "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.3.0.tgz", 681 | "integrity": "sha1-IPEzZIGwg811M3mSoWlxqi2QaUc=", 682 | "requires": { 683 | "ee-first": "1.1.1" 684 | } 685 | }, 686 | "on-headers": { 687 | "version": "1.0.2", 688 | "resolved": "https://registry.npmjs.org/on-headers/-/on-headers-1.0.2.tgz", 689 | "integrity": "sha512-pZAE+FJLoyITytdqK0U5s+FIpjN0JP3OzFi/u8Rx+EV5/W+JTWGXG8xFzevE7AjBfDqHv/8vL8qQsIhHnqRkrA==" 690 | }, 691 | "p-is-promise": { 692 | "version": "3.0.0", 693 | "resolved": "https://registry.npmjs.org/p-is-promise/-/p-is-promise-3.0.0.tgz", 694 | "integrity": "sha512-Wo8VsW4IRQSKVXsJCn7TomUaVtyfjVDn3nUP7kE967BQk0CwFpdbZs0X0uk5sW9mkBa9eNM7hCMaG93WUAwxYQ==" 695 | }, 696 | "parseurl": { 697 | "version": "1.3.3", 698 | "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz", 699 | "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==" 700 | }, 701 | "path-to-regexp": { 702 | "version": "0.1.7", 703 | "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz", 704 | "integrity": "sha1-32BBeABfUi8V60SQ5yR6G/qmf4w=" 705 | }, 706 | "process": { 707 | "version": "0.11.10", 708 | "resolved": "https://registry.npmjs.org/process/-/process-0.11.10.tgz", 709 | "integrity": "sha1-czIwDoQBYb2j5podHZGn1LwW8YI=" 710 | }, 711 | "process-nextick-args": { 712 | "version": "2.0.1", 713 | "resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-2.0.1.tgz", 714 | "integrity": "sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag==" 715 | }, 716 | "proxy-addr": { 717 | "version": "2.0.6", 718 | "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.6.tgz", 719 | "integrity": "sha512-dh/frvCBVmSsDYzw6n926jv974gddhkFPfiN8hPOi30Wax25QZyZEGveluCgliBnqmuM+UJmBErbAUFIoDbjOw==", 720 | "requires": { 721 | "forwarded": "~0.1.2", 722 | "ipaddr.js": "1.9.1" 723 | } 724 | }, 725 | "psl": { 726 | "version": "1.8.0", 727 | "resolved": "https://registry.npmjs.org/psl/-/psl-1.8.0.tgz", 728 | "integrity": "sha512-RIdOzyoavK+hA18OGGWDqUTsCLhtA7IcZ/6NCs4fFJaHBDab+pDDmDIByWFRQJq2Cd7r1OoQxBGKOaztq+hjIQ==" 729 | }, 730 | "punycode": { 731 | "version": "2.1.1", 732 | "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.1.1.tgz", 733 | "integrity": "sha512-XRsRjdf+j5ml+y/6GKHPZbrF/8p2Yga0JPtdqTIY2Xe5ohJPD9saDJJLPvp9+NSBprVvevdXZybnj2cv8OEd0A==" 734 | }, 735 | "qs": { 736 | "version": "6.7.0", 737 | "resolved": "https://registry.npmjs.org/qs/-/qs-6.7.0.tgz", 738 | "integrity": "sha512-VCdBRNFTX1fyE7Nb6FYoURo/SPe62QCaAyzJvUjwRaIsc+NePBEniHlvxFmmX56+HZphIGtV0XeCirBtpDrTyQ==" 739 | }, 740 | "range-parser": { 741 | "version": "1.2.1", 742 | "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz", 743 | "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==" 744 | }, 745 | "raw-body": { 746 | "version": "2.4.0", 747 | "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.4.0.tgz", 748 | "integrity": "sha512-4Oz8DUIwdvoa5qMJelxipzi/iJIi40O5cGV1wNYp5hvZP8ZN0T+jiNkL0QepXs+EsQ9XJ8ipEDoiH70ySUJP3Q==", 749 | "requires": { 750 | "bytes": "3.1.0", 751 | "http-errors": "1.7.2", 752 | "iconv-lite": "0.4.24", 753 | "unpipe": "1.0.0" 754 | } 755 | }, 756 | "readable-stream": { 757 | "version": "2.3.7", 758 | "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.7.tgz", 759 | "integrity": "sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw==", 760 | "requires": { 761 | "core-util-is": "~1.0.0", 762 | "inherits": "~2.0.3", 763 | "isarray": "~1.0.0", 764 | "process-nextick-args": "~2.0.0", 765 | "safe-buffer": "~5.1.1", 766 | "string_decoder": "~1.1.1", 767 | "util-deprecate": "~1.0.1" 768 | } 769 | }, 770 | "safe-buffer": { 771 | "version": "5.1.2", 772 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", 773 | "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==" 774 | }, 775 | "safer-buffer": { 776 | "version": "2.1.2", 777 | "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", 778 | "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==" 779 | }, 780 | "sax": { 781 | "version": "1.2.4", 782 | "resolved": "https://registry.npmjs.org/sax/-/sax-1.2.4.tgz", 783 | "integrity": "sha512-NqVDv9TpANUjFm0N8uM5GxL36UgKi9/atZw+x7YFnQ8ckwFGKrl4xX4yWtrey3UJm5nP1kUbnYgLopqWNSRhWw==" 784 | }, 785 | "send": { 786 | "version": "0.17.1", 787 | "resolved": "https://registry.npmjs.org/send/-/send-0.17.1.tgz", 788 | "integrity": "sha512-BsVKsiGcQMFwT8UxypobUKyv7irCNRHk1T0G680vk88yf6LBByGcZJOTJCrTP2xVN6yI+XjPJcNuE3V4fT9sAg==", 789 | "requires": { 790 | "debug": "2.6.9", 791 | "depd": "~1.1.2", 792 | "destroy": "~1.0.4", 793 | "encodeurl": "~1.0.2", 794 | "escape-html": "~1.0.3", 795 | "etag": "~1.8.1", 796 | "fresh": "0.5.2", 797 | "http-errors": "~1.7.2", 798 | "mime": "1.6.0", 799 | "ms": "2.1.1", 800 | "on-finished": "~2.3.0", 801 | "range-parser": "~1.2.1", 802 | "statuses": "~1.5.0" 803 | }, 804 | "dependencies": { 805 | "debug": { 806 | "version": "2.6.9", 807 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", 808 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", 809 | "requires": { 810 | "ms": "2.0.0" 811 | }, 812 | "dependencies": { 813 | "ms": { 814 | "version": "2.0.0", 815 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", 816 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" 817 | } 818 | } 819 | }, 820 | "ms": { 821 | "version": "2.1.1", 822 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.1.tgz", 823 | "integrity": "sha512-tgp+dl5cGk28utYktBsrFqA7HKgrhgPsg6Z/EfhWI4gl1Hwq8B/GmY/0oXZ6nF8hDVesS/FpnYaD/kOWhYQvyg==" 824 | } 825 | } 826 | }, 827 | "serve-favicon": { 828 | "version": "2.5.0", 829 | "resolved": "https://registry.npmjs.org/serve-favicon/-/serve-favicon-2.5.0.tgz", 830 | "integrity": "sha1-k10kDN/g9YBTB/3+ln2IlCosvPA=", 831 | "requires": { 832 | "etag": "~1.8.1", 833 | "fresh": "0.5.2", 834 | "ms": "2.1.1", 835 | "parseurl": "~1.3.2", 836 | "safe-buffer": "5.1.1" 837 | }, 838 | "dependencies": { 839 | "ms": { 840 | "version": "2.1.1", 841 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.1.tgz", 842 | "integrity": "sha512-tgp+dl5cGk28utYktBsrFqA7HKgrhgPsg6Z/EfhWI4gl1Hwq8B/GmY/0oXZ6nF8hDVesS/FpnYaD/kOWhYQvyg==" 843 | }, 844 | "safe-buffer": { 845 | "version": "5.1.1", 846 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.1.tgz", 847 | "integrity": "sha512-kKvNJn6Mm93gAczWVJg7wH+wGYWNrDHdWvpUmHyEsgCtIwwo3bqPtV4tR5tuPaUhTOo/kvhVwd8XwwOllGYkbg==" 848 | } 849 | } 850 | }, 851 | "serve-static": { 852 | "version": "1.14.1", 853 | "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.14.1.tgz", 854 | "integrity": "sha512-JMrvUwE54emCYWlTI+hGrGv5I8dEwmco/00EvkzIIsR7MqrHonbD9pO2MOfFnpFntl7ecpZs+3mW+XbQZu9QCg==", 855 | "requires": { 856 | "encodeurl": "~1.0.2", 857 | "escape-html": "~1.0.3", 858 | "parseurl": "~1.3.3", 859 | "send": "0.17.1" 860 | } 861 | }, 862 | "setprototypeof": { 863 | "version": "1.1.1", 864 | "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.1.tgz", 865 | "integrity": "sha512-JvdAWfbXeIGaZ9cILp38HntZSFSo3mWg6xGcJJsd+d4aRMOqauag1C63dJfDw7OaMYwEbHMOxEZ1lqVRYP2OAw==" 866 | }, 867 | "source-map": { 868 | "version": "0.6.1", 869 | "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", 870 | "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==" 871 | }, 872 | "statuses": { 873 | "version": "1.5.0", 874 | "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.5.0.tgz", 875 | "integrity": "sha1-Fhx9rBd2Wf2YEfQ3cfqZOBR4Yow=" 876 | }, 877 | "streamsearch": { 878 | "version": "0.1.2", 879 | "resolved": "https://registry.npmjs.org/streamsearch/-/streamsearch-0.1.2.tgz", 880 | "integrity": "sha1-gIudDlb8Jz2Am6VzOOkpkZoanxo=" 881 | }, 882 | "string_decoder": { 883 | "version": "1.1.1", 884 | "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz", 885 | "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==", 886 | "requires": { 887 | "safe-buffer": "~5.1.0" 888 | } 889 | }, 890 | "toidentifier": { 891 | "version": "1.0.0", 892 | "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.0.tgz", 893 | "integrity": "sha512-yaOH/Pk/VEhBWWTlhI+qXxDFXlejDGcQipMlyxda9nthulaxLZUNcUqFxokp0vcYnvteJln5FNQDRrxj3YcbVw==" 894 | }, 895 | "tough-cookie": { 896 | "version": "4.0.0", 897 | "resolved": "https://registry.npmjs.org/tough-cookie/-/tough-cookie-4.0.0.tgz", 898 | "integrity": "sha512-tHdtEpQCMrc1YLrMaqXXcj6AxhYi/xgit6mZu1+EDWUn+qhUf8wMQoFIy9NXuq23zAwtcB0t/MjACGR18pcRbg==", 899 | "requires": { 900 | "psl": "^1.1.33", 901 | "punycode": "^2.1.1", 902 | "universalify": "^0.1.2" 903 | } 904 | }, 905 | "tslib": { 906 | "version": "1.13.0", 907 | "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.13.0.tgz", 908 | "integrity": "sha512-i/6DQjL8Xf3be4K/E6Wgpekn5Qasl1usyw++dAA35Ue5orEn65VIxOA+YvNNl9HV3qv70T7CNwjODHZrLwvd1Q==" 909 | }, 910 | "tunnel": { 911 | "version": "0.0.6", 912 | "resolved": "https://registry.npmjs.org/tunnel/-/tunnel-0.0.6.tgz", 913 | "integrity": "sha512-1h/Lnq9yajKY2PEbBadPXj3VxsDDu844OnaAo52UVmIzIvwwtBPIuNvkjuzBlTWpfJyUbG3ez0KSBibQkj4ojg==" 914 | }, 915 | "type-is": { 916 | "version": "1.6.18", 917 | "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz", 918 | "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==", 919 | "requires": { 920 | "media-typer": "0.3.0", 921 | "mime-types": "~2.1.24" 922 | } 923 | }, 924 | "typedarray": { 925 | "version": "0.0.6", 926 | "resolved": "https://registry.npmjs.org/typedarray/-/typedarray-0.0.6.tgz", 927 | "integrity": "sha1-hnrHTjhkGHsdPUfZlqeOxciDB3c=" 928 | }, 929 | "uglify-js": { 930 | "version": "3.10.0", 931 | "resolved": "https://registry.npmjs.org/uglify-js/-/uglify-js-3.10.0.tgz", 932 | "integrity": "sha512-Esj5HG5WAyrLIdYU74Z3JdG2PxdIusvj6IWHMtlyESxc7kcDz7zYlYjpnSokn1UbpV0d/QX9fan7gkCNd/9BQA==", 933 | "optional": true 934 | }, 935 | "universalify": { 936 | "version": "0.1.2", 937 | "resolved": "https://registry.npmjs.org/universalify/-/universalify-0.1.2.tgz", 938 | "integrity": "sha512-rBJeI5CXAlmy1pV+617WB9J63U6XcazHHF2f2dbJix4XzpUF0RS3Zbj0FGIOCAva5P/d/GBOYaACQ1w+0azUkg==" 939 | }, 940 | "unpipe": { 941 | "version": "1.0.0", 942 | "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", 943 | "integrity": "sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw=" 944 | }, 945 | "util-deprecate": { 946 | "version": "1.0.2", 947 | "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", 948 | "integrity": "sha1-RQ1Nyfpw3nMnYvvS1KKJgUGaDM8=" 949 | }, 950 | "utils-merge": { 951 | "version": "1.0.1", 952 | "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz", 953 | "integrity": "sha1-n5VxD1CiZ5R7LMwSR0HBAoQn5xM=" 954 | }, 955 | "uuid": { 956 | "version": "8.2.0", 957 | "resolved": "https://registry.npmjs.org/uuid/-/uuid-8.2.0.tgz", 958 | "integrity": "sha512-CYpGiFTUrmI6OBMkAdjSDM0k5h8SkkiTP4WAjQgDgNB1S3Ou9VBEvr6q0Kv2H1mMk7IWfxYGpMH5sd5AvcIV2Q==" 959 | }, 960 | "vary": { 961 | "version": "1.1.2", 962 | "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz", 963 | "integrity": "sha1-IpnwLG3tMNSllhsLn3RSShj2NPw=" 964 | }, 965 | "walk": { 966 | "version": "2.3.14", 967 | "resolved": "https://registry.npmjs.org/walk/-/walk-2.3.14.tgz", 968 | "integrity": "sha512-5skcWAUmySj6hkBdH6B6+3ddMjVQYH5Qy9QGbPmN8kVmLteXk+yVXg+yfk1nbX30EYakahLrr8iPcCxJQSCBeg==", 969 | "requires": { 970 | "foreachasync": "^3.0.0" 971 | } 972 | }, 973 | "wordwrap": { 974 | "version": "1.0.0", 975 | "resolved": "https://registry.npmjs.org/wordwrap/-/wordwrap-1.0.0.tgz", 976 | "integrity": "sha1-J1hIEIkUVqQXHI0CJkQa3pDLyus=" 977 | }, 978 | "xml2js": { 979 | "version": "0.4.23", 980 | "resolved": "https://registry.npmjs.org/xml2js/-/xml2js-0.4.23.tgz", 981 | "integrity": "sha512-ySPiMjM0+pLDftHgXY4By0uswI3SPKLDw/i3UXbnO8M/p28zqexCUoPmQFrYD+/1BzhGJSs2i1ERWKJAtiLrug==", 982 | "requires": { 983 | "sax": ">=0.6.0", 984 | "xmlbuilder": "~11.0.0" 985 | } 986 | }, 987 | "xmlbuilder": { 988 | "version": "11.0.1", 989 | "resolved": "https://registry.npmjs.org/xmlbuilder/-/xmlbuilder-11.0.1.tgz", 990 | "integrity": "sha512-fDlsI/kFEx7gLvbecc0/ohLG50fugQp8ryHzMTuW9vSa1GJ0XYWKnhsUx7oie3G98+r56aTQIUB4kht42R3JvA==" 991 | }, 992 | "xtend": { 993 | "version": "4.0.2", 994 | "resolved": "https://registry.npmjs.org/xtend/-/xtend-4.0.2.tgz", 995 | "integrity": "sha512-LKYU1iAXJXUgAXn9URjiu+MWhyUXHsvfp7mcuYm9dSUKK0/CjtrUwFAxD82/mCWbtLsGjFIad0wIsod4zrTAEQ==" 996 | } 997 | } 998 | } 999 | -------------------------------------------------------------------------------- /v12/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "node-blobs-image-resize", 3 | "version": "0.0.12", 4 | "private": true, 5 | "engines": { 6 | "node": ">=10.14.1", 7 | "npm": ">=6.13.4" 8 | }, 9 | "scripts": { 10 | "start": "node ./bin/www" 11 | }, 12 | "dependencies": { 13 | "@azure/storage-blob": "^12.1.2", 14 | "body-parser": "^1.18.3", 15 | "cookie-parser": "~1.4.3", 16 | "debug": "^4.1.1", 17 | "dotenv": "^8.2.0", 18 | "express": "^4.17.1", 19 | "hbs": "^4.1.1", 20 | "into-stream": "^5.1.1", 21 | "morgan": "^1.10.0", 22 | "multer": "^1.4.1", 23 | "serve-favicon": "~2.5.0" 24 | } 25 | } 26 | -------------------------------------------------------------------------------- /v12/public/favicon.ico: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Azure-Samples/azure-sdk-for-js-storage-blob-stream-nodejs/50eb069440e805489c5d8c7232b4735f130322c2/v12/public/favicon.ico -------------------------------------------------------------------------------- /v12/public/stylesheets/style.css: -------------------------------------------------------------------------------- 1 | body { 2 | padding: 50px; 3 | font: 14px "Lucida Grande", Helvetica, Arial, sans-serif; 4 | } 5 | 6 | a { 7 | color: #00B7FF; 8 | } 9 | 10 | .upload-container { 11 | border: solid 1px #ccc; 12 | padding: 10px; 13 | max-width: fit-content; 14 | margin-bottom: 10px 15 | } -------------------------------------------------------------------------------- /v12/routes/index.js: -------------------------------------------------------------------------------- 1 | if (process.env.NODE_ENV !== 'production') { 2 | require('dotenv').config(); 3 | } 4 | 5 | const { 6 | BlobServiceClient, 7 | StorageSharedKeyCredential, 8 | newPipeline 9 | } = require('@azure/storage-blob'); 10 | 11 | const express = require('express'); 12 | const router = express.Router(); 13 | const containerName1 = 'thumbnails'; 14 | const multer = require('multer'); 15 | const inMemoryStorage = multer.memoryStorage(); 16 | const uploadStrategy = multer({ storage: inMemoryStorage }).single('image'); 17 | const getStream = require('into-stream'); 18 | const containerName2 = 'images'; 19 | const ONE_MEGABYTE = 1024 * 1024; 20 | const uploadOptions = { bufferSize: 4 * ONE_MEGABYTE, maxBuffers: 20 }; 21 | const ONE_MINUTE = 60 * 1000; 22 | 23 | const sharedKeyCredential = new StorageSharedKeyCredential( 24 | process.env.AZURE_STORAGE_ACCOUNT_NAME, 25 | process.env.AZURE_STORAGE_ACCOUNT_ACCESS_KEY); 26 | const pipeline = newPipeline(sharedKeyCredential); 27 | 28 | const blobServiceClient = new BlobServiceClient( 29 | `https://${process.env.AZURE_STORAGE_ACCOUNT_NAME}.blob.core.windows.net`, 30 | pipeline 31 | ); 32 | 33 | const getBlobName = originalName => { 34 | // Use a random number to generate a unique file name, 35 | // removing "0." from the start of the string. 36 | const identifier = Math.random().toString().replace(/0\./, ''); 37 | return `${identifier}-${originalName}`; 38 | }; 39 | 40 | router.get('/', async (req, res, next) => { 41 | 42 | let viewData; 43 | 44 | try { 45 | const containerClient = blobServiceClient.getContainerClient(containerName1); 46 | const listBlobsResponse = await containerClient.listBlobFlatSegment(); 47 | 48 | for await (const blob of listBlobsResponse.segment.blobItems) { 49 | console.log(`Blob: ${blob.name}`); 50 | } 51 | 52 | viewData = { 53 | title: 'Home', 54 | viewName: 'index', 55 | accountName: process.env.AZURE_STORAGE_ACCOUNT_NAME, 56 | containerName: containerName1 57 | }; 58 | 59 | if (listBlobsResponse.segment.blobItems.length) { 60 | viewData.thumbnails = listBlobsResponse.segment.blobItems; 61 | } 62 | } catch (err) { 63 | viewData = { 64 | title: 'Error', 65 | viewName: 'error', 66 | message: 'There was an error contacting the blob storage container.', 67 | error: err 68 | }; 69 | res.status(500); 70 | } finally { 71 | res.render(viewData.viewName, viewData); 72 | } 73 | }); 74 | 75 | router.post('/', uploadStrategy, async (req, res) => { 76 | const blobName = getBlobName(req.file.originalname); 77 | const stream = getStream(req.file.buffer); 78 | const containerClient = blobServiceClient.getContainerClient(containerName2);; 79 | const blockBlobClient = containerClient.getBlockBlobClient(blobName); 80 | 81 | try { 82 | await blockBlobClient.uploadStream(stream, 83 | uploadOptions.bufferSize, uploadOptions.maxBuffers, 84 | { blobHTTPHeaders: { blobContentType: "image/jpeg" } }); 85 | res.render('success', { message: 'File uploaded to Azure Blob storage.' }); 86 | } catch (err) { 87 | res.render('error', { message: err.message }); 88 | } 89 | }); 90 | 91 | module.exports = router; -------------------------------------------------------------------------------- /v12/views/error.hbs: -------------------------------------------------------------------------------- 1 |

{{message}}

2 |

{{error.status}}

3 |
{{error.stack}}
4 | -------------------------------------------------------------------------------- /v12/views/index.hbs: -------------------------------------------------------------------------------- 1 |

Image Resizer

2 | 3 |

Upload Photos

4 |
5 |

Select an image to upload:

6 |
7 |
8 |
9 | 10 |

{{message}}

11 | 12 |

Generated Thumbnails

13 | {{#each this.thumbnails}} 14 | 15 | {{/each}} 16 | 17 |
18 |

This app has no official privacy policy. Your data will be uploaded to a service in order to produce a thumbnail. Your images will be public once you upload them and there is no automated way to remove them.

19 | 20 | -------------------------------------------------------------------------------- /v12/views/layout.hbs: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | {{title}} 5 | 6 | 7 | 8 | {{{body}}} 9 | 10 | 11 | -------------------------------------------------------------------------------- /v12/views/success.hbs: -------------------------------------------------------------------------------- 1 |

Success!

2 | 3 |

{{message}}

4 | 5 | Return home to view thumbnails --------------------------------------------------------------------------------