Upload saves from old database

Hi,

A cloud save system I am already using is being shut down. I’d like to export the data from that service and upload it to Unity Cloud save. Is this possible? The key would be the person’s email.

Hi there,

Thanks for sharing your question.

The simple answer is YES, you can use Cloud Save to store the data exported from another cloud storage solution, indeed this is someting we have been able to help some GameSparks customers with, in advance of their shutdown at the end of September.

However, there are a few things you will need to take into consideration in your implementation.

  • When a player accesses Cloud Save their data will ba stored against the PlayerID they were assigned by the UGS Authentication package when they signed in. So if you updload their exported data to UGS Cloud Save and save it by their email address, it will be stored under a different ID that cannot be accessed directly by the player from the Cloud Save SDK. But, fear not, there is a workaround.

Whilst the player can’t reach the exported and uploaded data directly, it is possible to access it from Cloud Code using a service token to access it, then save it back to Cloud Save under the player’s ID.

  • Export the data from the 3rd party storage solution. Preferrably put it in a database somewhere so you can track status and rerun the following steps if you have any problems or subsequent data updates.

  • Create a tool to iterate over the players in the export and use the Cloud Code and Cloud Save API to upload them to Cloud Save , keyed by some string based ID that is common to the player before and after the Cloud storage migration.

  • At runtime, have the player run a Cloud Code script that pulls their data from step 2 out of Cloud Save and re-saves it back to their own Cloud Save profile. This script should only be run once, the first time the player reconnects after you have exported their data from the other storage solution.

  • Be aware that there are storage limits on Cloud Save, you can find out more here. You are limited to 200 data ‘slots’ per player and each slot has a size limit of 16KB of JSON serializable data.

  • You may hit rate limits or experience occasional timeouts if you try and run an uploader (step 2.) too aggressively. So I would recommend using an intermediate Database to track the state of each player data upload, so you can retry and update etc…

  • Related to the last point, when uploading exported data (step 2.) It will be more efficient to send a batch of player data to Cloud Code as a single request, then have the Cloud Code iterate over the individual players hititng Cloud Save for each one. But be aware there is an execution time limit on each Cloud Code script. I found the optimal batch size to be ~25 players.

  • After you have uploaded a player’s data to Cloud Save (under their email or another arbritray ID), you won’t be able to see it in the Find A Player UI tool, because this wasn’t really an authenticated player, but you could write a little cloud code script to check it. Or you could run the Migration script below from within the Cloud Code dashboard editor to see if it can find and load your saved player data.

  • Finally, these additional Cloud Save entries that you are upploading will count towards your metered billing for Cloud Save and Cloud Code for the month, but there is a generous Free Tier. You can find pricing information here

This is the Cloud Code script I have used to upload player data to Cloud Code. It should be run from a tool that iterates over each exported player and uploads them in batches.

GsSavePlayerDataBulk

/*
*  -------- Save GameSparks User to Cloud Save--------------------
*  Accepts an Array containing multiple players
* Use Service Token to save GameSparks player's data to Cloud Save
* ----------------------------------------------------------------
*/

const _ = require("lodash-4.17");
const { DataApi } = require("@unity-services/cloud-save-1.2");

module.exports = async ({ params, context, logger }) => {

  const {
    projectId,
    playerId,
    serviceToken
  } = context;
 
  logger.info("Script parameters: " + JSON.stringify(params));
 
  const players = params.GsPlayerDataArray;
  logger.info ("Players Found = " + players.length)
 
  // Service Token used to Authenticate so we can hit another player's data
  const api = new DataApi({ accessToken: serviceToken});
  logger.info("Authenticated within the following context: " + JSON.stringify(context));
 
  for(const player of players) {
    await Save(player);
  }

  // Return the JSON result to the client
  return {
  
  };
 
  async function Save(player)
  {
    logger.info("Saving " + player.GameSparksUserID);
    logger.info("ProjectId " + projectId);
     
    // Attempt to Save GameSparks player data from Cloud Save
    const GSSave = await api.setItemBatch(projectId, player.GameSparksUserID, {"data" :
      [
        { key:"GameSparksUserID", value:player.GameSparksUserID },
        { key:"ttl", value:player.ttl },
        { key:"version", value:player.version },
        { key:"data", value:player.data }
      ]}
    );

    const GSSaveResponse = JSON.stringify(GSSave.data);
    logger.info('GSSave result ' + GSSaveResponse);
   
  }
};

This is a Cloud Code script that I have used to migrate data that was loaded into Cloud Save under a different key than the PlayerID. It should be run by the player at runtime

/*
*  -------- Migrate GameSparks User from Cloud Save------------------
*  Use Service Token to load another player's data from Cloud Save
*  then save it back to Cloud Save as for Authenticated Player
* -------------------------------------------------------------------
*/

const _ = require("lodash-4.17");
const { DataApi } = require("@unity-services/cloud-save-1.2");

module.exports = async ({ params, context, logger }) => {

  const {
    projectId,
    playerId,
    serviceToken
  } = context;
 
  // GameSparksUserID input parmameter present?
  logger.info("Script parameters: " + JSON.stringify(params));
 
  // Service Token used to Authenticate so we can hit another player's data
  const api = new DataApi({ accessToken: serviceToken});
  logger.info("Authenticated within the following context: " + JSON.stringify(context));
 
  logger.info('Search Key' + params.playerSearchKey);
 
  // Attempt to get other player's data from Cloud Save
  const GSLoad = await api.getItems(projectId, params.playerSearchKey, ["GameSparksUserID", "ttl", "version", "data"]);
  const GSLoadResponse = JSON.stringify(GSLoad.data)
  logger.info('GameSparks PlayerData Load result ' + GSLoadResponse);
 
  // Access Token used to Authenticate so we can hit authenticated player's data
  const api2 = new DataApi(context);
 
  // Attempt to Save player data to Cloud Save
  const UGSSave = await api2.setItemBatch(projectId, playerId, {"data" :
    [
      { key: GSLoad.data.results[0].key, value:GSLoad.data.results[0].value},
      { key: GSLoad.data.results[1].key, value:GSLoad.data.results[1].value},
      { key: GSLoad.data.results[2].key, value:GSLoad.data.results[2].value},
      { key: GSLoad.data.results[3].key, value:GSLoad.data.results[3].value}
    ]}
  );
  const UGSSaveResponse = JSON.stringify(UGSSave.data)
  logger.info('UGS PlayerData Save result ' + UGSSaveResponse);
    
 
  // Return the JSON result to the client
  return {
    GameSparksUserID: params.GameSparksUserID,
    GameSparksLoadResponse: GSLoadResponse,
    UGSSaveResponse: UGSSaveResponse
  };
};

I hope that gives you a few pointers. Let me know if you need any more details.

Thank you for the detailed answer! I really appreciate it. I am going to look into this and I’ll let you know if I run into some issues.

Hi @Laurie-Unity ,

I think I’ve got most of the script set up, however I am running into one issue. I am calling api.setItemBatch( and getting the error response from a Unity program that is attempting to upload the data

ScriptError
(422) HTTP/1.1 422 Unprocessable Entity
Unprocessable Entity
Invocation Error
RequiredError: Required parameter playerId was null or undefined when calling setItemBatch.

Does api.setItemBatch require a playerId now? Or am I missing something?

It looks like the playerID is required. Is this going to cause issues uploading GameSparks since we don’t want a player ID associated with these entries especially if we are going to send up hundreds of separate players. Or do I need to generate a bunch of anonymous player IDs?

You linked to the documentation for Cloud Save v1.0

My example works and is using Cloud Save v1.2

const { DataApi } = require("@unity-services/cloud-save-1.2");

and calls setItemBatch with the playerID substituted with the gameSparks UserID

    // Attempt to Save GameSparks player data from Cloud Save
    const GSSave = await api.setItemBatch(projectId, player.GameSparksUserID, {"data" :
      [
        { key:"GameSparksUserID", value:player.GameSparksUserID },
        { key:"ttl", value:player.ttl },
        { key:"version", value:player.version },
        { key:"data", value:player.data }
      ]}
    );

My second script, initiated by the player, reloads the data based on the gameSparks UserID and saves it back under the player’s own playerID. e.g. I create an additional, intermediate Cloud Save record based on the player’s gameSparks ID

Thanks for the quick response.

I have copy pasted your code into the Cloud Code and still getting an issue. Here is the code on my Unity side. Maybe there is something I am missing.

    public async void Awake()
    {
        await UnityServices.InitializeAsync();
        await AuthenticationService.Instance.SignInAnonymouslyAsync();

        var arguments = new Dictionary<string, object>
        {
            { "GsPlayerDataArray", jsonData },
        };
        CloudCodeResponse response = await CloudCodeService.Instance.CallEndpointAsync<CloudCodeResponse>("UploadSaveDataFromGS", arguments);

        Debug.Log(response);
    }

And here is the JSON data I am sending up

"[ { "GameSparksUserID": "newplayer@website.com", "ttl": "5555", "version": "1.3", "data": "this is data" }]"

And still getting that same error response that the PlayerID is not provided. Do I need to send something else up from Unity’s side?

I am not too familiar with Javascript so maybe I am creating the data to be sent up wrong.

Can you DM me a link to your Cloud Code Script and I’ll take a look

Okay great, I DM’d you.

After chatting with the very helpful @Laurie-Unity about this, turns out you need to call on JArray.Parse before submitting to the Cloud Code.

Should be

        var arguments = new Dictionary<string, object>
        {
            { "GsPlayerDataArray", JArray.Parse(jsonData) },
        };

The C# wrapper won’t take a string parsed into JSON, has to be parsed into NewtonSoft’s JArray to work correctly.

2 Likes