Check if StreamWriter sw isFinished

Hi everyone,
I have a problem with my StreamWriter.

Explanation: I have a data.txt in my resources-folder and I write metadata in it. Then I upload the data.txt file to my server. The problem is, that my program uploads the data.txt first and changes the text in it later. Bad timing?

I call the create-function and the upload-function with one button at the same time.

I tried several debugs, but everything looks fine to me. I can’t figure out what is wrong.
Maybe there is a function that I can call which shows me when my StreamWriter is finished with writing.

//This is where I save my data during runtime. This works fine.
public static class Values
{
    public static string user_code;
}
public class CreateMetaFile : MonoBehaviour
{
using (StreamWriter sw = new StreamWriter("Assets/Resources/data.txt"))
        {
            //Add my string to the file.
            sw.Write(str);
        }
        //Every output shows: str == Values.user_code, so that is fine.

        //A StreamReader for testing shows the same result.
using (StreamReader sr = new StreamReader("Assets/Resources/data.txt"))
        {
            string line = sr.ReadLine();
            if (line == str)
                isReady = true;
        }
        //That means, at this point, the data should be written. The next step is, to get the isReady : bool and upload my file.
}
//I try to ask if the file is written, to be absolutely sure.
IEnumerator Upload_DataFile()
    {
        yield return new WaitUntil(() => CheckMetaData() == true);
    }

    private bool CheckMetaData()
    {
        if (this.GetComponent<CreateMetaFile>().isReady)
        {
            Debug.Log("MetaData is written.");
            return true;
        }
        else
        {
            Debug.Log("Waiting for MetaData to be written.");
            return false;
        }
    }

I’m not clear on exactly what the problem is here. Can you elaborate? What is going wrong? What are you expecting to happen and what is happening instead?

  1. I have a data.txt
  2. data.txt has content: “my_fist_content”
  3. On runtime, I change the content with: “my_second_content”
  4. Then I upload the data.txt to my server.
  5. I open my data.txt on the server and expect “my_second_content”, but it is “my_first_content”.
  6. If I do the same again and change my content to: “my_third_content”, then tha data.txt, will have “my_second_content”.

That means, that my program uploads my data.txt first and writes the new content second, which is bad.
My program should write the new content first and then, if everything is wrote, upload second.

Can you share your uploading code?

Ok, I constrained my errors. The problem is, that the Streamwriter won’t update early enough or won’t write properly.
I tried to manually flush and close this buddy but he stays riot.

    public void CreateMyTextFile()
    {
        str += Values.user_code;

        for(int i = 0; i < Values.my_images.Count; i++)
        {
            Create_My_String(i);
        }

        using (StreamWriter streamWriter = new StreamWriter("Assets/Resources/data.txt"))
        {
            //Add my string to the file.
            streamWriter.Write(str);
            print("my str: " + str);

            streamWriter.Flush();
            streamWriter.Close();

            isReady = true;
        }

        print("did it wrote?: " + Resources.Load<TextAsset>("data").ToString());
    }

This is kind of stupid. The text in data.txt only changes if I refresh the textfile. After the refresh, everything works fine. Where is the error?

Load the file directly, don’t go via Unity resources as they don’t update in editor play mode unless you force them too. You’ll want to be writing and reading a text file directly in a build anyway unless you don’t plan on modifying it once built.

Thx for the answer. I need to change the text in that textfile and upload it. That means i need to look after the Resources-Update thing. There should be another solution… create TextAsset at runtime, and upload it. This should work am I right?

I made it work, kind of a force. Can’t find a better solution and get a error, but it works.

    public void CreateMyTextFile()
    {
        str += Values.user_code;

        for(int i = 0; i < Values.my_images.Count; i++)
        {
            Create_My_String(i);
        }

        TextAsset textAsset = new TextAsset();
        AssetDatabase.Refresh(); //Error: Failed to load '.../Resources/data.txt'. File may be corrupted or was serialized with a newer version of Unity.
        AssetDatabase.CreateAsset(textAsset, "Assets/Resources/data.txt");
       
        StreamWriter sw = new StreamWriter("Assets/Resources/data.txt");
        sw.WriteLine(str);
        sw.Close();
        AssetDatabase.SaveAssets();
        AssetDatabase.Refresh();

        StreamReader sr = new StreamReader("Assets/Resources/data.txt");
        string line = sr.ReadLine();
        sr.Close();

        //Compare the line only, because there is a "Enter" at the end of the data.txt, that I don't want.
        if (line == str)
        {
            print("isReady = true");
            isReady = true;
        }
        else
        {
            print("isReady = false");
        }
    }

Asset database stuff isn’t going to work when you try to build your game. It’s an editor only namespace.

Your problem is that you’re mixing up text files and text assets. They’re not the same thing and it’s not really supported to modify assets at runtime. As @WarmedxMints_1 implied, just use a plain old text file. Write the initial file to Application.persistentDataPath, and continue to read/write the file from there. You can use a text asset as the initial template for the file if you want. After that though, forget about “assets” and just work with plain old files.

1 Like

The good old file-stuff is what i wanted all the time. I get rid of the

//TextAsset textAsset = new TextAsset();
//AssetDatabase.Refresh();
//AssetDatabase.CreateAsset(textAsset, "Assets/Resources/data.txt");

and still, everything works fine. The error is gone, too.

Solution to my problem was:
AssetDatabase.SaveAssets();
AssetDatabase.Refresh();