Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[Feature] Initial pass at removing qs logging in view of the PlayerEventLogging System #4542

Open
wants to merge 42 commits into
base: master
Choose a base branch
from

Conversation

neckkola
Copy link
Contributor

Description

In discussions with the devs, this is the start to migrate the qs logging system into the player event system. Given that the player event logging system uses a json payload, this will allow for that payload to be flatten into 'qs like' structures to allow for easier server operator queries.

At present, this enables:
aa purchase
killed npc, named and raid mobs
loot
merchant selling and purchasing
npc handins
player speech (off by default)
trades

The qs logging code for the above is still present. This can be removed (now or in the future) to allow for a migration path for operators.

Type of change

Please delete options that are not relevant.

  • New feature (non-breaking change which adds functionality)
  • Breaking change (fix or feature that would cause existing functionality to not work as expected)
  • This change requires a documentation update

Testing

Tested locally on RoF2. Requires further testing and documentation on my part. I wanted to start the review process.

Clients tested:
N/A

Checklist

  • I have tested my changes
  • I have performed a self-review of my code. Ensuring variables, functions and methods are named in a human-readable way, comments are added only where naming of variables, functions and methods can't give enough context.
  • [TBD] I have made corresponding changes to the documentation (if applicable, if not delete this line)
  • I own the changes of my code and take responsibility for the potential issues that occur
  • If my changes make database schema changes, I have tested the changes on a local database (attach image). Updated version.h CURRENT_BINARY_DATABASE_VERSION to the new version. (Delete this if not applicable)

Copy link
Member

@Akkadius Akkadius left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Amazing work!

Some things we need to follow up on

  • All QS table code needs to be removed, period both in world and in QS ingestion. We will provide an optional transformer who wants to transform old tables to new data.
  • We will provide a data transformer from old QS tables to new. I can take this on.
  • Player event processing code needs to be on its own thread and its own database connection, I can also take this on
  • We need a command in world that exposes the ETL settings in JSON format so it can feed into Spire admin. Similar to ./bin/world database:schema
  • As you see in the PR comments, most of the vectors can be reserved on which does help substantially to bulk allocate memory once instead of doing it each event. We know the size of all of the data coming in, so we just need to reserve again even if it means doing another loop prior to fetch each event, call reserve in a switch.

out_entries.augment_5_id = augments[4];
out_entries.augment_6_id = augments[5];
}
etl_queues.npc_handin_entries.push_back(out_entries);
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Since we know the size of etl_queues.npc_handin_entries from in.handin_items we can do a reserve before the loop to pre-allocate memory

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

While we know the 'entries' sizes, there could be other events of the same type in the batch queue, so we would need to cycle through the batch_queue to parse the total size for the potential multiple events. Would this be beneficial, instead of the dynamic size adjustments as the vectors are add to? If so, happy to do that.
Same comment for all the reserve items.

Copy link
Member

@Akkadius Akkadius Nov 17, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yeah it makes a huge difference to pre-allocate especially with higher event volume.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Added reserve to all etl_queues.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thank you sir

common/events/player_event_logs.cpp Outdated Show resolved Hide resolved
common/events/player_event_logs.cpp Outdated Show resolved Hide resolved
common/events/player_event_logs.cpp Outdated Show resolved Hide resolved
common/events/player_event_logs.cpp Outdated Show resolved Hide resolved
@@ -123,21 +134,466 @@ void PlayerEventLogs::ProcessBatchQueue()

BenchTimer benchmark;

EtlQueues etl_queues{};
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Looking at all of this in ProcessBatchQueue and in our deletions, I think we should wrap these functions in thread calls so we don't tie up the main thread.

This also means that we should create a separate database connection just for processing player events because what happens is even if we had all of this in another thread its going to be lock contended with database locks on the main thread.

World needs to be kept as lightweight as possible in the main thread and this certainly adds weight if anything starts to take more than 500ms which it most certainly could here.

Same would go for using QS if folks use QS to process.

I may end up taking this one on.

common/events/player_event_logs.cpp Show resolved Hide resolved
common/events/player_event_logs.cpp Show resolved Hide resolved
zone/client.cpp Show resolved Hide resolved
zone/client.cpp Outdated Show resolved Hide resolved
{
auto results = db.QueryDatabase(
fmt::format(
"SELECT AUTO_INCREMENT FROM information_schema.tables WHERE TABLE_NAME = '{}';",
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We should assume that the server user can't directly query information_schema is there a reason we leaned into this versus max + 1 ?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yes. I ran into a couple of test cases that failed. If I empty an etl table, the max + 1 will return 1, when in fact the auto_increment may be 100. Further, once the retention period is hit, the next id via max + 1 will again be different from the auto_increment.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yeah we will still need an alternative to this.

  • information_schema can be out of sync
  • we may not have permission to query information_schema
  • This query also doesn't filter on DB_SCHEMA as you can end up with many table names in different schemas

I'm thinking we could pull from

SHOW TABLE STATUS LIKE 'your_table';

Copy link
Member

@Akkadius Akkadius Jan 9, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I rigged this up for us to use.

uint64_t Database::GetNextTableId(const std::string &table_name)
{
	auto results = QueryDatabase(fmt::format("SHOW TABLE STATUS LIKE '{}'", table_name));
	for (auto row : results) {
		for (
			int row_index = 0;
			row_index < results.ColumnCount();
			row_index++
			) {
			std::string field_name = Strings::ToLower(results.FieldName(row_index));
			if (field_name == "auto_increment") {
				std::string value = row[row_index] ? row[row_index] : "null";
				return Strings::ToUnsignedBigInt(value, 1);
			}
		}
	}

	return 1;
}
void WorldserverCLI::TestCommand(int argc, char **argv, argh::parser &cmd, std::string &description)
{
	description = "Test command";

	if (cmd[{"-h", "--help"}]) {
		return;
	}

	for (auto &table: DatabaseSchema::GetStateTables()) {
		LogInfo("Table name [{}] next_free_id [{}]", table, database.GetNextTableId(table));
	}
}
eqemu@743ff8705acf:~/server$ ./bin/world test:test

  World |    Info    | TestCommand Table name [adventure_members] next_free_id [1] 
 World |    Info    | TestCommand Table name [banned_ips] next_free_id [1] 
 World |    Info    | TestCommand Table name [bug_reports] next_free_id [1] 
 World |    Info    | TestCommand Table name [bugs] next_free_id [1] 
 World |    Info    | TestCommand Table name [buyer] next_free_id [1] 
 World |    Info    | TestCommand Table name [buyer_buy_lines] next_free_id [1] 
 World |    Info    | TestCommand Table name [buyer_trade_items] next_free_id [1] 
 World |    Info    | TestCommand Table name [completed_shared_task_activity_state] next_free_id [1] 
 World |    Info    | TestCommand Table name [completed_shared_task_members] next_free_id [1] 
 World |    Info    | TestCommand Table name [completed_shared_tasks] next_free_id [1] 
 World |    Info    | TestCommand Table name [discord_webhooks] next_free_id [1] 
 World |    Info    | TestCommand Table name [dynamic_zone_members] next_free_id [1] 
 World |    Info    | TestCommand Table name [dynamic_zones] next_free_id [1] 
 World |    Info    | TestCommand Table name [expedition_lockouts] next_free_id [1] 
 World |    Info    | TestCommand Table name [expeditions] next_free_id [1] 
 World |    Info    | TestCommand Table name [gm_ips] next_free_id [1] 
 World |    Info    | TestCommand Table name [group_id] next_free_id [1] 
 World |    Info    | TestCommand Table name [group_leaders] next_free_id [1] 
 World |    Info    | TestCommand Table name [instance_list] next_free_id [107] 
 World |    Info    | TestCommand Table name [ip_exemptions] next_free_id [1] 
 World |    Info    | TestCommand Table name [lfguild] next_free_id [1] 
 World |    Info    | TestCommand Table name [merc_buffs] next_free_id [1] 
 World |    Info    | TestCommand Table name [merchantlist_temp] next_free_id [1] 
 World |    Info    | TestCommand Table name [mercs] next_free_id [1] 
 World |    Info    | TestCommand Table name [object_contents] next_free_id [1] 
 World |    Info    | TestCommand Table name [raid_details] next_free_id [1] 
 World |    Info    | TestCommand Table name [raid_leaders] next_free_id [1] 
 World |    Info    | TestCommand Table name [raid_members] next_free_id [1] 
 World |    Info    | TestCommand Table name [reports] next_free_id [1] 
 World |    Info    | TestCommand Table name [respawn_times] next_free_id [1] 
 World |    Info    | TestCommand Table name [saylink] next_free_id [3788] 
 World |    Info    | TestCommand Table name [server_scheduled_events] next_free_id [1] 
 World |    Info    | TestCommand Table name [spawn2_disabled] next_free_id [1] 
 World |    Info    | TestCommand Table name [player_event_aa_purchase] next_free_id [1] 
 World |    Info    | TestCommand Table name [player_event_killed_npc] next_free_id [1] 
 World |    Info    | TestCommand Table name [player_event_killed_named_npc] next_free_id [1] 
 World |    Info    | TestCommand Table name [player_event_killed_raid_npc] next_free_id [1] 
 World |    Info    | TestCommand Table name [player_event_log_settings] next_free_id [1] 
 World |    Info    | TestCommand Table name [player_event_logs] next_free_id [13707] 
 World |    Info    | TestCommand Table name [player_event_loot_items] next_free_id [1] 
 World |    Info    | TestCommand Table name [player_event_merchant_purchase] next_free_id [1] 
 World |    Info    | TestCommand Table name [player_event_merchant_sell] next_free_id [1] 
 World |    Info    | TestCommand Table name [player_event_npc_handin] next_free_id [1] 
 World |    Info    | TestCommand Table name [player_event_npc_handin_entries] next_free_id [1] 
 World |    Info    | TestCommand Table name [player_event_speech] next_free_id [1] 
 World |    Info    | TestCommand Table name [player_event_trade] next_free_id [1] 
 World |    Info    | TestCommand Table name [player_event_trade_entries] next_free_id [1] 
 World |    Info    | TestCommand Table name [shared_task_activity_state] next_free_id [1] 
 World |    Info    | TestCommand Table name [shared_task_dynamic_zones] next_free_id [1] 
 World |    Info    | TestCommand Table name [shared_task_members] next_free_id [1] 
 World |    Info    | TestCommand Table name [shared_tasks] next_free_id [1] 

@@ -441,7 +441,7 @@ int main(int argc, char **argv)
}

if (player_event_process_timer.Check()) {
player_event_logs.Process();
std::jthread event_thread(&PlayerEventLogs::Process, &player_event_logs);
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

There's also a spot in queryserv

@Akkadius
Copy link
Member

Akkadius commented Nov 18, 2024

Amazing work sir!

One more stretch goal if you are down for it -

  • Have zoneservers directly connect to queryserv if it is available so that events go straight to queryserv instead of going through world

@Akkadius
Copy link
Member

I'll be picking this one up in the next few days

@fryguy503
Copy link
Contributor

A note on PlayerEventLogging, it appears to be missing augs for anything that has augs (Destroy, turn-ins, etc)

@neckkola
Copy link
Contributor Author

A note on PlayerEventLogging, it appears to be missing augs for anything that has augs (Destroy, turn-ins, etc)

Thanks Trust. Looks like those events all need to be updated to include augments. I can add that in.

@fryguy503
Copy link
Contributor

A note on PlayerEventLogging, it appears to be missing augs for anything that has augs (Destroy, turn-ins, etc)

Thanks Trust. Looks like those events all need to be updated to include augments. I can add that in.

Thank you! I assume Spire will need to be updated by @Akkadius to support this as well.

Add retention for etl tables
Rename tables/fields to etl nomenclature
Combine database work to one atomic load
testing passed though appears that the event itself has a few bugs.  Will fix them in another commit
Cleanup and refactor a few items.
Cleanup and Formatting
Playerevent::Trade
PlayerEvent::Speech (new event to mirror functionality of qs_speech
Playerevent::KilledNPC, KilledNamedNPC and KilledRaidNPC
Add etl for Playerevent::AA_purchase
@Akkadius Akkadius marked this pull request as ready for review January 9, 2025 07:48
@Akkadius
Copy link
Member

Akkadius commented Jan 9, 2025

Made quite a bit of progress

  • Rebased to master
  • Fixed the database manifest since there were query errors
  • Swapped out GetNextTableId method to one generic method for all tables
  • Statically load the counter variable so its not re-allocated each process run
  • Created QS conversion scripts to all player event tables and that is in the optional folder now
  • Removed all qs_tables and all code referencing said tables
  • Simplified code in ProcessBatchQueue, event truncation

Event truncation now truncates in retention groups that are the same so we can heavily reduce individual queries. It looks like the below.

 World | PlayerEven | ProcessRetentionTruncation Running truncation 
 World |   Query    | QueryDatabase DELETE FROM player_event_aa_purchase WHERE created_at < (NOW() - INTERVAL 7 DAY) -- (0 rows affected) (0.000070s)
 World |   Query    | QueryDatabase DELETE FROM player_event_loot_items WHERE created_at < (NOW() - INTERVAL 7 DAY) -- (0 rows affected) (0.000040s)
 World |   Query    | QueryDatabase DELETE FROM player_event_merchant_purchase WHERE created_at < (NOW() - INTERVAL 7 DAY) -- (0 rows affected) (0.000024s)
 World |   Query    | QueryDatabase DELETE FROM player_event_merchant_sell WHERE created_at < (NOW() - INTERVAL 7 DAY) -- (0 rows affected) (0.000031s)
 World |   Query    | QueryDatabase DELETE FROM player_event_npc_handin WHERE created_at < (NOW() - INTERVAL 7 DAY) -- (0 rows affected) (0.000021s)
 World |   Query    | QueryDatabase DELETE FROM player_event_npc_handin_entries WHERE created_at < (NOW() - INTERVAL 7 DAY) -- (0 rows affected) (0.000020s)
 World |   Query    | QueryDatabase DELETE FROM player_event_trade WHERE created_at < (NOW() - INTERVAL 7 DAY) -- (0 rows affected) (0.000020s)
 World |   Query    | QueryDatabase DELETE FROM player_event_trade_entries WHERE created_at < (NOW() - INTERVAL 7 DAY) -- (0 rows affected) (0.000019s)
 World |   Query    | QueryDatabase DELETE FROM player_event_killed_npc WHERE created_at < (NOW() - INTERVAL 7 DAY) -- (0 rows affected) (0.000019s)
 World |   Query    | QueryDatabase DELETE FROM player_event_killed_named_npc WHERE created_at < (NOW() - INTERVAL 7 DAY) -- (0 rows affected) (0.000018s)
 World |   Query    | QueryDatabase DELETE FROM player_event_killed_raid_npc WHERE created_at < (NOW() - INTERVAL 7 DAY) -- (0 rows affected) (0.000018s)
 World |   Query    | QueryDatabase DELETE FROM player_event_speech WHERE created_at < (NOW() - INTERVAL 7 DAY) -- (0 rows affected) (0.000021s)
 World |   Query    | QueryDatabase DELETE FROM player_event_logs WHERE event_type_id IN (1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29, 30, 31, 32, 33, 34, 35, 36, 37, 38, 39, 40, 41, 42, 43, 44, 45, 46, 47, 48, 49, 50, 51, 52, 53, 54) AND created_at < (NOW() - INTERVAL 7 DAY) -- (0 rows affected) (0.000111s)

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants