Scripting Helpers is winding down operations and is now read-only. More info→
Ad
Log in to vote
0

Table expected, got nil & Attempt to index upvalue (a nil value)?

Asked by 5 years ago

So, I'm trying to save a Table using DataStore. It's coming up with 2 errors. I'm trying to print every value inside of the table and it's giving me this error

16:26:55.044 - ServerScriptService.Script:16: bad argument #1 to 'pairs' (table expected, got nil)

16:26:55.044 - Stack Begin

16:26:55.044 - Script 'ServerScriptService.Script', Line 16

16:26:55.045 - Stack End

And everytime I exit my test in studio, or in a live server, it gives me this error when I'm trying to save my table.

16:26:56.480 - ServerScriptService.Script:25: attempt to index upvalue 'testList' (a nil value)

This is the ServerScript inside of ServerScriptService

01local testList = {33788911}
02local testStore = game:GetService("DataStoreService"):GetDataStore("TestStore")
03 
04game.Players.PlayerAdded:Connect(function(player)
05    local success,errorMessage = pcall(function()
06        newList = testStore:GetAsync("testData-KB")
07    end)
08 
09    if success then
10        testList = newList
11        print("Test Data Was Loaded!")
12    else
13        warn(errorMessage.." | There Was An Error Loading Test Data!")
14    end
15 
View all 33 lines...

I'm not sure if I've formatted something wrong or if I just did this wrong in general.

2 answers

Log in to vote
1
Answered by 5 years ago

There are quite a lot of errors here. I'll give you a list of each and how to fix them.

  1. You have to remember to make an actual variable for variables you're giving values too. In this case, you didn't set the value for 'newList' when you were giving it a value (if data was found). The reason you weren't getting an error for that is because the data was returning nil because nothing was being saved in the first place. I'll get into that part later though.

To fix this, you simply need to make the variable first.

01game.Players.PlayerAdded:Connect(function(plr)
02 
03    local NewList --New Variable
04 
05    local success, err = pcall(function()
06        NewList = testStore:GetAsync('testData-KB')
07    end)
08 
09 
10    if NewList and type(NewList) == 'table' then else return end --ends the function if data was not found or the data was not a table
11end

Also, I would recommend using function based variables for datastores. In yours you used a script wide variable 'TestList' accessible by all the functions. Which can easily cause data errors. So instead, put that variable inside the PlayerAdded function.

01game.Players.PlayerAdded:Connect(function(plr)
02 
03    local NewList
04 
05    local TestList  = {} --Inside the function
06 
07    local success, err = pcall(function()
08        NewList = testStore:GetAsync('testData-KB')
09    end)
10 
11 
12    if NewList then else return end
13 
14    if success then
15        TestList = NewList
View all 21 lines...

However after looking at the rest of your script, you clearly want it to say that a player has entered the game if their user ID is found. So just include a separate table with the UserID's you want, then go through that with a for loop. Like this

1local IDs = {1111} -- (player id you want to check)
2 
3for i, v in pairs(IDs) do
4    if plr.UserId == v then
5        print(plr.Name..' has joined the game!')
6    end
7end
  1. When the player leaves the game, you aren't actually saving any data.

You did 'testList:SetAsync' which wouldn't do anything, since testList is a table and not a DataStore. I think what you meant to do is something like this...

1game.Players.PlayerRemoving:Connect(function(plr)
2    local PlayerData = testData:GetAsync('testData-KB')
3 
4    if PlayerData and type(PlayerData) == 'table' then
5        testData:SetAsync('testData-KB', PlayerData)
6    else
7        warn('Failed to save data for '..plr.Name)
8    end
9end

What it looks like you're trying to do here is just print something when a certain player joins the game. You don't need datastores to do this. All you need to do is something like this

1local SpecialList = {19034023, 23049023, 32949205} --random ID's of the players you want announced when they join
2 
3game.Players.PlayerAdded:Connect(function(plr)
4    for i, v in pairs(SpecialList) do
5        if v == plr.UserId then
6            print('The special '..plr.Name..' has joined the server!')
7        end
8    end
9end

If this doesn't make sense to you, try looking more at DataStores here

0
Thanks too! Your's was just a bit more detailed killerbrenden 1537 — 5y
Ad
Log in to vote
1
Answered by 5 years ago

First error: Global variable testlist got changed to nil on line 10 because new data defaults to nil.

Second error: as mentioned in the first error, since you changed a global testlist to nil. So that error occurred as well.

I'd recommend to check if the value is nil or not.

1if not testlist then
2    testlist={}; -- this is a replacement value.
3end;
0
Thanks! killerbrenden 1537 — 5y

Answer this question