mirror of
https://github.com/Ylianst/MeshCentral.git
synced 2025-01-13 07:53:20 -05:00
MCREC Player and Indexer now support indexes at the end of the file.
This commit is contained in:
parent
d94a15cd16
commit
eff70714dc
71
mcrec.js
71
mcrec.js
@ -35,15 +35,17 @@ function startEx(argv) {
|
|||||||
state.recFileSize = fs.statSync(infile).size;
|
state.recFileSize = fs.statSync(infile).size;
|
||||||
if (state.recFileSize < 32) { log("Invalid file: " + infile); return; }
|
if (state.recFileSize < 32) { log("Invalid file: " + infile); return; }
|
||||||
log("Processing file: " + infile + ", " + state.recFileSize + " bytes.");
|
log("Processing file: " + infile + ", " + state.recFileSize + " bytes.");
|
||||||
state.recFile = fs.openSync(infile, 'r');
|
state.recFile = fs.openSync(infile, 'r+');
|
||||||
state.indexTime = 10; // Interval between indexes in seconds
|
state.indexTime = 10; // Interval between indexes in seconds
|
||||||
state.lastIndex = 0; // Last time an index was writen in seconds
|
state.lastIndex = 0; // Last time an index was writen in seconds
|
||||||
state.indexes = [];
|
state.indexes = [];
|
||||||
state.width = 0;
|
state.width = 0;
|
||||||
state.height = 0;
|
state.height = 0;
|
||||||
state.basePtr = null;
|
state.basePtr = null;
|
||||||
readLastBlock(state, function (state, result) {
|
readLastBlock(state, function (state, result, time, extras) {
|
||||||
if (result == false) { log("Invalid file: " + infile); return; }
|
if (result == false) { log("Invalid file: " + infile); return; }
|
||||||
|
if (extras != null) { log("File already indexed: " + infile); return; }
|
||||||
|
state.lastTimeStamp = time;
|
||||||
readNextBlock(state, processBlock);
|
readNextBlock(state, processBlock);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@ -62,7 +64,7 @@ function createIndex(state, ptr) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function processBlock(state, block) {
|
function processBlock(state, block) {
|
||||||
if (block == null) { writeIndexedFile(state, function () { log("Done."); }); return; }
|
if (block == null) { writeIndex(state, function () { log("Done."); }); return; }
|
||||||
var elapseMilliSeconds = 0;
|
var elapseMilliSeconds = 0;
|
||||||
if (state.startTime != null) { elapseMilliSeconds = (block.time - state.startTime); }
|
if (state.startTime != null) { elapseMilliSeconds = (block.time - state.startTime); }
|
||||||
var flagBinary = (block.flags & 1) != 0;
|
var flagBinary = (block.flags & 1) != 0;
|
||||||
@ -164,29 +166,21 @@ function processBlock(state, block) {
|
|||||||
readNextBlock(state, processBlock);
|
readNextBlock(state, processBlock);
|
||||||
}
|
}
|
||||||
|
|
||||||
function writeIndexedFile(state, func) {
|
function writeIndex(state, func) {
|
||||||
var outfile = state.recFileName;
|
// Add the new indexes in extra metadata at the end of the file.
|
||||||
if (outfile.endsWith('.mcrec')) { outfile = outfile.substring(0, outfile.length - 6) + '-ndx.mcrec'; } else { outfile += '-ndx.mcrec'; }
|
var extraMetadata = {};
|
||||||
if (fs.existsSync(outfile)) { log("File already exists: " + outfile); return; }
|
extraMetadata.indexInterval = state.indexTime;
|
||||||
log("Writing file: " + outfile);
|
extraMetadata.indexStartTime = state.startTime;
|
||||||
state.writeFile = fs.openSync(outfile, 'w');
|
extraMetadata.indexes = state.indexes;
|
||||||
state.metadata.indexInterval = state.indexTime;
|
recordingEntry(state.recFile, 4, 0, state.lastTimeStamp, JSON.stringify(extraMetadata), function (state) {
|
||||||
state.metadata.indexStartTime = state.startTime;
|
recordingEntry(state.recFile, 3, 0, state.recFileSize - 32, 'MeshCentralMCNDX', function (state) {
|
||||||
state.metadata.indexes = state.indexes;
|
func(state);
|
||||||
var firstBlock = JSON.stringify(state.metadata);
|
}, state);
|
||||||
recordingEntry(state.writeFile, 1, state.metadataFlags, state.metadataTime, firstBlock, function (state) {
|
}, state, state.recFileSize - 32);
|
||||||
var len = 0, buffer = Buffer.alloc(4096), ptr = state.dataStartPtr;
|
|
||||||
while (ptr < state.recFileSize) {
|
|
||||||
len = fs.readSync(state.recFile, buffer, 0, 4096, ptr);
|
|
||||||
fs.writeSync(state.writeFile, buffer, 0, len);
|
|
||||||
ptr += len;
|
|
||||||
}
|
|
||||||
func(state);
|
|
||||||
}, state);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Record a new entry in a recording log
|
// Record a new entry in a recording log
|
||||||
function recordingEntry(fd, type, flags, time, data, func, tag) {
|
function recordingEntry(fd, type, flags, time, data, func, tag, position) {
|
||||||
try {
|
try {
|
||||||
if (typeof data == 'string') {
|
if (typeof data == 'string') {
|
||||||
// String write
|
// String write
|
||||||
@ -196,7 +190,11 @@ function recordingEntry(fd, type, flags, time, data, func, tag) {
|
|||||||
header.writeInt32BE(blockData.length, 4); // Size
|
header.writeInt32BE(blockData.length, 4); // Size
|
||||||
header.writeIntBE(time, 10, 6); // Time
|
header.writeIntBE(time, 10, 6); // Time
|
||||||
var block = Buffer.concat([header, blockData]);
|
var block = Buffer.concat([header, blockData]);
|
||||||
fs.write(fd, block, 0, block.length, function () { func(tag); });
|
if (typeof position == 'number') {
|
||||||
|
fs.write(fd, block, 0, block.length, position, function () { func(tag); });
|
||||||
|
} else {
|
||||||
|
fs.write(fd, block, 0, block.length, function () { func(tag); });
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
// Binary write
|
// Binary write
|
||||||
var header = Buffer.alloc(16); // Header: Type (2) + Flags (2) + Size(4) + Time(8)
|
var header = Buffer.alloc(16); // Header: Type (2) + Flags (2) + Size(4) + Time(8)
|
||||||
@ -205,7 +203,11 @@ function recordingEntry(fd, type, flags, time, data, func, tag) {
|
|||||||
header.writeInt32BE(data.length, 4); // Size
|
header.writeInt32BE(data.length, 4); // Size
|
||||||
header.writeIntBE(time, 10, 6); // Time
|
header.writeIntBE(time, 10, 6); // Time
|
||||||
var block = Buffer.concat([header, data]);
|
var block = Buffer.concat([header, data]);
|
||||||
fs.write(fd, block, 0, block.length, function () { func(tag); });
|
if (typeof position == 'number') {
|
||||||
|
fs.write(fd, block, 0, block.length, position, function () { func(tag); });
|
||||||
|
} else {
|
||||||
|
fs.write(fd, block, 0, block.length, function () { func(tag); });
|
||||||
|
}
|
||||||
}
|
}
|
||||||
} catch (ex) { console.log(ex); func(state, tag); }
|
} catch (ex) { console.log(ex); func(state, tag); }
|
||||||
}
|
}
|
||||||
@ -218,7 +220,24 @@ function readLastBlock(state, func) {
|
|||||||
var size = buf.readInt32BE(4);
|
var size = buf.readInt32BE(4);
|
||||||
var time = (buf.readInt32BE(8) << 32) + buf.readInt32BE(12);
|
var time = (buf.readInt32BE(8) << 32) + buf.readInt32BE(12);
|
||||||
var magic = buf.toString('utf8', 16, 32);
|
var magic = buf.toString('utf8', 16, 32);
|
||||||
func(state, (type == 3) && (size == 16) && (magic == 'MeshCentralMCREC'));
|
if ((type == 3) && (size == 16) && (magic == 'MeshCentralMCNDX')) {
|
||||||
|
// Extra metadata present, lets read it.
|
||||||
|
extraMetadata = null;
|
||||||
|
var buf2 = Buffer.alloc(16);
|
||||||
|
fs.read(state.recFile, buf2, 0, 16, time, function (err, bytesRead, buf2) {
|
||||||
|
var xtype = buf2.readInt16BE(0);
|
||||||
|
var xflags = buf2.readInt16BE(2);
|
||||||
|
var xsize = buf2.readInt32BE(4);
|
||||||
|
var xtime = (buf2.readInt32BE(8) << 32) + buf.readInt32BE(12);
|
||||||
|
var buf3 = Buffer.alloc(xsize);
|
||||||
|
fs.read(state.recFile, buf3, 0, xsize, time + 16, function (err, bytesRead, buf3) {
|
||||||
|
func(state, true, xtime, JSON.parse(buf3.toString()));
|
||||||
|
});
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
// No extra metadata or fail
|
||||||
|
func(state, (type == 3) && (size == 16) && (magic == 'MeshCentralMCREC'), time, null);
|
||||||
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -10978,6 +10978,7 @@
|
|||||||
//
|
//
|
||||||
|
|
||||||
// Converts string to UTF8 byte array, polyfill for IE.
|
// Converts string to UTF8 byte array, polyfill for IE.
|
||||||
|
// Following method is code from Mozilla: https://developer.mozilla.org/en-US/docs/Web/API/TextEncoder
|
||||||
if (typeof TextEncoder === 'undefined') {
|
if (typeof TextEncoder === 'undefined') {
|
||||||
window.TextEncoder=function TextEncoder(){};
|
window.TextEncoder=function TextEncoder(){};
|
||||||
TextEncoder.prototype.encode = function encode(str) {
|
TextEncoder.prototype.encode = function encode(str) {
|
||||||
|
@ -28,9 +28,6 @@
|
|||||||
<span id="deskstatus"></span>
|
<span id="deskstatus"></span>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div id=deskarea2 style="">
|
|
||||||
<div class="areaProgress" style="cursor:pointer" onclick="progressBarSeek(event)"><div id="progressbar" style="height:6px;cursor:pointer"></div></div>
|
|
||||||
</div>
|
|
||||||
<div id=deskarea3x style="max-height:calc(100vh - 58px);height:calc(100vh - 58px);" onclick="togglePause()">
|
<div id=deskarea3x style="max-height:calc(100vh - 58px);height:calc(100vh - 58px);" onclick="togglePause()">
|
||||||
<div id="bigok" style="display:none;left:calc((100vh / 2))"><b>✓</b></div>
|
<div id="bigok" style="display:none;left:calc((100vh / 2))"><b>✓</b></div>
|
||||||
<div id="bigfail" style="display:none;left:calc((100vh / 2))"><b>✗</b></div>
|
<div id="bigfail" style="display:none;left:calc((100vh / 2))"><b>✗</b></div>
|
||||||
@ -43,6 +40,9 @@
|
|||||||
</div>
|
</div>
|
||||||
<div id=p11DeskConsoleMsg style="display:none;cursor:pointer;position:absolute;left:30px;top:17px;color:yellow;background-color:rgba(0,0,0,0.6);padding:10px;border-radius:5px" onclick=clearConsoleMsg()></div>
|
<div id=p11DeskConsoleMsg style="display:none;cursor:pointer;position:absolute;left:30px;top:17px;color:yellow;background-color:rgba(0,0,0,0.6);padding:10px;border-radius:5px" onclick=clearConsoleMsg()></div>
|
||||||
</div>
|
</div>
|
||||||
|
<div id=deskarea2 style="">
|
||||||
|
<div class="areaProgress" style="cursor:pointer" onclick="progressBarSeek(event)"><div id="progressbar" style="height:6px;cursor:pointer"></div></div>
|
||||||
|
</div>
|
||||||
<div id=deskarea4 class="areaFoot">
|
<div id=deskarea4 class="areaFoot">
|
||||||
<div class="toright2">
|
<div class="toright2">
|
||||||
<div id="timespan" style="padding-top:4px;padding-right:4px">00:00:00</div>
|
<div id="timespan" style="padding-top:4px;padding-right:4px">00:00:00</div>
|
||||||
@ -165,7 +165,24 @@
|
|||||||
var flags = ReadShort(this.result, 2);
|
var flags = ReadShort(this.result, 2);
|
||||||
var size = ReadInt(this.result, 4);
|
var size = ReadInt(this.result, 4);
|
||||||
var time = (ReadInt(this.result, 8) << 32) + ReadInt(this.result, 12);
|
var time = (ReadInt(this.result, 8) << 32) + ReadInt(this.result, 12);
|
||||||
if ((type == 3) && (size == 16) && (this.result.substring(16, 32) == 'MeshCentralMCREC')) { func(type, flags, time); } else { func(-1); }
|
var magic = this.result.substring(16, 32);
|
||||||
|
if ((type == 3) && (size == 16) && (magic == 'MeshCentralMCNDX')) {
|
||||||
|
// Extra metadata present, lets read it.
|
||||||
|
var fr2 = new FileReader();
|
||||||
|
fr2.onload = function () {
|
||||||
|
var xtype = ReadShort(this.result, 0);
|
||||||
|
var xflags = ReadShort(this.result, 2);
|
||||||
|
var xsize = ReadInt(this.result, 4);
|
||||||
|
var xtime = (ReadInt(this.result, 8) << 32) + ReadInt(this.result, 12);
|
||||||
|
var extras = JSON.parse(this.result.substring(16));
|
||||||
|
func(type, flags, xtime, extras); // Include extra metadata
|
||||||
|
}
|
||||||
|
fr2.readAsBinaryString(recFile.slice(time, recFile.size - 32));
|
||||||
|
} else if ((type == 3) && (size == 16) && (magic == 'MeshCentralMCREC')) {
|
||||||
|
func(type, flags, time); // No extra metadata
|
||||||
|
} else {
|
||||||
|
func(-1); // Fail
|
||||||
|
}
|
||||||
};
|
};
|
||||||
fr.readAsBinaryString(recFile.slice(recFile.size - 32, recFile.size));
|
fr.readAsBinaryString(recFile.slice(recFile.size - 32, recFile.size));
|
||||||
}
|
}
|
||||||
@ -183,6 +200,7 @@
|
|||||||
if ((type != 1) || (flags != 0)) { cleanup(); return; }
|
if ((type != 1) || (flags != 0)) { cleanup(); return; }
|
||||||
try { recFileMetadata = JSON.parse(data) } catch (ex) { cleanup(); return; }
|
try { recFileMetadata = JSON.parse(data) } catch (ex) { cleanup(); return; }
|
||||||
if ((recFileMetadata == null) || (recFileMetadata.magic != 'MeshCentralRelaySession') || (recFileMetadata.ver != 1)) { cleanup(); return; }
|
if ((recFileMetadata == null) || (recFileMetadata.magic != 'MeshCentralRelaySession') || (recFileMetadata.ver != 1)) { cleanup(); return; }
|
||||||
|
if (recFileExtras) { for (var i in recFileExtras) { recFileMetadata[i] = recFileExtras[i]; } }
|
||||||
var x = '';
|
var x = '';
|
||||||
x += addInfo("Time", recFileMetadata.time);
|
x += addInfo("Time", recFileMetadata.time);
|
||||||
if (recFileEndTime != 0) { var secs = Math.floor((recFileEndTime - time) / 1000); x += addInfo("Duration", format("{0} second{1}", secs, (secs > 1) ? 's' : '')); }
|
if (recFileEndTime != 0) { var secs = Math.floor((recFileEndTime - time) / 1000); x += addInfo("Duration", format("{0} second{1}", secs, (secs > 1) ? 's' : '')); }
|
||||||
@ -355,8 +373,17 @@
|
|||||||
cleanup();
|
cleanup();
|
||||||
recFile = files[0];
|
recFile = files[0];
|
||||||
recFilePtr = 0;
|
recFilePtr = 0;
|
||||||
readNextBlock(processFirstBlock);
|
readLastBlock(function (type, flags, time, extras) {
|
||||||
readLastBlock(function (type, flags, time) { if (type == 3) { recFileEndTime = time; } else { recFileEndTime = 0; } });
|
if (type == 3) {
|
||||||
|
// File is ok
|
||||||
|
recFileEndTime = time;
|
||||||
|
recFileExtras = extras;
|
||||||
|
readNextBlock(processFirstBlock);
|
||||||
|
} else {
|
||||||
|
// This is not a good file
|
||||||
|
recFileEndTime = 0;
|
||||||
|
}
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
var dragtimer = null;
|
var dragtimer = null;
|
||||||
@ -399,7 +426,17 @@
|
|||||||
recFile = files[0];
|
recFile = files[0];
|
||||||
recFilePtr = 0;
|
recFilePtr = 0;
|
||||||
readNextBlock(processFirstBlock);
|
readNextBlock(processFirstBlock);
|
||||||
readLastBlock(function (type, flags, time) { if (type == 3) { recFileEndTime = time; } else { recFileEndTime = 0; } });
|
readLastBlock(function (type, flags, time) {
|
||||||
|
if (type == 3) {
|
||||||
|
// File is ok
|
||||||
|
recFileEndTime = time;
|
||||||
|
recFileExtras = extras;
|
||||||
|
readNextBlock(processFirstBlock);
|
||||||
|
} else {
|
||||||
|
// This is not a good file
|
||||||
|
recFileEndTime = 0;
|
||||||
|
}
|
||||||
|
});
|
||||||
Q('OpenFileButton').blur();
|
Q('OpenFileButton').blur();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user