How to upload a base64 or binary image?

Hi there,
i have a problem with uploading a new image with the CMA.

The image comes through an input file via a form, so i only have it as binary or base64 data string.
I thried it with space.createAsset(), but this doesn’t work, because he wants an url path.

Older solutions, that i found, are deprecated like space.createUpload()

I’m not able to find any more solutions. :frowning:

So, is there any possibility to upload a new image via CMA with base64 or binary?

Thanks
Alex

Hi @alexander.blache ,

I believe you are following this tutorial:

The createUpload function should be now on the environment, not on the space:

Hi @Alma
i tryed several approaches by now, and yes i use the environment.createUpload().
I also tryed to send the image data in different formates. Stream Text and ArrayBuffer.

In my desperation, i also send it as a base64, wrote it to disc and send the path to my temporary file, nothing.

Every time its the same, the asset will be created in contentful, but the image isn’t there

After the upload part, i receive this, so i assume, that the upload was correct, after that, i create the createAsset() statement:

Upload {
sys: {
id: ‘5w5SsjVNiyLdOMh6m5B1jt’,
createdAt: ‘2021-05-18T12:46:30.000Z’,
expiresAt: ‘2021-05-20T00:00:00.000Z’,
createdBy: { sys: [Object] },
type: ‘Upload’,
space: { sys: [Object] }
}
}

My code:
async function newImage(json, image) {
try {
let type = json.type
let name = json.title

    const upload = await imageClient.getSpace(config.space)
    .then((space) => space.getEnvironment('master'))
    .then((env) => env.createUpload({file: image}))
    .then((upload) => {
        return upload
    })

    console.log("Upload ", upload)

    const asset = await managementClient.getSpace(config.space)
    .then((space) => space.getEnvironment('master'))
    .then((env) => {
        env.createAsset({
            fields: {
                title: {
                    'en-US': name
                },
                file: {
                    'en-US': {
                        fileName: name,
                        contentType: type,
                        uploadFrom: {
                            sys: {
                                type: 'Link',
                                linkType: 'Upload',
                                id: upload.sys.id
                            }
                        }
                    }
                }
            }
        })
        .then((asset) => {
            return asset.processForAllLocales({processingCheckWait: 2000})
        })
        .then((asset) => {
            return asset.publish()
        })
        .then ((asset) => {
            return asset
        })
    })
    .catch(console.error)
} catch (err) {
    console.log("ERROR: ", err);
}

}

Thats the result

Could you provide further aid?

Greetings Alex

Hi @alexander.blache ,

I tried to reproduce the issue, and I was able to identify that possibly some informations were missing in the method newImage. Here my code for reference:

async function newImage(json, image) {
  try {
    let type = json.type
    let name = json.title

    const upload = await managementClient.getSpace(config.space)
      .then((space) => space.getEnvironment('master'))
      .then((env) => env.createUpload(
        {
          file: image
        })
      )
      .then((upload) => {
        return upload
      })

    console.log("Upload ", upload)

    const asset = await managementClient.getSpace(config.space)
      .then((space) => space.getEnvironment('master'))
      .then((env) => {
        env.createAsset({
          fields: {
            title: {
              'en-US': name
            },
            file: {
              'en-US': {
                fileName: name,
                contentType: type,
                uploadFrom: {
                  sys: {
                    type: 'Link',
                    linkType: 'Upload',
                    id: upload.sys.id
                  }
                }
              }
            }
          }
        })
          .then((asset) => {
            return asset.processForAllLocales({processingCheckWait: 2000})
          })
          .then((asset) => {
            return asset.publish()
          })
          .then((asset) => {
            return asset
          })
      })
      .catch(console.error)
  } catch (err) {
    console.log("ERROR: ", err);
  }
}

Be aware to create a managementClient like the following (I assume from the previous code, you are using a config array. In this case token is the CMA token):

const contentful = require('contentful-management')

const managementClient = contentful.createClient({
  accessToken: config.token
})

And, as said before, the Json Data should look like this, where both (mime) type and title are set

let jsonData = {
  'type': 'image/png',
  'title': 'my-cute-cat.png'
}

Once you have all in place, you could pass a file to it in two ways:

const fs = require('fs')

let fileName = './myfile.png'
let imageResource = fs.readFileSync(fileName)
let imageStream = fs.createReadStream(fileName)

newImage(jsonData, imageResource)
// Alternatively
// newImage(jsonData, imageStream)

And, if you need to use a base64 image, before ‘opening’ the file, you would need to save it temporarily (ie: local file name ./image.png):

let base64String = ''
let base64Image = base64String.split(';base64,').pop();

fs.writeFile('./image.png', base64Image, {encoding: 'base64'}, function(err) {
  console.log('File created');
});

let image64File = fs.readFileSync('./image.png')

newImage(jsonData, image64File)

I hope this can help you and other people that might need it in the future.