2.0.1 • Published 2 years ago

@psyora/psy-s3-uploader v2.0.1

Weekly downloads
-
License
-
Repository
-
Last release
2 years ago

Introduction

Psy-S3-Uploader library is created to securely upload images and files to S3 bucket from and Angular application. This library was generated with Angular CLI version 10.0.14.

This library requries a backend that generates pre-signed url for POST request using createPreSignedPost( ) method. So, bucket inforamtion, secret key and access key are provided in the backend.

Based on the response provided by the backend, the library will directly upload the file to S3 bucket.

Installation Instructions

Step 1: Install the library from npm

To install, run the following command in your angular project

npm i @psyora/psy-s3-uploader

Step 2: Import the library to Angular

To use the library in your application, import it to your angular application in your app or shared module.

import { PsyS3UploaderModule } from '@psyora/psy-s3-uploader';

@NgModule({
    imports: [
        ...,
        PsyS3UploaderModule,
        ...
    ]
})

export class AppModule { }

Step 3: Use the library in the application using the selector

selector: psy-s3-uploader

<psy-s3-uploader></psy-s3-uploader>

Step 4: Verify the image upload dropbox

image

Usage instructions

User Interaface

You can upload a file inside the application in two ways. 1. By dragging and dropping an image 1. By clicking on 'Browse File' and selecting a file to upload

By default, only jpeg, jpg or png images are allowed to be uploaded. However, other types of can be uploaded by specifiying their extensions. They will be explored later.

On successful load, a preview will be shown if an image is selected.

image

You can click on the x button to remove the file from upload.

Uploading a File

To upload the selected file to S3 bucket, you need to call the upload method from your application. This give you control to time your upload programatically.

In your parent component (ts)

import { Component, ViewChild } from '@angular/core';
import { PsyS3UploaderComponent } from 'psy-s3-uploader';

@Component({
    selector: 'app-root',
    templateUrl: './app.component.html',
    styleUrls: ['./app.component.scss']
})

export class AppComponent {
  
    @ViewChild(PsyS3UploaderComponent) psyS3Uploader: PsyS3UploaderComponent;

    public upload(): void {
        this.psyS3Uploader.upload();
    };
}

The library supports uploading files to S3 bucket using pre-signed url's. So you need to specify the API endpoint of a function that is able to generate a pre-signed url.

So, you need to construct a params object and provide the upload endpoint. You can import the IPsyS3UploaderConfig from the library and declare it as follows:

import { Component, ViewChild } from '@angular/core';
import { IPsyS3UploaderConfig, PsyS3UploaderComponent } from 'psy-s3-uploader';

@Component({
    selector: 'app-root',
    templateUrl: './app.component.html',
    styleUrls: ['./app.component.scss']
})

export class AppComponent {
  
    @ViewChild(PsyS3UploaderComponent) psyS3Uploader: PsyS3UploaderComponent;

    public params: IPsyS3UploaderConfig = {
        uploadApi: 'upload API'
    };

    public upload(): void {
        this.psyS3Uploader.upload();
    };
}

Next when the upload is in progress, two events are emitted to notify the status of upload. 1. response 1. error

Create two functions to handle these events.

import { Component, ViewChild } from '@angular/core';
import { IPsyS3UploaderConfig, PsyS3UploaderComponent } from 'psy-s3-uploader';

@Component({
    selector: 'app-root',
    templateUrl: './app.component.html',
    styleUrls: ['./app.component.scss']
})

export class AppComponent {
  
    @ViewChild(PsyS3UploaderComponent) psyS3Uploader: PsyS3UploaderComponent;

    public params: IPsyS3UploaderConfig = {
        uploadApi: 'upload API'
    };

    public upload(): void {
        this.psyS3Uploader.upload();
    };

    public handleResponse(res): void {
        console.log(res)
    };

    public handleError(err): void {
        console.log(err)
    };
}

Finally, in your parent component (html) update the selector with input and output operators

<psy-s3-uploader
    [params]="params"
    (response)="handleResponse($event)"
    (error)="handleError($event)"
></psy-s3-uploader>
<button (click)="upload()">Upload</button>

Now when you can click on Upload button to initiate the upload to S3.

  1. The library will reach out to the upload api end-point and get the pre-signed url, signature info, policy info, upload bucket info etc.
  2. It will then create a form object and will upload to the S3 bucket that was provided by upload api with all the details.
  3. Finally it will notify the parent component of its status

And the results will be logged by the app component.

image

Finally, verify the uploaded image in S3 bucket.

image

Uploading multiple Files

To upload multiple files, specify the multipleFiles param and set it to true

import { Component, ViewChild } from '@angular/core';
import { IPsyS3UploaderConfig, PsyS3UploaderComponent } from 'psy-s3-uploader';

@Component({
    selector: 'app-root',
    templateUrl: './app.component.html',
    styleUrls: ['./app.component.scss']
})

export class AppComponent {
  
    @ViewChild(PsyS3UploaderComponent) psyS3Uploader: PsyS3UploaderComponent;

    public params: IPsyS3UploaderConfig = {
        uploadApi: 'upload API',
        multipleFiles: true
    };

    public upload(): void {
        this.psyS3Uploader.upload();
    };

    public handleResponse(res): void {
        console.log(res)
    };

    public handleError(err): void {
        console.log(err)
    };
}

Now, you can select and upload multiple files. When you can click on Upload button to initiate the upload to S3. And the results will be logged by the app component.

image

  1. The library will iterate through all the files and reach out to the upload api end-point and get the pre-signed url, signature info, policy info, upload bucket info etc.
  2. It will then create a form object for every object and will upload to the S3 bucket that was provided by upload api with all the details.
  3. Finally, once all the files are uploaded, it will notify the parent component of its status. And the results will be logged by the app component.

image

Note: Due to asynchronous nature of Javascript, the upload process will be almost simultaneous rather than sequential.

Handling a Failed Upload

On upload failure, you can

  1. retry the process by clicking on the Retry button or by calling the upload method again.
  2. remove the failied upload by clicking on 'x'

image

In case of multiple files, hitting the retry button will reinitate the upload process for only the failed files.

image

Note: A response will not be triggered by the library until all files are successfully uploaded.

In such cases, to get the upload response, the user needs to remove the failed files manually and call the upload method again.

Specifying Maximum files to upload

In cases of multiple file uploads, the library allows you to upload upto 10 files by default. You can adjust the number of files being uploaded by specifying 'maxFiles' parameter.

In case the number of files selected is greater than the maximum files specified, the library will load valid files until the max file limit is reached.

You will be notified with an alert message if you try to upload more than maximum files.

image

If/when the maximum files are uploded, the drop-area disappears.

image

To upload more files, you can remove currently selected files to free up a slot for upload.

Supported Icons

If the uploaded file is not an image. Then the library displays an icon preview.

Currently icon preview is supported for 1. PDF 1. Word 1. Excel

image

Preview Mode

Preview mode is used to render 1. 'preview of an image' by passing image URL 1. 'icon for a document' by specifying the fileType

You can pass an array of URLs to the libray using the previewArray property. Refer to the interface IFilePreview for more information.

public params: IPsyS3UploaderConfig = {
    uploadApi: "value",
    previewArray: [{
        "url": "https://images.unsplash.com/photo-1622522486900-37e2c44647cb?ixlib=rb-1.2.1&ixid=MnwxMjA3fDB8MHxwaG90by1wYWdlfHx8fGVufDB8fHx8&auto=format&fit=crop&w=1868&q=80",
        "fileType": "image/jpeg"
    }, {
        "url": "https://images.unsplash.com/photo-1624024946645-d02c208b1c4d?ixid=MnwxMjA3fDB8MHxwaG90by1wYWdlfHx8fGVufDB8fHx8&ixlib=rb-1.2.1&auto=format&fit=crop&w=1951&q=80",
        "fileType": "image/jpeg"
    }]
};

image

These files are readonly and upload activities cannot be performed on them. If you try to upload, you will get a notification that satys 'No files found'.

image

Deleting a File

A file/image can be deleted from preview mode by specifying 'canDelete' as 'true'. When specified, a delete 'icon' appears on the top right corner of the preview.

public params: IPsyS3UploaderConfig = {
    uploadApi: "value",
    previewArray: [{
        url: "https://images.unsplash.com/photo-1622522486900-37e2c44647cb?ixlib=rb-1.2.1&ixid=MnwxMjA3fDB8MHxwaG90by1wYWdlfHx8fGVufDB8fHx8&auto=format&fit=crop&w=1868&q=80",
        fileType: "image/jpeg",
        canDelete: true
    }, {
        url: "https://images.unsplash.com/photo-1624024946645-d02c208b1c4d?ixid=MnwxMjA3fDB8MHxwaG90by1wYWdlfHx8fGVufDB8fHx8&ixlib=rb-1.2.1&auto=format&fit=crop&w=1951&q=80",
        fileType: "image/jpeg"
    }]
};

image

However, just by clicking on the "🗑" icon, the delete function will not be executed. Since the library is created to delete S3 items from S3 bucket, the key of the object is a required parameter in performing the operation.

They key of the object required to delete the file can be passed to the library by specifying it in the request. It can be passed in two ways: 1. Inside the Delete URL - deleteUrl 1. Inside Query String Parameters - queryStringParameters

By default, the library uses same url that is used for uploading. DELETE method is used and a call is made. However, in case the delete URL is different from the upload URL, it can be specified per object by using deleteUrl params.

Since there are multiple files in previewMode, every file has their own individual delete URLs. They can be passed individally as follows:

In case the delete of S3 Object is done through lambda function, the lambda API can be specified in deleteUrl.

public params: IPsyS3UploaderConfig = {
    uploadApi: "value",
    previewArray: [{
        url: "https://s3.amazonaws.com/uploads.uppy-upload-svc.psyora/obj_1.jpg",
        fileType: "image/jpeg",
        deleteUrl: "https://pbi55zcp89.execute-api.us-east-1.amazonaws.com/poc/upload?key=obj_1",
        canDelete: true
    }, {
        url: "https://s3.amazonaws.com/uploads.uppy-upload-svc.psyora/obj_2.jpg",
        fileType: "image/jpeg",
        deleteUrl: "https://pbi55zcp89.execute-api.us-east-1.amazonaws.com/poc/upload?key=obj_2",
        canDelete: true
    }]
};

In case the key is a queryStringParameter, it can be passed as an Object.

public params: IPsyS3UploaderConfig = {
    uploadApi: "value",
    previewArray: [{
        url: "https://s3.amazonaws.com/uploads.uppy-upload-svc.psyora/obj_1.jpg",
        fileType: "image/jpeg",
        deleteUrl: "https://pbi55zcp89.execute-api.us-east-1.amazonaws.com/poc/upload",
        canDelete: true,
        queryStringParameters: {
            _key: "obj_1"
        },
    }, {
        url: "https://s3.amazonaws.com/uploads.uppy-upload-svc.psyora/obj_2.jpg",
        fileType: "image/jpeg",
        deleteUrl: "https://pbi55zcp89.execute-api.us-east-1.amazonaws.com/poc/upload",
        canDelete: true,
        queryStringParameters: {
            _key: "obj_2"
        }
    }]
};

Objects before deleting

The delete process for a file can be initiated by clicking on the '🗑' icon and confirming the prompt.

Objects confirmation prompt

Object during deletion

Object deleted


Note If all files are deleted, then library will switch from 'preview mode' to 'upload mode'.


Patching files through dataURL

In case only base64 string of a file/image is known, it can be provided to the library as dataURL to the library using fileData parameter.

The library converts the base64 string into a file and will treat just as normally selected file from local PC. Such files can be then be removed from the library or can be uploaded to S3 bucket in a regular manner.

The data is passed as an array using a @Input() parameter:

public fileData: IFileBase64Item[] = [{
    "dataURL": "",
    "name": "image2.jfif",
    "type": "image/jpeg",
  }, {
    "dataURL": "",
    "name": "kube-pod.png",
    "type": "image/png"
}]

The interface IFileBase64Item, available from library has the following properties.

interface IFileBase64Item {
   dataURL: "string",
   name: "string",
   type: string
}

It is important to consider the role of multipleFiles property when using the fileData input params.

In case multipleFiles is false, then the first element of the array is patched into the library. If it true, then the array elements are uploaded as files until the maxFiles limit is reached.

The above base64 data, when patched, will render the following output.

Single file

Single file Patch data

Multiple files

Multiple files Patch data

API Reference (Library)

Params

You can further configure the upload behavior through following API's. You can update the params object with the following values.

These values are available in the IPsyS3UploaderConfig interface.

interface IPsyS3UploaderConfig {
    uploadApi: "string",
    uuid: boolean,
    pathPrefix: "string",
    validExtensions: string[],
    width: "string",
    height: "string",
    multipleFiles: boolean,
    maxFileSizeinMB: boolean
    maxFiles: number,
    previewArray?: IFilePreview[],
    customDeleteUrl?: "string",
}
ParamDescriptionRequired
uploadApiAPI Endpoint of the function that generates the pre-signed urlY
uuidChanges file name to a random uuid(v4)N
pathPrefixAllows you to specify a path that will creates/updates a folder in S3 to update the image. Use '/' as a delimter. Note: During input, 'path prefix' should not end with '/' Eg: "public/" (invalid) "public" (valid) )N
validExtensionsOverides the default extensions allowed for uploadN
widthSpecifes the width of the drag area. Default is 20emN
heightSpecifes the height of the drag area. Default is 20emN
multipleFilesAllows to switch between single or multiple file upload mode. Default is 'false'N
maxSizeinMBAllows you to specify the maximum file size allowed for upload in MB. Default is '1 MB'N
maxFilesSpecifies the maximum number of files. Default value is 10.N
previewArrayProvides specifications of images/files to be previewed inside dropzone. The specifications are in the form of IFilePreviewN

These values are available in the IFilePreview interface used in Preview Mode.

interface IFilePreview {
    url: "string",
    key?: "string",
    name?: "string",
    canDelete?: "boolean",
    deleteUrl?: "string",
    fileType: "string",
    queryStringParameters?: "object"
}
ParamDescriptionRequired
urlAPI Endpoint of the file. In case the file is image, preview can be shownY
fileTypeDetermines to preview the icon or display an icon from provided applicationY
keyS3 Key of the objectN
nameName of the DocumentN
canDeleteA flag that determines whether to show the delete icon which call the delete methodN
deleteUrlAPI end point that library should call on invoking the delete methodN
queryStringParametersAn object that is converted into queryStringParams while making an query to backend. Eg: ({ key: "img_key" } ) to ?key=img_keyN

Methods

By using ViewChild() decorator in Parent component, you can call following methods in the library directly.

MethodDescription
upload( )Initiates file upload process
clearAll( )Clears all the selected Files

API Response

On successful upload, the following objects are returned from library. 1. URL of the uploaded file 2. xmlResponse that is returned from the S3 bucket as text 3. response contains details of the uploaded file/files


Note: URL and xmlResponse will be depreciated in the future


interface IPsyS3UploaderResponse {
   url: string | string[],
   xmlResponse: null | string | string[],
   response: IUploadedFileInfo | IUploadedFileInfo[]
};

interface IUploadedFileInfo {
   url: "string",
   xmlResponse: "string",
   name: "string",
   size: number,
   key: "string",
   type: "string"
}
{
    "url": "uploaded url of the object",
    "xmlResponse": "the xml response provided by s3 bucket, converted as a string",
    "response": {
        "url": "<uploaded url of the object>",
        "xmlResponse": "<the xml response provided by s3 bucket, converted as a string>",
        "name": "<name of the file>",
        "size": "<size of the file in bytes>",
        "key": "<object key>",
        "type": "<file Type of the object>"
    }
}

In case of multiple file upload, the responses are converted into array

{
    "url": "array of uploaded url of the objects",
    "xmlResponse": "the array of xml responses provided by s3 bucket, converted as a string",
    "response": [{
        "url": "<uploaded url of the object>",
        "xmlResponse": "<the xml response provided by s3 bucket, converted as a string>",
        "name": "<name of the file>",
        "size": "<size of the file in bytes>",
        "key": "<object key>",
        "type": "<file Type of the object>"
    }]
}

Note: The interface is exported from library and can be used in the parent component.

Events

In preview mode, when selecting or unselecting a file, an event - fileChange is emitted.

interface IOnFileChange {
    "action": eFileChangeStates,
    "index"?: "number",
    "data": {
        "dataUrl": "string",
        "name": "string",
        "type": "string",
        "size": "number"
    }
}

enum eFileChangeStates {
    "fileSelect" = "fileSelect",
    "fileRemove" = "fileRemove"
}

For example:

Single File

{
    "action": "fileSelect",
    "data": {
        "dataURL": "string",
        "name": "string",
        "type": "string",
        "size": "number"
    }
}

Multiple Files will have an additional 'index' property and will send out the its position in the file upload data-array.

{
    "action": "fileRemove",
    "index": 0,
    "data": {
        "dataURL": "string",
        "name": "string",
        "type": "string",
        "size": "number"
    }
}
2.0.1

2 years ago

2.0.0

2 years ago

1.3.4

2 years ago

1.3.3

2 years ago

1.3.2

3 years ago

1.3.1

3 years ago

1.2.2

3 years ago

1.2.0

3 years ago

1.2.1

3 years ago

1.1.1

3 years ago

1.1.0

3 years ago

1.0.4

3 years ago

1.0.3

3 years ago

1.0.2

3 years ago

1.0.1

3 years ago

1.0.0

3 years ago

0.0.1

3 years ago