Implemented Safari, Journal. RM JournalInterface

This commit is contained in:
punkfairie 2023-05-11 21:20:06 -07:00
parent 03ca5902b1
commit 93ef09e8c6
4 changed files with 13 additions and 373 deletions

View file

@ -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}`))
}
}

View file

@ -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}.`)

View file

@ -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...')
}
}

View file

@ -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 ---- */