OnTrack by Uplegacy | The first self-hosted Roblox In-Game activity tracker with Discord integration

Hello everyone! I am Marc, a little known developer on the Roblox platform. I currently run UpLegacy Companies, which is a Roblox development and asset management firm. We have been providing many cheap Roblox group solutions and services. And with the recent increase in popularity in our Activity Tracking module I decided to release a SELF-HOSTED version to the public! ( This is actually my first community resource post :slight_smile: )

The following features are included with OnTrack:

Features
  • Automatic logging and tracking of all your staff members in-game activity!
  • Support for MULTIPLE games! (Only counts to a total time in-game and not separate game tracking. Future updates will release that!)
  • DISCORD integration with instructions on a SLASH COMMAND bot!
  • AUTOMATIC searching of usernames via Discord Integration. (It will autofill any username of your staff that has been in-game.)
Showcase

https://youtube.com/shorts/pdq0Z4hzsmo?feature=shared

And that is that! I hope people out there will find this useful!
Lets get into it!

Requirements:

  • IT IS RECOMMENDED YOU ARE AN EXPERIENCED CODER! (The guides are as simple as I could make them, but it we are not providing support for free self hosted versions.)

  • Linux Server (And basic understanding of Linux)

  • Domain or Public IPV4 Address on your Server (I will show Cloudflared Linking Instructions)

  • Discord Bot Token (And recommend understanding of Java Script.)

  • A lot of time and patience.

(If you don’t meet the requirements above but are still interested in our service, please contact us for paid plans!)

Here we go! (I recommend making a project directory and separate directories for each step EXCEPT Step 1.)

1. Your Server is Accessible from the Internet

Ensure that your Domain is Cloudflared Proxied (Or have a publicly Accessible IPV4 Address)

  • Please ensure that your domain is open through a public IPV4 address. If not, heres a cloudflare tutorial.
  1. Download the cloudflared binary:
curl -O https://downloads.cloudflare.com/cloudflared/latest/linux/amd64/cloudflared
  1. Make it executable:
chmod +x cloudflared
  1. Move it to a directory in your PATH:
sudo mv cloudflared /usr/local/bin/
  1. Verify the installation:
cloudflared --version

If it returns a version! Congrats, Cloudflared is installed! If you need to trouble shoot, check Cloudflare Documentation.

(Find that here: Cloudflare Errors · Cloudflare Support docs)

Now that you have Cloudflared installed, lets make a tunnel!

  • Authenticate:
cloudflared tunnel login

This will open a browser to authenticate (or create) your Cloudflare account.

  • Create a new tunnel:
cloudflared tunnel create <TUNNEL_NAME>

Replace <TUNNEL_NAME> with a name for your tunnel. This command generates a tunnel credential file stored at ~/.cloudflared/<TUNNEL_ID>.json. I recommend that you copy this down for later.

LOOK HERE!! Awesome! You’ve got a tunnel. Your good to go! Just make sure your DNS servers are updated! (Roblox.yourdomain.com is tunneled to TUNNEL_ID.cfargotunnels.com) Now lets define the Ingress Rules! (Ingress is what port incoming requests go to on your server. In this case, our server is going to run on port 5000.)

  1. Create a configuration file: Create a file called config.yml in the ~/.cloudflared/ directory:
nano ~/.cloudflared/config.yml
  1. Add ingress rules to the config.yml file: Example config.yml:
tunnel: <TUNNEL_ID> 
credentials-file: /home/user/.cloudflared/<TUNNEL_ID>.json

ingress:
  - hostname: roblox.yourdomain.com
    service: http://localhost:6000
  - service: http_status:404

Replace <TUNNEL_ID> with the ID of your tunnel (copied down from earlier), and configure the hostname and services as required. You set your hostname up in your DNS settings.

Step 4: Run the Tunnel

  1. Run the tunnel manually:
cloudflared tunnel run <TUNNEL_NAME>

And thats it! You got it!
Now, if you SSH’ed into your client, odds are after you closed the terminal the Cloudflare tunnel will stop. If it doesn’t, congrats! You can move onto the next step. If it does, heres is how to troubleshoot!

Start the Cloudflare Tunnel with nohup

nohup allows you to run a command in the background, ignoring hangups (disconnections).

  1. Install nohup and run the tunnel!
sudo apt install nohup
  1. Now run the tunnel!
nohup cloudflared tunnel run <TUNNEL_NAME> > cloudflared.log 2>&1 &

Whew! Hard part is over! For those of you that have a public IPv4, I envy you!

2. Install your Database

Hey hey! Welcome! I’m glad you got through that Cloudflare mess- its almost as bad as 285 at 5:30pm on a Friday…

Lets get your database running! For this we are using sqlite (yes, you could use mySQL. But I don’t.)

Install sqLite

Run:

sudo apt update sudo apt install sqlite3 -y

Step 2: Create a New Database

sqlite3 activity.db

Whew! Database done. That was a little easier wasn’t it?

Run

.quit
to exist back to terminal.

3. Get your Discord Bot Running

Alright! It only get easier now! I’m going to spoon-feed it all to you from here!
We are going to get your discord bot running now.

Go to ULC - Github OnTrack and find the Code button in the top left. Now find the Download ZIP and download it! Then extract it to your computer and find scbot folder. Now take bot.js and add it to your directory! (sudo nano /home/user/project/bot/bot.js). Replace all the config details use role id’s and now Control + X and save your file! In the main terminal run:

sudo apt install npm nodejs -y

And after that is done, run

sudo npm i

This will install all your node modules.

Now lets get your node rolling!

Run
npm start bot.js

Now your bot is running, but again- you might want it online all the time eh?

Use this to do that!

  • Install PM2 Globally:
sudo npm install -g pm2

3. Start the Application with PM2

Use PM2 to start the file.

  • Start the application:
pm2 start bot.js
  • Allow PM2 to Start on Boot
pm2 startup

This will output a command specific to your Linux distribution. Run the command as instructed.

  • Save the current processes:
pm2 save

Now your bot should be running successfully! (If you can’t find your token, or client ID’s I’d recommend you look them up.

Lets move on to the API!

4. Building your API

Remember that ZIP file from earlier? Well we need it again, inside there is a folder names scapi. Open that folder, edit the group and point requirements, and copy and paste server.js into a new file in your directory called server.js (Same way as before just change the name from /bot/bot.js to /server/server.js)

5. Add to Roblox

You’ve made it!
Now, ensure HTTP request are on in your game, and add this script to your roblox game!

local HttpService = game:GetService("HttpService")
local Players = game:GetService("Players")
local activityApiUrl = "https://yourdomain.name/log"
local groupId = 0  -- Replace with your group ID
local minRank = 2  -- Minimum rank required to log activity (inclusive)
local maxRank = 8  -- Maximum rank allowed to log activity (inclusive)

-- Function to check if the player is in the allowed rank range
local function isPlayerInRankRange(player)
	local rank = player:GetRankInGroup(groupId)
	return rank >= minRank and rank <= maxRank
end

-- Function to log activity data by sending a POST request to the API
local function logActivityToAPI(username, time)
	local data = {
		username = username,
		time = time
	}

	local jsonData = HttpService:JSONEncode(data)  -- Convert the data to JSON

	local headers = {
		["Content-Type"] = "application/json"
	}

	-- Send a POST request to the API
	local success, response = pcall(function()
		return HttpService:PostAsync(activityApiUrl, jsonData, Enum.HttpContentType.ApplicationJson)
	end)

	if success then
		print("Successfully logged activity for " .. username .. " to the API.")
	else
		print("Failed to log activity for " .. username .. ": " .. response)
	end
end

-- Track time spent by each player in the game
local playerTimes = {}  -- Table to store player times

-- Automatically log activity when a player joins, if they meet the rank requirements
Players.PlayerAdded:Connect(function(player)
	if isPlayerInRankRange(player) then
		print(player.Name .. " is in the allowed rank range. Logging their activity.")

		-- Track the time the player spends in the game
		local startTime = tick()  -- Get the current time when the player joins

		-- When the player leaves, calculate the time spent in-game
		player.AncestryChanged:Connect(function(_, parent)
			if not parent then  -- Player has left the game
				local endTime = tick()  -- Get the current time when the player leaves
				local timeSpent = math.floor((endTime - startTime) / 60)  -- Convert time to minutes

				-- Send the activity data as a POST request to the API
				logActivityToAPI(player.Name, timeSpent)
				print(player.Name .. " played for " .. timeSpent .. " minutes.")
			end
		end)
	else
		print(player.Name .. " does not have the required rank to log activity.")
	end
end)

That’s it! Your done! Let me know any feed back and how this could be improved/bugs!
Have a great day all, and I hope you find this useful!

Marc

6 Likes

cc @pyxfluff

Who cares??? The aesthetics of the thread has absolutely nothing to do with the thread topic at all. Plenty of resources have limited emphasis on aesthetics, because the point is the code, models, or concepts being discussed or made available. These criticisms are as relevant as commenting on OP’s avatar and basically come off as just posting criticism for the sake of spreading negativity or giving yourself boosted post count. Again, these criticisms have nothing to do with the actual content of the thread.

__

@ OP

I’m sure this will be a useful tool or starting point for developers looking to get better external monitoring/analysis on a specific player-subset. Not that this necessarily needs it, but are you planning to introduce new features for people to look out for, or is this a finished product (pending minor fixes that need addressing)?

1 Like

I first off, apologize that my brand appeal is not what was expected.
I hear your criticism, and soon intend to correct the branding images.
However this was the best I could do in the dedicated time I had. (Graphic design- not my thing…)

—

In response! Yes!
I will be releasing (bi-weekly or monthly) updates and features to this project and intend to keep it supported! (Additionally, I have will probably introduce my support team and provide limited staff support for this in the future.)

As of current I am working on individualized sessions and separating games for the log!
Also, in tandem to this I may release the automatic promotions via leaderstats points value system I have!

But hopefully the concept will be found relevant and useful. And I know a few services who have been looking for concepts like this!

This is simply an activity logging module, that additionally will be updated with more features.
This is a SELF HOSTED version, as stated in my post. Last I recall BloxCord is both, without a self hosted tutorial, and currently halted.

1 Like

I can’t see any “Self Hostet” in the title, only thinks that e.g Bloxcord already does.

Well if you’d care to read the content of my message, I can assure you that I very clearly state SELF HOSTED within the first paragraph.

v1.3

Addition of Roblox Points Auto-Promote

Now found in our Github is the server.js, and autopromote.lua scripts available to all users to have a fully self hosted points autopromote system.

A guide may be featured soon, and better organiztion to come.

Happy New Year!

(Additional planned features include, ranking centers, application centers, and so much more! We always accept suggestions.)

Bug Fix and Github Re-Organization
v1.3.1

Patched a weird issue where the Roblox token didn’t validate for some users.

Additionally reworked the organization on the files, as well as start implementation of a documentation site.

Additionally, we are offering setup support via our communications server.

REWORKING THE CLOUDFLARE SETUP

Notice of Intent to Privatize
Greetings OnTrack Community,

Starting February 4th, 2025, all OnTrack assets will be removed from any free distributions and all updates will only be available through a paid version on our communications server.

We apologize for this, however with lack in feedback or any community interest we do not see any further reasoning to leave this project free.

There is still a chance to get a free license in our server before this product becomes paid! So don’t wait!

Thank you for all the support,

Marc

I feel like this is just BloxStrapRPC with extra steps, no?