Assets
We all know that working with assets (and mainly images) is really painful with Discord bots.
Indeed, embeds allow you to only use hosted images, which can be quite annoying to manage and to work with.
But TSCord solves this issue!
It has an auto image upload service that will scan every image file in the assets/images directory and upload them on imgur.
The uploaded images url are then saved in the image table of the database, along with some useful metadata.
Enable image upload
Go to src/config/general.ts and set the automaticUploadImageToImgur property to true
export const generalConfig: GeneralConfigType = {
    //...
    automaticUploadImagesToImgur: true,
    //...
}
Then, you'll have to get a Client ID in order to use the Imgur API.
Steps to follow:
- Sign in or register on imgur - https://imgur.com/
- Navigate to the following page to register an OAuth application - https://api.imgur.com/oauth2/addclient
- Fill the form with your application details
- On the authorization callback URL section, register the following URL: https://imgur.com/
- Click on "Save" and that's it!
Now just paste the Client ID in your .env file on the following property:
IMGUR_CLIENT_ID="YOUR_IMGUR_CLIENT_ID"
Now that the upload service is enabled, the next time you start the bot it'll upload all images recursively in the assets/images directory.
Plus, if an image is removed from the directory, it'll also be removed from both database and imgur, and if one has a broken link on imgur, it'll be automatically reuploaded!
Image entity
When an image is uploaded, the following data is saved in the db:
| Property | Type | Description | 
|---|---|---|
| fileName | string | The file name of the image (including its format) | 
| basePath | string | Relative path to the image, using the assets/imagesdirectory as root path | 
| url | string | The imgur hosted URL to access the image | 
| size | number | Total size of the image in Bytes | 
| tags | string[] | Editable list of tags to batch search images of the same type. By default, it will split the basePathof the image in different tags (ex: if the image is located inassets/images/foo/bar/image.png, it'll have thefooandbartags | 
The hash and deleteHash properties are also stored in the database, but it is meant for an internal operation so don't worry about these.
Use images
To use images in your code, just retrieve it from the database:
import { Database } from '@/services'
import { Image } from '@/entities'
import { Discord, Injectable } from '@/decorators'
@Discord()
@Injectable()
export default class PingCommand {
    constructor(
        private db: Database
    )
    @Slash({ name: 'ping' })
    async pingHandler(
        interaction: CommandInteraction
    ) {
        const image = await this.db.get(Image).findOne({ fileName: 'image.png' }) // find by fileName
        // or you can find by tags
        const imagesExplicit = await this.db.get(Image).findByTags(['foo', 'bar']) // will search for images that have 'foo' AND 'bar'
        const imagesNonExplicit = await this.db.get(Image).findByTags(['foo', 'bar'], false) // will search for images that have 'foo' OR 'bar'
        interaction.followUp(image.url)
    }
}