diff --git a/README.md b/README.md index 715c80e..f05ffa4 100644 --- a/README.md +++ b/README.md @@ -3,7 +3,7 @@ Share (and sync) Objects Online with the power of websockets. Keys, Values AND R
**Please note**: - To view example usage of the modules this library provides, please refer to the _[tests](https://github.com/Y0ursTruly/webject/blob/main/tests.js)_ -- TypeScript Definitions Added >:D +- Supporting the `Date` type soon. In the meantime it is advised to `Date` instances into their string or number equivalent # Installation Multiple ways diff --git a/for_browser.js b/for_browser.js index 1f8c9d4..4b3fabb 100644 --- a/for_browser.js +++ b/for_browser.js @@ -13,6 +13,7 @@ //see if an enumerable property(of key) exists(in obj) function includes(obj,key){ + if(!obj) return false; if( (obj instanceof Array) && (key==="length") ) return true; let existing=describe(obj,key) return existing?existing.enumerable:false @@ -80,12 +81,12 @@ } if(typeof item!=="object") return item; //numbers, strings //also functions but these get ignored either way - let shell=item instanceof Array? ([]): ({}), keys=Object.keys(item); - for(let i=0;i{const{Object:t,JSON:e,WeakMap:n,ReferenceError:r,TypeError:o,RangeError:i}=window,{keys:l,getOwnPropertyDescriptor:u}=t,{stringify:a,parse:c}=e,f=new n;function s(t,e){if(t instanceof Array&&"length"===e)return!0;let n=u(t,e);return!!n&&n.enumerable}function y(t,e){if(t===e)return!0;if("symbol"==typeof t&&"symbol"==typeof e)return t.description===e.description;let n=typeof t==typeof e,r=t instanceof Array==e instanceof Array,o=(t?t[Symbol.toStringTag]:null)===(e?e[Symbol.toStringTag]:null),i=n&&r&&o&&"object"==typeof t;return i&&t[Symbol.toStringTag]?.includes("Array")?t.length===e.length:i}function g(t,e,n=0){if(!t||!t.length)return e;var o=e,i=0;for(i=0;i128)throw new i("Given object goes too many levels inward (>128)");let f=l(t),g=l(e),p=n.get(t);c&&a.set(t,!0),t instanceof Array&&(f.push("length"),g.push("length"));for(let i=0;i=2?[p[1],l]:g,S=!y(t[l],e[l]),A=n.get(c);"object"==typeof t[l]&&t[l]||"object"!=typeof e[l]||!e[l]||w(n,e[l]);let m=!s(e,l)||S;if(!m&&A&&(A[2]=0),m)if((S||void 0===c)&&(e[l]=A?A[3]:d(c,!0)),A){A[4]++;let[t,e,n]=A;n?t=e:(A[2]=2,A[1]=r.length),r.push([h,t,n])}else if(null===c||void 0!==c&&!c[Symbol.toStringTag])r.push([h,d(c)]);else{c&&"bigint"!=typeof c&&a.set(c,!0);let t=c||"bigint"==typeof c?c[Symbol.toStringTag]:"undefined";r.push([h,d(c),0,t])}("symbol"==typeof c||"object"==typeof c&&null!==c)&&(A||(n.set(c,[g,r.length-1,m?1:0,e[l],1]),n.set(e[l],c)),a.get(c)||(a.set(c,!0),b(t[l],e[l],n,r,g,u+1,a)))}}const S={__proto__:null,1:function(){return!1},2:function(t,e){if(1===t.length)return!1;try{return g(t[0],e),!1}catch{return!0}},3:function(){return!0}};let A=[];for(let t=0;t<256;t++)A[t]=String.fromCharCode(t);window.objToString=function(t,e){if(Array.isArray(t)||"object"!=typeof t||null===t)throw new o("root element of data MUST be an OBJECT");if(e)var r=d(t,!0);else var{clone:r,map:i}=f.get(t)||{};const l=[[[],i?{}:d(t)]],u=[],c=t&&t[Symbol.toStringTag]||null;if(p[c]&&l[0].push(0,c),!i){r=d(t,!0);var i=new n;f.set(t,{clone:r,map:i})}return i.set(t,[u,0,1,r,1]),b(t,r,i,l,u,1,new n,!0),a(l)},window.stringToObj=function(e,n,r=3){if(1===r||"1"===r)return n;let o="function"==typeof r?r:S[r];var i=c("string"==typeof e?e:function(t){let e=new Uint8Array(t),n="";for(let t=0;t0?i[f][s-1]:f,n);else{let t=g(c,n,1);y(t[h],l[1])||(t[h]=p[l[3]](l[1]))}}return n},window.objValueFrom=g,window.partFilter=function(t,e=!1){Array.isArray(t)||(t=[t]);for(let e=0;e{var t=objToString({});WebSocket.prototype.on=WebSocket.prototype.addEventListener;let e=new Map,n=(t,e)=>crypto.getRandomValues(new Uint32Array(1))[0]%(t-e)+e;var r="abcdefgjiklmnopqrstuvwxyz-_0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ :;.,\\/\"'?!(){}[]@<>=+*#$&`|~^%".split("");function o(t,o){"string"!=typeof o&&(o="webject_");do{var i="",l=n(2*t,t);for(let t=0;t:|");if(!0===u&&(u=t=>e(n,r,i,l,u)),"function"!=typeof u&&u)throw new Error("If you choose the optional parameter onFail, it must be a function >:|");if(l&&"object"==typeof l&&("function"!=typeof l.encoder||"function"!=typeof l.decoder))throw new TypeError("If coding parameter is used, it MUST be an object with both 'encoder' and 'decoder' functions");let a=null,c=null,f=null,s=null,y=!1,g=new WebSocket(n),p=new Promise(((t,e)=>(a=t,c=e)));function h(t,e){if(y)return null;let n=t,r=null;isNaN(n)&&(n=t.code),r=1006==n?"closed ABNORMALLY: either you or the server just LOST connection :|":1002==n?"authToken ENCODING FAULT: incorrect encoding/faulty protocol used, according to the server":1001==n?"authToken LOCKED: this is a correct key, but it takes no new connections 0_0":"closed PURPOSEFULLY: check your location and token parameters, OR you got BOOTED :/";let o=`connection with server is OVER due to event: ${e}\n${r}`;a?c(o):console.error(o),clearInterval(f),clearInterval(s),u&&(a?u().then(a):u())}return g.onerror=t=>{console.error("Attempting to connect to a websocket using the location parameter produced the following error :/\n~",t.message),u&&(a?u().then(a):u())},g.on("open",(async()=>{g.send(JSON.stringify(l?[r,await l.encoder(JSON.stringify([o(32,""),Date.now()]))]:r));let e=!1,n=Date.now();async function u(){let e=objToString(i);e!==t&&g.send(l?await l.encoder(e):e)}s=setInterval((function(){Date.now()-n>32768&&(y=!0,h(1006,"Connection Broken"))}),5e3),g.on("message",(async function(t){if(void 0!==t.data&&(t=t.data),"PING"==t)return n=Date.now();i=stringToObj(l?await l.decoder(t):t,i),e||(objToString(i),f=setInterval(u,20),e=!0,a(i))})),g.on("disconnect",(t=>h(t,"disconnect"))),g.on("close",(t=>h(t,"close"))),g.on("error",(t=>h(t,"error")))})),await p}})(); \ No newline at end of file +(()=>{const{Object:t,JSON:e,WeakMap:n,ReferenceError:r,TypeError:o,RangeError:i}=window,{keys:l,getOwnPropertyDescriptor:u}=t,{stringify:f,parse:a}=e,c=new n;function s(t,e){if(!t)return!1;if(t instanceof Array&&"length"===e)return!0;let n=u(t,e);return!!n&&n.enumerable}function y(t,e){if(t===e)return!0;if("symbol"==typeof t&&"symbol"==typeof e)return t.description===e.description;let n=typeof t==typeof e,r=t instanceof Array==e instanceof Array,o=(t?t[Symbol.toStringTag]:null)===(e?e[Symbol.toStringTag]:null),i=n&&r&&o&&"object"==typeof t;return i&&t[Symbol.toStringTag]?.includes("Array")?t.length===e.length:i}function g(t,e,n=0){if(!t||!t.length)return e;var o=e,i=0;for(i=0;i128)throw new i("Given object goes too many levels inward (>128)");let c=l(t),g=l(e),p=n.get(t);a&&f.set(t,!0),t instanceof Array&&(c.push("length"),g.push("length"));for(let i=0;i=2?[p[1],l]:g,S=!y(t[l],e[l]),A=n.get(a);"object"==typeof t[l]&&t[l]||"object"!=typeof e[l]||!e[l]||w(n,e[l]);let m=!s(e,l)||S;if(!m&&A&&(A[2]=0),m)if((S||void 0===a)&&(e[l]=A?A[3]:d(a,!0)),A){A[4]++;let[t,e,n]=A;n?t=e:(A[2]=2,A[1]=r.length),r.push([h,t,n])}else if(null===a||void 0!==a&&!a[Symbol.toStringTag])r.push([h,d(a)]);else{a&&"bigint"!=typeof a&&f.set(a,!0);let t=a||"bigint"==typeof a?a[Symbol.toStringTag]:"undefined";r.push([h,d(a),0,t])}if("symbol"==typeof a||"object"==typeof a&&null!==a){if(!A){const t=n.get(e[l]);t&&n.delete(t),n.set(a,[g,r.length-1,m?1:0,e[l],1]),n.set(e[l],a)}f.get(a)||(f.set(a,!0),b(t[l],e[l],n,r,g,u+1,f))}}}const S={__proto__:null,1:function(){return!1},2:function(t,e){if(1===t.length)return!1;try{return g(t[0],e),!1}catch{return!0}},3:function(){return!0}};let A=[];for(let t=0;t<256;t++)A[t]=String.fromCharCode(t);window.objToString=function(t,e){if(Array.isArray(t)||"object"!=typeof t||null===t)throw new o("root element of data MUST be an OBJECT");if(e)var r=d(t,!0);else var{clone:r,map:i}=c.get(t)||{};const l=[[[],i?{}:d(t)]],u=[],a=t&&t[Symbol.toStringTag]||null;if(p[a]&&l[0].push(0,a),!i){r=d(t,!0);var i=new n;c.set(t,{clone:r,map:i})}return i.set(t,[u,0,1,r,1]),b(t,r,i,l,u,1,new n,!0),f(l)},window.stringToObj=function(e,n,r=3){if(1===r||"1"===r)return n;let o="function"==typeof r?r:S[r];var i=a("string"==typeof e?e:function(t){let e=new Uint8Array(t),n="";for(let t=0;t0?i[c][s-1]:c,n);else{let t=g(a,n,1);y(t[h],l[1])||(t[h]=p[l[3]](l[1]))}}return n},window.objValueFrom=g,window.partFilter=function(t,e=!1){Array.isArray(t)||(t=[t]);for(let e=0;e{var t=objToString({});WebSocket.prototype.on=WebSocket.prototype.addEventListener;let e=new Map,n=(t,e)=>crypto.getRandomValues(new Uint32Array(1))[0]%(t-e)+e;var r="abcdefgjiklmnopqrstuvwxyz-_0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ :;.,\\/\"'?!(){}[]@<>=+*#$&`|~^%".split("");function o(t,o){"string"!=typeof o&&(o="webject_");do{var i="",l=n(2*t,t);for(let t=0;t:|");if(!0===u&&(u=t=>e(n,r,i,l,u)),"function"!=typeof u&&u)throw new Error("If you choose the optional parameter onFail, it must be a function >:|");if(l&&"object"==typeof l&&("function"!=typeof l.encoder||"function"!=typeof l.decoder))throw new TypeError("If coding parameter is used, it MUST be an object with both 'encoder' and 'decoder' functions");let f=null,a=null,c=null,s=null,y=!1,g=new WebSocket(n),p=new Promise(((t,e)=>(f=t,a=e)));function h(t,e){if(y)return null;let n=t,r=null;isNaN(n)&&(n=t.code),r=1006==n?"closed ABNORMALLY: either you or the server just LOST connection :|":1002==n?"authToken ENCODING FAULT: incorrect encoding/faulty protocol used, according to the server":1001==n?"authToken LOCKED: this is a correct key, but it takes no new connections 0_0":"closed PURPOSEFULLY: check your location and token parameters, OR you got BOOTED :/";let o=`connection with server is OVER due to event: ${e}\n${r}`;f?a(o):console.error(o),clearInterval(c),clearInterval(s),u&&(f?u().then(f):u())}return g.onerror=t=>{console.error("Attempting to connect to a websocket using the location parameter produced the following error :/\n~",t.message),u&&(f?u().then(f):u())},g.on("open",(async()=>{g.send(JSON.stringify(l?[r,await l.encoder(JSON.stringify([o(32,""),Date.now()]))]:r));let e=!1,n=Date.now();async function u(){let e=objToString(i);e!==t&&g.send(l?await l.encoder(e):e)}s=setInterval((function(){Date.now()-n>32768&&(y=!0,h(1006,"Connection Broken"))}),5e3),g.on("message",(async function(t){if(void 0!==t.data&&(t=t.data),"PING"==t)return n=Date.now();i=stringToObj(l?await l.decoder(t):t,i),e||(objToString(i),c=setInterval(u,20),e=!0,f(i))})),g.on("disconnect",(t=>h(t,"disconnect"))),g.on("close",(t=>h(t,"close"))),g.on("error",(t=>h(t,"error")))})),await p}})(); \ No newline at end of file diff --git a/package-lock.json b/package-lock.json index ea59e14..580087e 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,15 +1,19 @@ { "name": "webject", - "version": "1.4.7", + "version": "1.4.8", "lockfileVersion": 3, "requires": true, "packages": { "": { "name": "webject", - "version": "1.4.7", + "version": "1.4.8", "license": "Apache-2.0", "dependencies": { "ws": "^8.17.1" + }, + "engines": { + "node": ">= 16.20.2", + "npm": ">= 8.19.4" } }, "node_modules/ws": { diff --git a/package.json b/package.json index f438fad..3f208cd 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "webject", - "version": "1.4.7", + "version": "1.4.8", "description": "Share Objects Online with the power of websockets. Keys, Values AND references. Webject is short for Web Object and it really is a system for sharing objects on the web. Someone can host an object, and make authTokens for others online to share this object", "main": "webject.js", "scripts": { diff --git a/serial.js b/serial.js index 9619e9b..7eb32fe 100644 --- a/serial.js +++ b/serial.js @@ -2,10 +2,7 @@ - //every index in an objToString used to be {path,value,reference,delete} - //now it's [[indicator,...path],otherArgument] - //for otherArgument: 0 means delete(no otherArgument), 1 means reference, no indicator means value - const {Object,JSON,WeakMap,ReferenceError,TypeError,RangeError}=global + const {Object,JSON,WeakMap,ReferenceError,TypeError}=global const {keys,getOwnPropertyDescriptor:describe}=Object const {stringify:str,parse}=JSON, CACHE=new WeakMap() @@ -13,6 +10,7 @@ //see if an enumerable property(of key) exists(in obj) function includes(obj,key){ + if(!obj) return false; if( (obj instanceof Array) && (key==="length") ) return true; let existing=describe(obj,key) return existing?existing.enumerable:false @@ -82,12 +80,12 @@ } if(typeof item!=="object") return item; //numbers, strings //also functions but these get ignored either way - let shell=item instanceof Array? ([]): ({}), keys=Object.keys(item); - for(let i=0;i=2? [data[1],key]: Path; + let Path=[...PATH,key], path=data[2]>0&&(data[1]=2)? [data[1],key]: Path; let notSame=!same( obj[key],clone[key] ), temp=map.get(item) if((typeof obj[key]!=="object"||!obj[key]) && (typeof clone[key]==="object"&&clone[key])) @@ -172,6 +170,8 @@ } if(typeof item==="symbol" || (typeof item==="object" && item!==null)){ if(!temp){ + const former=map.get(clone[key]) + if(former) map.delete(former); map.set(item,[Path,list.length-1,newEntry?1:0,clone[key],1]); map.set(clone[key],item); } @@ -301,4 +301,4 @@ -})() \ No newline at end of file +})() diff --git a/tests.js b/tests.js index fb52a21..36b5793 100644 --- a/tests.js +++ b/tests.js @@ -74,6 +74,27 @@ const {serve, connect, sync, desync, objToString, stringToObj, partFilter, objVa assert.strictEqual(testObj.a.b.f,undefined) assert.strictEqual(testObj.a.b.c,3) }) + await t.test("Serialisation Crash Testing",async function(){ + const date=new Date(), obj={events:[{date,arr:['value1']}]} + const clone=stringToObj(objToString(obj)), {events}=obj + const standard={"events":[{"date":date-0,"arr":["value1","value2","value2","value2","value2","value2"]}]} + let update=_=>stringToObj(objToString(obj),clone); + for(let i=0;i<5;i++){ + obj.events[0].date-=0; + update() + obj.events=[{date}]; + update() + obj.events[0].date-=0; + update() + obj.events=null; + update() + events[0].arr.push("value"+(events.length+1)); + obj.events=events; + update() //the call that crashes it + } + assert.deepStrictEqual(obj,clone) + assert.deepStrictEqual(obj,standard) + }) }) //third set of tests