A Configurable and Context Driven Project 1999 Log Parser with NCurses Interface for Linux
Best used with a launch script which maintains reasonable log file size
# Download
$ pipx install eqalert
# Start eqalert
$ eqalert
# Clone the repository
$ git clone https://github.com/mgeitz/eqalert.git
$ cd eqalert
# Build
$ docker compose build
# Start eqalert
$ docker compose run eqalert
# Install poetry
$ pipx install poetry==1.7.1
# Clone the repository
$ git clone https://github.com/mgeitz/eqalert.git
$ cd eqalert
# Install playsound
$ ./scripts/install-playsound.sh
# Retrieve dependencies
$ poetry update
$ poetry install
# Build
$ poetry build
# Start eqalert
$ poetry run eqalert
Note: If running through docker after installing and running on your host, update or regenerate
~/.eqa/config/settings.json
to reflect local container paths in/home/eqalert
Start things up
$ eqalert
You should now see ~/.eqa/
with the following structure
$HOME/.eqa
⎿ config/
characters.json
⎿ line-alerts/
ability-output.json
chat-received.json
chat-received-npc.json
chat-sent.json
combat.json
command-output.json
emotes.json
group-system-messages.json
loot-trade.json
other.json
pets.json
spell-general.json
spell-specific.json
system-messages.json
who.json
settings.json
zones.json
⎿ data/
⎿ encounters/
players.json
spell-casters.json
spell-items.json
spell-lines.json
spell-timers.json
⎿ timers/
⎿ log/
⎿ debug/
eqalert.log
⎿ sound/
tick.wav
tock.wav
Note: Some folders or files may only be created after enabling certain settings
Spot check these default paths generated in config/settings.json
"settings": {
"paths": {
"eqalert_log": "[$HOME/.eqa/]log/",
"data": "[$HOME/.eqa/]data/",
"everquest_logs": "[$HOME]/.wine/drive_c/Program Files/Sony/EverQuest/Logs/",
"everquest_files": "[$HOME]/.wine/drive_c/Program Files/Sony/EverQuest/",
"sound": "[$HOME/.eqa/]sound/",
"tmp_sound": "/tmp/eqa/sound/"
},
}
Press
0
to reload your configs or restart the program if any changes were made. Though generally, it's a good idea to stop eqalert before manually editing your config files.
EQ Alert will generate a file for spell timers in data/spell-timers.json
by default by parsing spells_us.txt
in your EverQuest directory.
This file will only regenerate if it is missing, malformed, or a newer spells_us.txt
file is present.
EQ Alert will generate/overwrite data/spell-casters.json
each version. This file contains which classes can cast a given buff which timers should be made for.
Many spells cast in EverQuest share the same log output lines. EQ Alert will generate/overwrite data/spell-lines.json
each version as a mapping of which possible spells a given output line could be associated to.
EQ Alert will generate/overwrite data/spell-items.json
each version as a mapping of which items a given spell could be cast by.
EQ Alert uses in-game /who
output to keep an up-to-date list of each seen players class, level, and guild organized by server in data/players.json
for alerting spell duration.
If the active character is changed or EQ Alert is stopped, all active timers are saved to a file in data/timers/
by the previously active character and server name. This file is consumed when the parser sets that character as active again, adding all non-expired timers back to the timer list.
When encounter parse saving is enabled, complete encounter reports are saved under data/encounters/[zone_name]/[date]/
as the time stamp of the encounter and the encounter target.
Note: WASD or arrow keys will work
Settings and options can be modified in config/settings.json
auto mob timer
: Create timer events after gaining experience for a duration based on the zone you are inauto mob timer delay
: Set a delay for the auto mob timer notification n seconds before the actual eventcharacter mention alert
: if enabled, speak any chat received line containing the active players nameconsider eval
: Speak "safe" or "danger" based on consider outputdebug mode
: Slows down parser performance and produces lots of file outputdetect character
: Automatically set parser to listen to the most recently active eqlogencounter parsing
: Parse encounter damageencounter parsing auto save
: Save verbose encounter parse results to a filemute
: Disable all text-to-speech outputpersist player data
: Save /who player output for spell timersraid mode auto set
: Auto-set raid context by zonespeech expand lingo
: When speaking a line, replace common EQ abbreviations with complete wordsspeech gtts lang
: The language (IETF language tag) to read the text in - gTTS documentation referencespeech gtts tld
: Top-level domain for the Google Translate host - gTTS documentation referencespeech local tts enabled
: If enabled, use the local TTS model to generate all speech. When disabled, use Google TTS - enabling this requires a restart to take effectspeech local tts model
: Coqui TTS model to use. A few good ones can be found here, otherwise check the full documentation - changing this requires a restart to take effectspell timer consolidate
: When a spell timer expires consolidate timers for the same spell in the next 3 seconds to one alertspell timer delay
: Set a delay for all spell timer notifications n seconds before the actual eventspell timer filter by list
: If enabled only create spell timers for spells in filter listspell timer filter guild only
: Filter all spell timer events so they are only for yourself or guild membersspell timer filter list
: List of spells, when true, to filter timers tospell timer filter yours only
: Filter all spell timer events to be only spells you castspell timer guess
: If there is moderate uncertainty in guessing a spell, go for itspell timer other
: Set spell timers for spells that land on other playersspell timer self
: Set spell timers for spells that land on yourselfspell timer zone drift
: If enabled add time between zoning to spell timers targetting yourselfYou can control some parser settings using /say
in-game. This is better suited for one monitor setups.
/say parser debug
- Toggle debug mode
/say parser raid
- Toggle raid mode
/say parser mute/unmute
- Toggle global mute/unmute
/say parser mute/unmute alert
- Toggle mute/unmute on all alert
alerts.
/say parser mute clear
- Clear all muted line types and players
Does not effect global mute
/say parser mute/unmute line
- Mute all sound alerts from a type of line
/say parser mute/unmute line player
- Mute all sound alerts of a given line type from a specific source
example: /say parser mute tell indef
/say parser mute/unmute speak
- Toggle mute/unmute on all speak
alerts.
example: /say parser mute engage
/say parser encounter
- Toggle encounter parsing
/say parser encounter clear
- Clear the encounter stack
/say parser encounter end
- Sometimes, your logs don't catch an encounter end. Use this command to fix that!
/say parser metronome [seconds]
- Set a tick/tock metronome with interval of [seconds]
/say parser metronome stop
- Stop the metronome
/say parser timer [seconds]
- Set a timer for [seconds], says "times up" when done
/say parser timer clear
- Clear all timers
/say parser timer respawn
- Create timers for the default current zone time after seeing an experience message. Timer response is "pop [zone name]"
/say parser timer respawn stop
- Stop creating mob respawn timers automatically
/say parser consider
- Toggle consider evaluation
/say parser hello
- Hello
/say parser what context
- Speak context state
/say parser what state
- Speak everything in the state object
/say parser where
- Where am I?
/say parser who [player_name]
- Speaks player level, class, and guild if known
Modify ~/.eqa/config/line-alerts/*.json
to customize alerts.
Due to how many line matches there are, the configuration for their reactions have been split into several json files under [$HOME/.eqa]/config/line-alerts/*.json
Anything matched by the parser not found in configuration is automatically added to [$HOME/.eqa]/config/line-alerts/other.json
Modify line_type
values to customize alerts accordingly.
Example configuration for a line type:
"line_type_name": {
"alert": {},
"reaction": false,
"sound": false
},
"alert"
: Alert for matching strings in alert
for the line type using the set sound"all"
: Alert for all lines of a given line type using the set soundfalse
: Disable alerting for this line type"afk"
: Alert only when afk"group"
: Alert when in a group and raiding"group_only"
: Alert only when grouped"raid"
: Alert when in a raid"solo"
: Alert when solo, grouped, and raiding"solo_group_only"
: Alert only when not raiding"solo_only"
: Alert only when solo"alert"
can be populated with key value pairs. The key here is any string you would like an alert for within that line type.
Alert for the word hey
when someone else /says
it:
"say": {
"alert": {
"hey": true
},
"reaction": "alert",
"sound": "hey"
},
Alert for the word run
when someone else /says
it and you are grouped or in a raid:
"say": {
"alert": {
"run": "group"
},
"reaction": "alert",
"sound": "oh god run"
},
Alert for a spell not taking hold only when grouped:
"spell_not_hold": {
"alert": {},
"reaction": "group_only",
"sound": true
},
Alert for the item Hand Made Backpack
when someone else /auctions
it and is selling:
"auction_wts": {
"alert": {
"Hand Made Backpack": true
},
"reaction": "alert",
"sound": "wow buy that"
},
More examples can be referenced in the default config
false
: Disable alerting for the string (does not negate line type reactions)true
: Alert for the string"afk"
: Alert only when afk"group"
: Alert when in a group and raiding"group_only"
: Alert only when grouped"raid"
: Alert when in a raid"solo"
: Alert when solo, grouped, and raiding"solo_group_only"
: Alert only when not raiding"solo_only"
: Alert only when solofalse
: Play no sound when an alert is raisedtrue
: When an alert is raised speak the entire lineAny other sound value (as a string) will be spoken as the audio trigger for that line type
This line type behaves the same as any specific line type configuration, but configuration here will be used against all log lines.
For example, the below configuration will alert if the word help
is found in any line while in a raid context, even if that line isn't matched to a type by the parser.
"all": {
"alert": {
"help": raid
},
"reaction": "alert",
"sound": "help is needed"
},
This can be helpful if you would like to alert for something not yet matched by the parser, though your contribution to a new line type match in the parser would also be welcome!
Zone data is stored in config/zones.json
Whether or not this zone is considered indoors. Currently does nothing.
false
: If enabled, auto-disable raid mode in this zonetrue
: If enabled, auto-enable raid mode in this zone#
: The value in seconds to set a timer for after defeating a mobNote: No support for zones with multiple default timers, stick with the manual timer command for those for now. For zones with tiered default timers, the shortest timer was set as the default. You can change this value to be any number in seconds you prefer.