2023-12-24 17:44:02 +11:00
|
|
|
const PAGE_LENGTH = 4; // The size of our page tag as stored in the msgbase for echomail/netmail
|
|
|
|
const PAGE_LAST_KEY = 'last_page'; // Last page which has the latest message
|
|
|
|
const MAX_PAGE_NUM = 9999; // Maximum page number. @todo Can this be changed to '9'.repeat(PAGE_LENGTH)?
|
2022-04-20 21:19:01 +10:00
|
|
|
|
2022-04-18 21:27:25 +10:00
|
|
|
// Our message bases
|
|
|
|
function MsgAreas() {
|
|
|
|
'use strict';
|
|
|
|
|
|
|
|
this.areas = [];
|
|
|
|
this.areas_excluded = [];
|
|
|
|
var zone_id;
|
|
|
|
var zone_name;
|
|
|
|
var ma;
|
|
|
|
|
|
|
|
for(var g in msg_area.grp_list) {
|
|
|
|
if (msg_area.grp_list[g].name.indexOf(':') !== -1) {
|
|
|
|
zone_id = msg_area.grp_list[g].name.split(':')[0];
|
|
|
|
zone_name = msg_area.grp_list[g].name.split(':')[1];
|
|
|
|
|
|
|
|
for (var a in msg_area.grp_list[g].sub_list) {
|
|
|
|
if (msg_area.grp_list[g].sub_list[a].name.indexOf(':') !== -1) {
|
|
|
|
ma = new MsgArea();
|
|
|
|
ma.zone_id = zone_id;
|
|
|
|
ma.zone_name = zone_name;
|
|
|
|
ma.area_id = msg_area.grp_list[g].sub_list[a].name.split(':')[0];
|
|
|
|
ma.area_name = msg_area.grp_list[g].sub_list[a].name.split(':')[1];
|
|
|
|
ma.code = msg_area.grp_list[g].sub_list[a].code;
|
|
|
|
|
|
|
|
this.areas.push(ma);
|
|
|
|
} else {
|
|
|
|
this.areas_excluded.push(zone_name+':'+msg_area.grp_list[g].sub_list[a].name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
} else {
|
|
|
|
zone_name = msg_area.grp_list[g].name;
|
|
|
|
|
|
|
|
for (var a in msg_area.grp_list[g].sub_list) {
|
|
|
|
this.areas_excluded.push(zone_name+':'+msg_area.grp_list[g].sub_list[a].name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-12-24 17:44:02 +11:00
|
|
|
|
|
|
|
Object.defineProperty(this,'list',{
|
|
|
|
get: function() {
|
|
|
|
writeln('Areas that we are NOT managing mail:'+this.areas_excluded.length);
|
|
|
|
writeln('Areas that we ARE managing mail:'+this.areas.length);
|
|
|
|
|
|
|
|
for(var x in this.areas_excluded) {
|
2024-01-01 22:46:53 +11:00
|
|
|
writeln(x+':'+((this.areas_excluded[x].area_name === undefined)
|
|
|
|
? this.areas_excluded[x]
|
|
|
|
: JSON.stringify(this.areas_excluded[x])));
|
2023-12-24 17:44:02 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
2022-04-18 21:27:25 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
function MsgArea() {
|
|
|
|
this.zone_id = undefined;
|
|
|
|
this.zone_name = undefined;
|
|
|
|
this.area_id = undefined;
|
|
|
|
this.area_name = undefined;
|
|
|
|
this.msgbase = undefined;
|
|
|
|
this.headers = undefined;
|
|
|
|
this.tagged_list = undefined;
|
|
|
|
this.untagged_list = undefined;
|
2022-04-22 22:42:20 +10:00
|
|
|
this.grp_number = undefined;
|
|
|
|
this.subnum = undefined;
|
2022-04-18 21:27:25 +10:00
|
|
|
|
2022-05-12 22:27:06 +10:00
|
|
|
/**
|
|
|
|
* Build a MsgArea once we are given the code
|
|
|
|
*/
|
2022-04-18 21:27:25 +10:00
|
|
|
Object.defineProperty(this,'code',{
|
|
|
|
set: function(code) {
|
|
|
|
this.msgbase = new MsgBase(code);
|
|
|
|
|
2022-05-13 23:32:22 +10:00
|
|
|
try {
|
|
|
|
if (this.msgbase.open()) {
|
2022-12-15 23:28:25 +11:00
|
|
|
headers = this.msgbase.get_all_msg_headers(false,false) || [];
|
|
|
|
|
|
|
|
// Just take the last MAX_MESSAGES
|
|
|
|
this.headers = Object.keys(headers).slice(-(MAX_PAGE_NUM+1)).map(function(key) { return headers[key]; });
|
|
|
|
headers = undefined;
|
|
|
|
|
2022-05-13 23:32:22 +10:00
|
|
|
this.msgbase.close();
|
|
|
|
|
|
|
|
} else {
|
|
|
|
log(LOG_ERROR,code+' cannot be opened:'+this.msgbase.error);
|
|
|
|
this.headers = [];
|
|
|
|
}
|
2022-04-18 21:27:25 +10:00
|
|
|
|
2022-05-13 23:32:22 +10:00
|
|
|
} catch (e) {
|
|
|
|
log(LOG_ERROR,code+' cannot be opened:'+e.message);
|
|
|
|
this.headers = [];
|
|
|
|
}
|
2022-04-18 21:27:25 +10:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
// Get Area's full name
|
|
|
|
Object.defineProperty(this,'full_name',{
|
|
|
|
get: function() {
|
|
|
|
return this.zone_name+':'+this.area_name;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
// Total tagged messages
|
|
|
|
Object.defineProperty(this,'list_tagged',{
|
|
|
|
get: function() {
|
|
|
|
if (this.tagged_list === undefined) {
|
|
|
|
this.tagged_list = [];
|
|
|
|
|
|
|
|
if (! this.headers)
|
|
|
|
return this.tagged_list;
|
|
|
|
|
|
|
|
for(var x in this.headers) {
|
|
|
|
if (this.headers[x].tags && (this.headers[x].tags.length === PAGE_LENGTH)) {
|
|
|
|
this.tagged_list.push(this.headers[x]);
|
|
|
|
write(); // @todo This is needed for this to work?
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return this.tagged_list;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
// List untagged messages
|
|
|
|
Object.defineProperty(this,'list_untagged',{
|
|
|
|
get: function() {
|
|
|
|
if (this.untagged_list === undefined) {
|
|
|
|
this.untagged_list = [];
|
|
|
|
|
|
|
|
if (! this.headers)
|
|
|
|
return this.untagged_list;
|
|
|
|
|
|
|
|
for(var x in this.headers) {
|
|
|
|
if ((! this.headers[x].tags) || (this.headers[x].tags.length !== PAGE_LENGTH)) {
|
|
|
|
this.untagged_list.push(this.headers[x]);
|
|
|
|
write(); // @todo This is needed for this to work?
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-24 22:36:47 +10:00
|
|
|
return this.untagged_list;
|
2022-04-18 21:27:25 +10:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
// Get Next page number
|
|
|
|
Object.defineProperty(this,'page_next',{
|
|
|
|
get: function() {
|
|
|
|
var f = new File(file_cfgname(system.mods_dir,'ansitex/ctrl/videotex.ini'));
|
|
|
|
if (! f.open('r')) {
|
|
|
|
writeln('Unable to open ini file');
|
|
|
|
exit(2);
|
|
|
|
}
|
|
|
|
|
2022-04-26 21:54:34 +10:00
|
|
|
var page = f.iniGetValue('prefix:'+this.page_prefix,PAGE_LAST_KEY)
|
2022-04-18 21:27:25 +10:00
|
|
|
f.close();
|
|
|
|
|
|
|
|
return page ? page : '0'.repeat(PAGE_LENGTH);
|
|
|
|
},
|
|
|
|
|
|
|
|
set: function(page) {
|
|
|
|
var f = new File(file_cfgname(system.mods_dir,'ansitex/ctrl/videotex.ini'));
|
|
|
|
if (! f.open('r+')) {
|
|
|
|
writeln('Unable to open ini file');
|
|
|
|
exit(2);
|
|
|
|
}
|
|
|
|
|
2022-04-26 21:54:34 +10:00
|
|
|
f.iniSetValue('prefix:'+this.page_prefix,PAGE_LAST_KEY,(''+page).padStart(4,'0'));
|
2022-04-18 21:27:25 +10:00
|
|
|
f.close();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
// Our page prefix for this msg area
|
|
|
|
Object.defineProperty(this,'page_prefix',{
|
|
|
|
get: function() {
|
|
|
|
return ''+this.zone_id+this.area_id;
|
|
|
|
},
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2022-05-03 21:10:09 +10:00
|
|
|
/**
|
2022-12-15 23:28:25 +11:00
|
|
|
* Unread messages [1..]
|
|
|
|
* Array key 0 returns the last read message
|
2022-05-06 15:32:35 +10:00
|
|
|
*
|
|
|
|
* @returns {*[]}
|
2022-05-03 21:10:09 +10:00
|
|
|
*/
|
|
|
|
MsgArea.prototype.newMsgs = function() {
|
|
|
|
var msgs = [];
|
|
|
|
var stats = this.getUserStats();
|
2022-05-12 22:27:06 +10:00
|
|
|
//log(LOG_DEBUG,'Users last_read pointer: '+JSON.stringify(stats.last_read));
|
2022-05-03 21:10:09 +10:00
|
|
|
|
2022-05-12 22:27:06 +10:00
|
|
|
for(var x in this.list_tagged) {
|
2022-05-06 15:32:35 +10:00
|
|
|
// Advance past our last scan_ptr
|
2022-05-12 22:27:06 +10:00
|
|
|
if (this.list_tagged[x].number <= stats.last_read)
|
2022-05-06 15:32:35 +10:00
|
|
|
continue;
|
|
|
|
|
2022-05-12 22:27:06 +10:00
|
|
|
msgs.push(this.list_tagged[x]);
|
2022-05-06 15:32:35 +10:00
|
|
|
|
|
|
|
write(); // @todo This is needed for this to work?
|
|
|
|
}
|
|
|
|
|
|
|
|
return msgs;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* New Messages for the logged in user
|
|
|
|
*/
|
|
|
|
MsgArea.prototype.newMsgsToMe = function() {
|
|
|
|
var msgs = [];
|
|
|
|
var stats = this.getUserStats();
|
2022-12-15 23:28:25 +11:00
|
|
|
var last = null;
|
2022-05-12 22:27:06 +10:00
|
|
|
//log(LOG_DEBUG,'Users scan_ptr pointer: '+JSON.stringify(stats.scan_ptr));
|
2022-05-06 15:32:35 +10:00
|
|
|
|
2022-05-12 22:27:06 +10:00
|
|
|
for(var x in this.list_tagged) {
|
2022-05-03 21:10:09 +10:00
|
|
|
// Advance past our last scan_ptr
|
2022-12-15 23:28:25 +11:00
|
|
|
if (this.list_tagged[x].number <= stats.scan_ptr) {
|
|
|
|
if ((this.list_tagged[x].to === user.name) || (this.list_tagged[x].to === user.alias))
|
|
|
|
last = x;
|
|
|
|
|
2022-05-03 21:10:09 +10:00
|
|
|
continue;
|
2022-12-15 23:28:25 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
// Add our previous to me message
|
|
|
|
if (msgs.length === 0)
|
|
|
|
msgs.push(last !== null ? this.list_tagged[last] : []);
|
2022-05-03 21:10:09 +10:00
|
|
|
|
2022-05-12 22:27:06 +10:00
|
|
|
if ((this.list_tagged[x].to === user.name) || (this.list_tagged[x].to === user.alias))
|
|
|
|
msgs.push(this.list_tagged[x]);
|
2022-05-03 21:10:09 +10:00
|
|
|
|
|
|
|
write(); // @todo This is needed for this to work?
|
|
|
|
}
|
|
|
|
|
|
|
|
return msgs;
|
|
|
|
}
|
|
|
|
|
2022-04-18 21:27:25 +10:00
|
|
|
/**
|
|
|
|
* Get a specific message with a tag
|
|
|
|
*/
|
|
|
|
MsgArea.prototype.getMessage = function(page) {
|
|
|
|
var msg = undefined;
|
|
|
|
|
2022-05-12 22:27:06 +10:00
|
|
|
for(var x in this.list_tagged) {
|
|
|
|
if (this.list_tagged[x].tags === page) {
|
|
|
|
msg = this.list_tagged[x];
|
2022-04-18 21:27:25 +10:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
write(); // @todo This is needed for this to work?
|
|
|
|
}
|
|
|
|
|
|
|
|
if (! msg)
|
|
|
|
return undefined;
|
|
|
|
|
2022-04-20 21:19:01 +10:00
|
|
|
if (! this.msgbase.open()) {
|
|
|
|
writeln(code+' cannot be opened:'+this.msgbase.error);
|
|
|
|
return undefined;
|
|
|
|
}
|
|
|
|
|
2022-04-22 22:42:20 +10:00
|
|
|
msg.grp_number = this.msgbase.cfg.grp_number;
|
|
|
|
var cfg = this.msgbase.cfg;
|
|
|
|
msg.subnum = msg_area.grp_list[cfg.grp_number].sub_list.filter(function(x) { return x.number === cfg.number; }).pop().index;
|
2022-04-18 21:27:25 +10:00
|
|
|
msg.content = this.msgbase.get_msg_body(false,msg.number,false,false,true,true);
|
|
|
|
this.msgbase.close();
|
|
|
|
|
|
|
|
return msg;
|
|
|
|
}
|
|
|
|
|
2022-05-03 21:10:09 +10:00
|
|
|
/**
|
|
|
|
* Get a message page by pointer
|
|
|
|
*
|
2022-05-06 20:14:58 +10:00
|
|
|
* @param number
|
2022-05-03 21:10:09 +10:00
|
|
|
* @returns {string}
|
|
|
|
*/
|
2022-05-06 20:14:58 +10:00
|
|
|
MsgArea.prototype.getMessagePage = function(number) {
|
|
|
|
log(LOG_DEBUG,'Get Message Page with number ['+number+']');
|
|
|
|
|
|
|
|
var r;
|
2022-05-03 21:10:09 +10:00
|
|
|
|
2022-05-12 22:27:06 +10:00
|
|
|
for (var x in this.list_tagged) {
|
|
|
|
if (this.list_tagged[x].number === number) {
|
|
|
|
r = this.list_tagged[x];
|
|
|
|
|
2022-05-06 20:14:58 +10:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2022-05-03 21:10:09 +10:00
|
|
|
|
2022-05-06 20:14:58 +10:00
|
|
|
if (! r || r.tags === undefined)
|
2022-05-03 21:10:09 +10:00
|
|
|
return null;
|
|
|
|
|
2022-05-06 20:14:58 +10:00
|
|
|
return '1'+this.zone_id+this.area_id+r.tags;
|
2022-05-03 21:10:09 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
MsgArea.prototype.getUserStats = function() {
|
2022-05-13 23:32:22 +10:00
|
|
|
return this.msgbase.cfg ? msg_area.grp_list[this.msgbase.cfg.grp_number].sub_list[msg_area.sub[this.msgbase.cfg.code].index] : [];
|
2022-05-03 21:10:09 +10:00
|
|
|
}
|
|
|
|
|
2022-12-15 23:28:25 +11:00
|
|
|
MsgArea.prototype.MessageNext = function(page) {
|
|
|
|
var x = null;
|
|
|
|
|
|
|
|
if (! page)
|
|
|
|
return undefined;
|
|
|
|
|
|
|
|
var msgid = page.substr(7,4);
|
|
|
|
|
|
|
|
for(x in this.list_tagged) {
|
|
|
|
if (this.list_tagged[x].tags === msgid) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
write(); // @todo This is needed for this to work?
|
|
|
|
}
|
|
|
|
|
|
|
|
//log(LOG_DEBUG,'- Next Message is:'+JSON.stringify(this.list_tagged[(parseInt(x)+1)])+', msgid:'+msgid+', page:'+page+', x:'+x);
|
|
|
|
|
|
|
|
/*
|
|
|
|
= Our next message is either
|
|
|
|
+ x+1 if x < this.list_tagged.length
|
|
|
|
+ x=0 if x == this.list_tagged.length (-1)
|
|
|
|
+ null if this.list_tagged.length == null; (thus no messages)
|
|
|
|
*/
|
|
|
|
|
|
|
|
return x === null ? null : this.list_tagged[(parseInt(x) === this.list_tagged.length-1) ? 0 : (parseInt(x)+1)];
|
|
|
|
}
|
|
|
|
|
|
|
|
MsgArea.prototype.MessagePrev = function(page) {
|
|
|
|
var prev = null;
|
|
|
|
var x = null;
|
|
|
|
|
|
|
|
if (! page)
|
|
|
|
return undefined;
|
|
|
|
|
|
|
|
var msgid = page.substr(7,4);
|
|
|
|
|
|
|
|
for(x in this.list_tagged) {
|
|
|
|
if (this.list_tagged[x].tags === msgid) {
|
|
|
|
break;
|
|
|
|
|
|
|
|
} else {
|
|
|
|
prev = x;
|
|
|
|
}
|
|
|
|
|
|
|
|
write(); // @todo This is needed for this to work?
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
= Our previous message is either
|
|
|
|
+ prev if a tag was found, unless
|
|
|
|
+ prev is null, in which case it is this.list_tagged.length -1
|
|
|
|
+ null if x is still null (thus no messages)
|
|
|
|
*/
|
|
|
|
|
|
|
|
// If prev is still null, then our last message must be the last one, unless x is null then there are no messages
|
|
|
|
return x === null ? null : this.list_tagged[(prev === null) ? this.list_tagged.length-1 : parseInt(prev)];
|
|
|
|
}
|
|
|
|
|
2022-04-18 21:27:25 +10:00
|
|
|
/**
|
|
|
|
* Tag messages with a frame number
|
2022-12-15 23:28:25 +11:00
|
|
|
* @note: May need to run jsexec with -m 32MB to overcome memory issues
|
2022-04-18 21:27:25 +10:00
|
|
|
*
|
|
|
|
* @returns {boolean}
|
|
|
|
*/
|
|
|
|
MsgArea.prototype.tag_msgs = function() {
|
|
|
|
var msgs = this.list_untagged;
|
|
|
|
|
2022-12-15 23:28:25 +11:00
|
|
|
writeln("We have "+msgs.length+" messages to tag.");
|
|
|
|
|
|
|
|
// See if we need to tag something
|
2022-04-18 21:27:25 +10:00
|
|
|
if (! msgs.length)
|
|
|
|
return;
|
|
|
|
|
2022-04-20 21:19:01 +10:00
|
|
|
if (! this.msgbase.open()) {
|
|
|
|
writeln(code+' cannot be opened:'+this.msgbase.error);
|
2022-04-18 21:27:25 +10:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
var page_next = this.page_next;
|
2022-04-22 22:42:20 +10:00
|
|
|
|
2022-04-18 21:27:25 +10:00
|
|
|
for(var x in msgs) {
|
2022-04-22 22:42:20 +10:00
|
|
|
msgs[x].tags = (''+(page_next)).padStart(4,'0');
|
|
|
|
|
|
|
|
if(! this.msgbase.put_msg_header(msgs[x].number,msgs[x])) {
|
2022-04-18 21:27:25 +10:00
|
|
|
writeln('ERROR:'+this.msgbase.error);
|
|
|
|
|
|
|
|
} else {
|
|
|
|
page_next++;
|
2022-04-22 22:42:20 +10:00
|
|
|
if (page_next > MAX_PAGE_NUM)
|
|
|
|
page_next = 0;
|
2022-04-18 21:27:25 +10:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
this.msgbase.close();
|
|
|
|
this.page_next = page_next;
|
2022-04-20 21:19:01 +10:00
|
|
|
|
2022-04-18 21:27:25 +10:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2022-12-15 23:28:25 +11:00
|
|
|
MsgArea.prototype.page = function(msgid) {
|
|
|
|
return '1'+this.page_prefix+msgid;
|
|
|
|
}
|
|
|
|
|
2022-05-01 17:42:19 +10:00
|
|
|
MsgAreas.prototype.getArea = function(area) {
|
|
|
|
log(LOG_DEBUG,'- AREA:'+JSON.stringify(area));
|
|
|
|
if (area === undefined)
|
2022-04-20 21:19:01 +10:00
|
|
|
return undefined;
|
|
|
|
|
2022-05-01 17:42:19 +10:00
|
|
|
var zone = (''+area).substr(1,4);
|
|
|
|
var echo = (''+area).substr(5,2);
|
2022-05-06 09:59:23 +10:00
|
|
|
log(LOG_DEBUG,' - zone:'+zone);
|
|
|
|
log(LOG_DEBUG,' - echo:'+echo);
|
2022-04-18 21:27:25 +10:00
|
|
|
|
2022-05-01 17:42:19 +10:00
|
|
|
return this.areas.filter(function(x) {
|
2022-04-18 21:27:25 +10:00
|
|
|
return x.zone_id === zone && x.area_id === echo;
|
|
|
|
})[0]
|
2022-05-01 17:42:19 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
MsgAreas.prototype.getMessage = function(page) {
|
|
|
|
var area = this.getArea(page);
|
|
|
|
log(LOG_DEBUG,' - msg:'+JSON.stringify(page.substr(7,4)));
|
|
|
|
|
|
|
|
return area ? area.getMessage(page.substr(7,4)) : undefined;
|
|
|
|
}
|
|
|
|
|
|
|
|
MsgAreas.prototype.getUserStats = function(page) {
|
|
|
|
var area = this.getArea(page);
|
2022-04-18 21:27:25 +10:00
|
|
|
|
2022-05-01 17:42:19 +10:00
|
|
|
return area ? msg_area.grp_list[area.msgbase.cfg.grp_number].sub_list[msg_area.sub[area.msgbase.cfg.code].index] : undefined;
|
2022-04-18 21:27:25 +10:00
|
|
|
}
|