How do I implement this Get Time and Date From Server Script into this DailyReward Script?

I am trying to figure out how to implement my TimeManager script into my DailyReward script. My DailyReward script currently rewards players every 24 hours with a daily reward or every 12 hours if they have a special item. The issue with the DailyReward script is if you set the time or date ahead offline, the daily reward will be available early even if the server time is the same. My TimeManager script gets the time and date from a server, which prevents you from setting time and date ahead to get the reward early.
This is the TimeManager script:

using System.Collections;
using UnityEngine;

public class TimeManager : MonoBehaviour
{
    /*
    NEED TO FIND A WORKING URL TO GET THE DATE AND TIME

    necessary variables to hold all the things we need.
    php url
    timedata, the data we get back
    current time
    current date
    */

    public static TimeManager sharedInstance = null;
    private string url = "http://leatonm.net/wp-content/uploads/2017/candlepin/getdate.php"; //This is a placeholder URL. Change URL in final script version.
    private string timeData;
    private string currentTime;
    private string currentDate;

    //Make sure there is only one instance of this always.
    private void Awake()
    {
        if (sharedInstance == null)
        {
            sharedInstance = this;
        }
        else if (sharedInstance != this)
        {
            Destroy(gameObject);
        }
        DontDestroyOnLoad(gameObject);
    }

    //Time fether coroutine
    public IEnumerator GetTime()
    {
        Debug.Log("==> step 1. Getting info from internet now!");
        //Debug.Log("connecting to php");
        WWW www = new WWW(url);
        yield return www;
        //if (www.error != null) {
        //Debug.Log ("Error");
        //} else {
        //Debug.Log("got the php information");
        //}
        Debug.Log("==> step 2. Got the info from internet!");
        timeData = www.text;
        string[] words = timeData.Split('/');
        //timerTestLabel.text = www.text;
        Debug.Log("The date is : " + words[0]);
        Debug.Log("The time is : " + words[1]);

        //setting current time
        currentDate = words[0];
        currentTime = words[1];
    }

    //get the current time at startup
    private void Start()
    {
        Debug.Log("==> TimeManager script is Ready.");
        //Debug.Log ("TimeManager script is Ready.");
        StartCoroutine("getTime");
    }

    //get the current date - also converting from string to int.
    //where 12-4-2017 is 1242017

    public string GetCurrentDateNow()
    {
        return currentDate;
    }

    //public int getCurrentDateNow()
    //{
    //string[] words = _currentDate.Split('-');
    //int x = int.Parse(words[0] + words[1] + words[2]);
    //return x;
    //}

    //get the current Time
    public string GetCurrentTimeNow()
    {
        return currentTime;
    }
}

This is the DailyReward script:

using System;
using UnityEngine;
using UnityEngine.UI;

public class DailyReward : MonoBehaviour {
    public static DailyReward DailyRwd;
    private Text chestTimer;
    private float msToWait = 86400000.0f;
    private Button chestButton;
    private ulong lastChestOpen;

    void Start(){
        if (DailyRwd == null)
        {
            DailyRwd = this;
        }

        chestTimer = GetComponentInChildren<Text>();
        chestButton = GetComponent<Button>();
        lastChestOpen = ulong.Parse(PlayerPrefs.GetString("LastChestOpen"));

        if(!isChestReady())
            chestButton.interactable = false;
    }

    void Update(){
        if (GameController.GameCon.tierTwoPrizeCharmThree == 1) {
            msToWait = 43200000.0f;
        }
        if(!chestButton.IsInteractable()){
            if(isChestReady()){
                chestButton.interactable = true;
                return;
            }

            //Set the timer
            ulong diff = ((ulong)DateTime.Now.Ticks - lastChestOpen);
            ulong m = diff / TimeSpan.TicksPerMillisecond;
            float secondsLeft = (float)(msToWait - m) / 1000.0f;

            string r = "";
            //Hours
            r += ((int)secondsLeft / 3600).ToString() + "h ";
            secondsLeft -= ((int)secondsLeft / 3600) * 3600;
            //Minutes
            r += ((int)secondsLeft / 60).ToString("00") + "m ";
            //Seconds
            r += (secondsLeft % 60).ToString("00") + "s"; ;
            chestTimer.text = r;
        }
        if (chestButton.interactable == true) {
            GameController.GameCon.claimFreeCards.SetActive (false);
        } else {
            GameController.GameCon.claimFreeCards.SetActive (true);
        }
    }

    public void ChestClick(){
        lastChestOpen = (ulong)DateTime.Now.Ticks;
        PlayerPrefs.SetString("LastChestOpen", lastChestOpen.ToString());
        chestButton.interactable = false;
        GameController.GameCon.alreadySwiped = true;
    }

    private bool isChestReady(){
        ulong diff = ((ulong)DateTime.Now.Ticks - lastChestOpen);
        ulong m = diff / TimeSpan.TicksPerMillisecond;
        float secondsLeft = (float)(msToWait - m) / 1000.0f;

        if(secondsLeft < 0){
            chestTimer.text = "Daily Reward";
            return true;
        }
            return false;
       
    }
}

If you just ask the server for the time and see if enough time has passed, that won’t cut it: a player will simply hack the on-device data so that it says they got the last reward yesterday, or last week.

To make this actually server authoritative, you also have to send the time from the player’s device up to the server and store it in a database specifically for that player, which means you have to uniquely identify each player.

Identifying each player means you also need to store personally identifiable information (PII) and are subject to GDPR and CCPA regulations, which you can google for yourself. Failure to do so is a fine in the neighborhood of $10,000/day per customer.

It’s a pretty big client-server piece of code, and your server has to be maintained and available 24/7.

I recommend just using the system clock and checking against a local variable. When you reach 10,000 daily average users, you can reward yourself by doing all the above work. But before you do you might want to see if anyone is even actually cheating. It’s pretty rare, and those people won’t pay anyway, generally.

1 Like

Well, I think it’s not exactly as you said Kurt.
The average player can change the clock hour, it takes 30 segs, but he would not search for your PlayerPrefs or your saving system to change the values. This take more time and you need a minimal knowledge about what you’re doing… and with a simply obfuscation it could be harder yet… And anyway, if so much people try to do it, you should start to think about hire people because your game is really addictive and people might spend bunch of money on it…

I am completely open to this possibility.

I just choose to spend my time making a better game experience rather than trying to change people’s behavior.

Good luck!

2 Likes

About how to apply it… Since GetTime() is a Coroutine and the time of the server could not be loaded when it’s required for daily reward, TimeManager should be which actives the Chest of DailyReward. When GetTime() ends without any error, it could active an event or set a bool at DailyReward. If this bool is not true, chest will be unavaiable.

Remember at ChestClick to change the value given to lastChestOpen to the value of the server, not the phone value… Or someone could open a chest with the phone date changed and imediatly open another indefinitely.

Also, i think it would be easier if you use Unixtime instead of date and time as strings.
https://showcase.api.linx.twenty57.net/UnixTime/tounix?date=now
This page gives you a timestamp with the UnixTime which you can store at an integer. You only need to know one day is 86400 seconds so if currentTime - lastTime > 86400, a day has been completed.