diff --git a/MeshCentralServer.njsproj b/MeshCentralServer.njsproj index b99da564..0c2055a1 100644 --- a/MeshCentralServer.njsproj +++ b/MeshCentralServer.njsproj @@ -193,7 +193,7 @@ - + diff --git a/views/default.handlebars b/views/default.handlebars index cf98b450..74d75fad 100644 --- a/views/default.handlebars +++ b/views/default.handlebars @@ -5198,7 +5198,7 @@ } csv += '\r\n'; } - saveAs(new Blob([csv], { type: 'application/octet-stream' }), "devicelist.csv"); + saveAs(stringToUtf8Blob(csv), "devicelist.csv"); } return false; } @@ -12598,7 +12598,7 @@ if (eventList[i].nodeid) { var node = getNodeFromId(eventList[i].nodeid); if (node && node.name) { nodename = node.name; } } csv += '"' + eventList[i].time + '","' + printDateTime(new Date(eventList[i].time)) + '","' + eventList[i].etype + '","' + ((eventList[i].action != null) ? eventList[i].action : '') + '","' + ((eventList[i].username != null) ? eventList[i].username : '') + '","' + EscapeHtml(nodename) + '","' + ((eventList[i].msg != null) ? eventList[i].msg : '').split(',').join(' -') + '"\r\n'; } - saveAs(new Blob([csv], { type: 'application/octet-stream' }), "eventslist.csv"); + saveAs(stringToUtf8Blob(csv), "eventslist.csv"); return false; } @@ -12971,7 +12971,7 @@ } csv += '"' + users[i]._id + '","' + users[i].name + '","' + (users[i].email ? users[i].email : '') + '","' + (users[i].creation ? new Date(users[i].creation * 1000) : '') + '","' + (users[i].login ? new Date(users[i].login * 1000) : '') + '","' + (users[i].groups ? users[i].groups.join(',') : '') + '","' + (multiFactor ? factors.join(',') : '') + '"\r\n'; } - saveAs(new Blob([csv], { type: 'application/octet-stream' }), "userlist.csv"); + saveAs(stringToUtf8Blob(csv), "userlist.csv"); return false; } @@ -14957,7 +14957,7 @@ csv += new Date(serverTimelineStats[i].time) + ', ' + serverTimelineStats[i].conn.ca + ', ' + serverTimelineStats[i].conn.cu + ', ' + serverTimelineStats[i].conn.us + ', ' + serverTimelineStats[i].conn.rs + ', ' + (serverTimelineStats[i].conn.am ? serverTimelineStats[i].conn.am : '') + ', ' + serverTimelineStats[i].mem.external + ', ' + serverTimelineStats[i].mem.heapUsed + ', ' + serverTimelineStats[i].mem.heapTotal + ', ' + serverTimelineStats[i].mem.rss + '\r\n'; } } - saveAs(new Blob([csv], { type: 'application/octet-stream' }), "ServerStats.csv"); + saveAs(stringToUtf8Blob(csv), "ServerStats.csv"); } // @@ -15028,7 +15028,7 @@ function p41downloadServerTrace() { var csv = "time, source, message" + '\r\n'; for (var i in serverTrace) { csv += '"' + new Date(serverTrace[i].time).toLocaleTimeString() + '","' + serverTrace[i].source + '","' + serverTrace[i].args.join(', ') + '"\r\n'; } - saveAs(new Blob([csv], { type: 'application/octet-stream' }), "servertrace.csv"); + saveAs(stringToUtf8Blob(csv), "servertrace.csv"); return false; } @@ -15631,6 +15631,17 @@ } } + // Convert a string into a UTF8 blob with the UTF8 header in front of it. + function stringToUtf8Blob(str) { + const bytes = new TextEncoder().encode(str); + var bytes2 = new Uint8Array(3 + bytes.length); + bytes2[0] = 0xEF; // This is the UTF-8 header for CSV files, add it to the start of the file. + bytes2[1] = 0xBB; + bytes2[2] = 0xBF; + for (var i = 0; i < bytes.length; i++) { bytes2[i + 3] = bytes[i]; } + return new Blob([bytes2], { type: 'application/octet-stream' }) // application/json;charset=utf-8 + } + function getNodeAmtVersion(node) { if ((node == null) || (node.intelamt == null) || (typeof node.intelamt.ver != 'string')) return 0; var verSplit = node.intelamt.ver.split('.'); if (verSplit.length < 2) return 0; return parseInt(verSplit[0]) + (parseInt(verSplit[1]) / 100); } function getstore(name, val) { try { if (typeof (localStorage) === 'undefined') return val; var v = localStorage.getItem(name); if ((v == null) || (v == null)) return val; return v; } catch (e) { return val; } } function addLink(x, f) { return '' + x + ' '; }