diff --git a/src/interfaces/JournalInterface.ts b/src/interfaces/JournalInterface.ts deleted file mode 100644 index bed3654..0000000 --- a/src/interfaces/JournalInterface.ts +++ /dev/null @@ -1,359 +0,0 @@ -import type { Tail as TailType } from 'tail' -import type { autoScan, completeFsdJump, detailedScan, journalEntry, navRoute, planetScan } from '../@types/journalLines' - -const chokidar = require('chokidar') -const EventEmitter = require('node:events') -const fs = require('node:fs') -const { globSync } = require('glob') -import * as _ from 'lodash-es' -const os = require('node:os') -const path = require('node:path') -const { readFile } = require('node:fs/promises') -const reverseLineReader = require('reverse-line-reader') -const Tail = require('tail').Tail - -import { Body } from '../models/Body' -import { System } from '../models/System' - -// Set log() to console.log() so whenever I get around to setting up a log file, I don't have to -// search and replace all the console.log()'s. -const log = console.log.bind(console) - -export class JournalInterface extends EventEmitter { - journalDir: null|string - journalPattern: string - currentJournal: string|undefined - location: System - navRoute: System[] - - - constructor(isPackaged: boolean) { - super() - - this.journalDir = null - if (!isPackaged) { // Account for WSL during development - this.journalDir = "/mnt/c/Users/marle/Saved\ Games/Frontier\ Developments/Elite\ Dangerous/" - } else if (os.platform() === 'win32') { // Windows - this.journalDir = os.homedir() + '\\Saved Games\\Frontier Developments\\Elite Dangerous' - } else if (os.platform() === 'linux') { // Linux - this.journalDir = os.homedir() + '/.local/share/Steam/steamapps/compatdata/359320/pfx/drive_c/users/steamuser/Saved Games/Frontier Developments/Elite Dangerous/' - } else { - log(`ERROR: Journal files not found. OS: ${os.platform()}.`) - } - - this.journalPattern = this.journalDir + "Journal.*.log" - - this.currentJournal = this.getLatestJournal() - log(`New journal file found, now watching ${path.basename(this.currentJournal)}.`) - - this.navRoute = [] - - // LineReader seems to be async, so start async processes here. - this.location = new System('Unknown') - - log('JournalInterface initialized. Attempting to find current location.') - this.getCurrentLocation() - // -> getScannedBodies() - // --> getNavRoute() - } - - /* -------------------------------------------------------------------- getLatestJournal ---- */ - - // https://stackoverflow.com/questions/15696218/get-the-most-recent-file-in-a-directory-node-js - getLatestJournal(): string|undefined { - const journals = globSync(this.journalPattern) - - return _.maxBy(journals, file => fs.statSync(file).mtime) - } - - /* ------------------------------------------------------------------ getCurrentLocation ---- */ - - // Get current location on setup, so if app is restarted, user can pick up where they left off - // Rather than waiting til they jump to the next system to use the program again. - getCurrentLocation(): void { - reverseLineReader.eachLine(this.currentJournal, (raw: string, last: boolean) => { - if (raw) { // skip blank line at end of file - const line = JSON.parse(raw) - - if (line.event === 'FSDJump') { - this.location = new System(line) - log(`Current location set to ${this.location.name}.`) - this.emit('ENTERED_NEW_SYSTEM') - return false - } else if (last) { - log('Unable to find last hyperspace jump. Searching for last known location.') - return false - } - } - }).then(() => { - reverseLineReader.eachLine(this.currentJournal, (raw: string, last: boolean) => { - // TODO: figure out if we can avoid entering eachLine() altogether? realyyy wish - // it returned a promise :( - if (this.location.name !== 'Unknown') { - return false - } - - if (raw) { - const line = JSON.parse(raw) - - if (line.event === 'Location') { - this.location = new System(line) - log(`Current location set to ${this.location.name}.`) - this.emit('ENTERED_NEW_SYSTEM') - return false - } else if (last) { - log('WARNING: Unable to find last known location.') - return false - } - } - }).then(() => { - if (this.location.name !== 'Unknown') { - log('Attempting to find scanned bodies in current system.') - this.getScannedBodies() - } - }) - }) - } - - /* -------------------------------------------------------------------- getScannedBodies ---- */ - - // Look for all scanned bodies before last FSDJump, for same reasons as getCurrentLocation(). - getScannedBodies(): void { - let detailedScanLine: detailedScan|null = null - - reverseLineReader.eachLine(this.currentJournal, (raw: string, last: boolean) => { - - if (raw) { // Skip blank line at end of file. - const line: journalEntry = JSON.parse(raw) - - // Check if previous line was ScanType = Detailed, and handle that. - if (detailedScanLine !== null) { - if (line.event === 'SAAScanComplete') { - // This was a DSS, so add to list with DSS flag set to true. - this.location.bodies.push(new Body(detailedScanLine, true)) - } else { - // Else, check that the body hasn't already been added (by a DSS scan line). - let dupChecker = {'BodyName': detailedScanLine.BodyName, 'BodyID': detailedScanLine.BodyID} - let r = _.find(this.location.bodies, dupChecker) - - if (r === undefined) { - // Body was not already logged, so add to list. - this.location.bodies.push(new Body(detailedScanLine)) - } - } - - // Finally, clear the variable. - detailedScanLine = null - } - - // Now move on to evaluating the current line. - if (line.event === 'Scan' && 'ScanType' in line) { - // If ScanType = Detailed and body is not a star, save the line so we can check - // the one immediately above for event = SAAScanComplete, which indicates this - // was a DSS. - if (line.ScanType === 'Detailed' && !('StarType' in line)) { - detailedScanLine = (line as detailedScan) - - } else if ('StarType' in line) { // Save stars to bodies list. - this.location.bodies.push(new Body((line as autoScan|detailedScan))) - - } else if (line.ScanType === 'AutoScan') { // Save auto/discovery scan bodies. - // Check if planet, and then do the duplicate check (otherwise it's an - // astroid, as we've already accounted for stars). - if ('PlanetClass' in line) { - let dupChecker = {'BodyName': (line as planetScan<'AutoScan'>).BodyName, 'BodyID': (line as planetScan<'AutoScan'>).BodyID} - let r = _.find(this.location.bodies, dupChecker) - - if (r === undefined) { - this.location.bodies.push(new Body((line as autoScan))) - } - - } else { // Asteroids. - this.location.bodies.push(new Body((line as autoScan))) - } - } - } else if (line.event === 'FSDJump') { - // Stop evaluating once we reach the beginning of current system entries. - return false - } - } - }).then(() => { - log('Scanned bodies found.') - log('Reading current nav route.') - this.getNavRoute(true) - }) - } - - /* ---------------------------------------------------------------------- watchDirectory ---- */ - - // Set up journal directory watcher to catch new journal files as the game seems to sometimes - // make more than one journal per day. - // Also for instances where UTC day switches over mid-play session. - watchDirectory(): void { - const watcher = chokidar.watch(this.journalPattern, {usePolling: true, persistent: true}) - - watcher.on('add', () => this.currentJournal = this.getLatestJournal()) - - log('Watching journal folder for changes...') - } - - /* ----------------------------------------------------------------------- parseScanLine ---- */ - - // Parse and handle scan lines. - parseScanLine(line: autoScan|detailedScan, DSS: boolean = false) { - const dupChecker = {'BodyName': line.BodyName, 'BodyID': line.BodyID} - let body: Body|null = null - - // If it's a DSS scan, then we should have already added the body to the list. But we'll - // check to make sure. - if (DSS) { - // Using findIndex() rather than find() so we can edit the body if found. - // @ts-ignore since it doesn't understand dupChecker is a valid predicate. - let bodyIndex: number = _.findIndex(this.location.bodies, dupChecker) - - if (bodyIndex > -1) { // Body was found in list, so simply toggle the DSS flag. - body = (this.location.bodies[bodyIndex] as Body) - body.DSSDone = true - } else { // Body was missed on initial journal scan, so add it to the list. - body = new Body(line, true) - this.location.bodies.push(body) - } - - } else { // Otherwise it's an FSS or auto scan, and needs to be added to the list. - // Probably overkill, but do a duplicate check just in case. - let r = _.find(this.location.bodies, dupChecker) - - if (r === undefined) { - body = new Body(line) - this.location.bodies.push(body) - } - } - - log(`Scan detected. Body: ${line.BodyName}.`) - this.emit('BODY_SCANNED', body, DSS) - } - - /* ------------------------------------------------------------------------- getNavRoute ---- */ - - async getNavRoute(init: boolean = false) { - this.navRoute = [] // clear previous route, to catch overwritten routes - let routeFile: string|null = null - - try { - routeFile = await readFile(this.journalDir + 'NavRoute.json', { encoding: 'utf8' }) - } catch (err) { - log(`Error reading nav route file: ${err.message}.`) - } - - if (routeFile) { - const route: navRoute = JSON.parse(routeFile) - - // system -> skip - // CURRENT -> push = true; skip - // system -> push - - let push: boolean = false - route.Route.forEach((system) => { - if (!push && system.SystemAddress === this.location.SystemAddress) { - push = true - } - - if (push && system.SystemAddress !== this.location.SystemAddress) { - this.navRoute.push(new System(system)) - } - }) - - if (this.navRoute.length > 0) { - log('Nav route set.') - } else { - log('No nav route found.') - } - - if (init) { - this.emit('INIT_COMPLETE') - } - - this.emit('SET_NAV_ROUTE') - } - } - - /* ----------------------------------------------------------------------- handleFsdJump ---- */ - - handleFsdJump(line: completeFsdJump): void { - this.location = new System((line as completeFsdJump)) - log(`FSD Jump detected, current location updated to ${this.location.name}.`) - - if (this.navRoute.length > 0) { - _.remove(this.navRoute, (system) => { - return system.SystemAddress === this.location.SystemAddress - }) - } - - this.emit('ENTERED_NEW_SYSTEM') - } - - /* --------------------------------------------------------------------------- parseLine ---- */ - - // Parse and handle journal lines. - parseLine(raw: string) { - const line: journalEntry = JSON.parse(raw) - let DSSFlag: boolean = false - - switch (line.event) { - // Hyperspace jump started (3.. 2.. 1..) - case 'StartJump': { - if ('JumpType' in line && line.JumpType === 'Hyperspace') { - this.emit('ENTERING_WITCH_SPACE') - } - break - } - - // CMDR jumped to new system, so update current location. - case 'FSDJump': { - this.handleFsdJump((line as completeFsdJump)) - break - } - - // CMDR completed DSS scan, so set flag for when next line processes and we want to - // figure out what kind of scan occurred. - case 'SAAScanComplete': { - DSSFlag = true - break - } - - // A scan occurred, so let's hand that info off to the appropriate function and then - // reset the DSS flag. - case 'Scan': { - this.parseScanLine((line as autoScan|detailedScan), DSSFlag) - DSSFlag = false - break - } - - // CMDR set a new nav route. - case 'NavRoute': { - this.getNavRoute() - break - } - - // CMDR cleared the nav route. - case 'NavRouteClear': { - this.navRoute = [] - log('Nav route cleared.') - this.emit('SET_NAV_ROUTE') - break - } - } - } - - /* ------------------------------------------------------------------------ watchJournal ---- */ - - // Watch the journal for changes. - watchJournal(): void { - const tail: TailType = new Tail(this.currentJournal, {useWatchFile: true}) - - log(`Watching ${path.basename(this.currentJournal)}...`) - - tail.on('line', data => data ? this.parseLine(data) : undefined) - tail.on('error', err => log(`Tail error in JournalInterface.watchJournal(): ${err}`)) - } -} \ No newline at end of file diff --git a/src/models/Journal.ts b/src/models/Journal.ts index c1ab055..4ba0401 100644 --- a/src/models/Journal.ts +++ b/src/models/Journal.ts @@ -1,13 +1,13 @@ import type { Tail as TailType } from 'tail' import type { autoScan, completeFsdJump, detailedScan, journalEntry, navRoute, planetScan } from "../@types/journalLines" +const EventEmitter = require('node:events') import * as _ from 'lodash-es' const path = require('node:path') const { readFile } = require('node:fs/promises') const reverseLineReader = require('reverse-line-reader') const Tail = require('tail').Tail -import EventEmitter from "events" import { System } from "./System" import { Log } from "./Log" import { Body } from "./Body" @@ -176,7 +176,7 @@ export class Journal extends EventEmitter { let routeFile: string|null = null try { - const filePath: string = path.dirname(this.#path) + 'NavRoute.json' + const filePath: string = path.dirname(this.#path) + '/NavRoute.json' routeFile = await readFile(filePath, {encoding: 'utf8'}) } catch (err) { Log.write(`Error reading nav route file: ${err.message}.`) diff --git a/src/models/App.ts b/src/models/Safari.ts similarity index 88% rename from src/models/App.ts rename to src/models/Safari.ts index 46134b7..228fcd2 100644 --- a/src/models/App.ts +++ b/src/models/Safari.ts @@ -8,7 +8,7 @@ const path = require('node:path') import { Journal } from "./Journal" import { Log } from "./Log" -export class App { +export class Safari { #journalDir?: string #journalPattern?: string journal?: Journal @@ -49,10 +49,10 @@ export class App { /* --------------------------------------------------------------------- watchJournalDir ---- */ watchJournalDir(): void { - const watcher = chokidar.watch(this.#journalPattern, {usePolling: true, persistant: true}) - + const options = {usePolling: true, persistent: true, ignoreInitial: true} + const watcher = chokidar.watch(this.#journalPattern, options) + + watcher.on('ready', () => Log.write('Watching journal folder for changes...')) watcher.on('add', () => this.journal = this.#getLatestJournal()) - - Log.write('Watching journal folder for changes...') } } \ No newline at end of file diff --git a/src/renderer.js b/src/renderer.js index ef270f9..118e577 100644 --- a/src/renderer.js +++ b/src/renderer.js @@ -31,7 +31,7 @@ import './assets/index.css' import './icons/flaticon.css' const { app } = require('electron') -import { JournalInterface } from './interfaces/JournalInterface' +import { Safari } from './models/Safari' import { UI } from './models/UI' import { Body } from './models/Body' import { sep } from 'path' @@ -46,16 +46,15 @@ window.process.argv.forEach((item) => { /* ------------------------------------------------------------------------------- app setup ---- */ -const journal = new JournalInterface(isPackaged) +const safari = new Safari(isPackaged) +const journal = safari.journal -if (journal.journalDir === null) { +if (!journal) { // handle error } -journal.watchDirectory() -journal.watchJournal() - -const test = {name: 'Test', ID: 'TestID'} +safari.watchJournalDir() +journal.watch() /* ------------------------------------------------------------------------- build body list ---- */