Re: Information Storage Question
So this came up in my own code. Some info for TinyMUX users with standard installs:
-
Data in a single attribute: 8000 characters counting the attribute name, plus attribute settings and seemingly some other weirdness, so really somewhere in the neighborhood of 7900 characters if you don't want to risk cutting it off.
-
Max attribute name length without data loss: 63
-
Attributes on a single object: I got up to 2665 in a number of tests with full-named attributes all filled with 7900+ characters of data, but some of my other tests I managed only as high as 2513, so decided to cut it off at 2500. Again, some kind of attribute naming scheme weirdness.
-
Number of attributes able to be stored on an object does not appear to have anything to do with the object's memory as reported by objmem() - you get the same results if you just put in a 1 for the value and use simple numeric attribute names.
Testing methodology:
@wait me=th attrcnt(#672)
@dolist/notify lnum(700)=@set/quiet #672=_L.##:1
Repeat in increments of 700 until object won't take anymore. MUX fails silently so you'll have to keep an eye on it.
Next test:
@dolist/notify lnum(700)=@set/quiet #672=_L.##.[iter(lnum(rand(sub(59,strlen(##)))),pickrand(a b c d e f g h i j k l m n o p q r s t u v w x y z),,@@)]:[repeat(0123456789, 799)]
Screwing around like that got me the "magic" numbers above.
I decided I didn't want to worry about this crap so I'd write code to do it for me. That code had the following requirements:
- Pure MUX. I leave the SQL to someone willing to write something that will do it right in a non-blocking fashion.
- Store at least 20k attributes.
- Do it all behind-the-scenes, no wizard intervention required when things get over-stuffed.
- Must not lose data. MUX's "I eat data for breakfast" habit bugged me.
- Must function transparently to the coder using it - so just like set().
- Be expandable in case 20k isn't enough.
From that came this code, including all the documentation I could think of:
@create Data Storage Functions <DSF>=10
@set DSF=safe inherit
@@ Unique key goes here. Required if you want to call this from code without
@@ opening the function up to everyone.
&vK DSF=UNIQUE KEY HERE
@@ Exercise caution expanding this. On MUX with extensive testing I
@@ you can push this up to 2662, but my tests were inconclusive when I started
@@ playing with attribute length. Data loss is possible with values higher than 2550.
@@ Playing it safe and setting this low enough that there
&d.max-attributes DSF=2500
@@ From my tests, these work. 7900 because I ran into trouble with numbers past
@@ 7940 with long attribute names. Rounding a bit because I like round numbers.
&d.max-attr-name-length DSF=63
&d.max-attr-value-length DSF=7900
@@ Who can run this? Strong recommendation: at least wizard. Should take care of
@@ your inherit objects and code stuff - we only want this being called by
@@ system code if possible. Since I couldn
@@ worked I stuck in a unique key. Fix this if you know what the heck you
@@ doing.
@@
@@ Arg:
@@ %0 - Unique key, if there is one.
&l.canset DSF=or(gte(bittype(%#), 5), strmatch(%0, %vK))
@@ Find the first parent that does NOT hit the attr cap, or the parent that
@@ has the attribute on it, if it already exists.
@@
@@ Arg:
@@ %0 - target object to parent
@@ %1 - target attribute
@@ Note: I don
@@ If the name is too long (more than 399 characters on my game) it just gets
@@ partially eaten. The objects are still there. I didn
@@ because I didn
@@ parents, since MUX doesn
&f.find-first-free-parent DSF=
if(
or(
lt(attrcnt(%0), v(d.max-attributes)),
hasattr(%0, %1)
),
%0,
if(
t(parent(%0)),
ulocal(f.find-first-free-parent, parent(%0), %1),
null(strcat(
setr(0, create(name(%0) - DSP %0, 10)),
parent(%0, %q0),
set(%q0, safe),
tel(%q0, loc(%0))
))%q0
)
)
@@ Set the data (global user function, should be wiz-only)
@@
@@ Args:
@@ %0 - target object
@@ %1 - attr name:value (just like set())
@@ %2 - optional key, required for calling if not a wizard
@@ Results: Nothing, just like set().
&f.global.setdata DSF=
case(1,
not(ulocal(l.canset, %2)),
#-1 ERROR: HIGHER PERMISSIONS REQUIRED.,
gt(strlen(%1), v(d.max-attr-value-length)),
#-2 ERROR: TOO MUCH DATA.,
not(strmatch(%1, *:*)),
#-3 ERROR: INVALID DATA FORMAT. MUST BE NAME:VALUE.,
gt(strlen(first(%1, :)), v(d.max-attr-name-length)),
#-4 ERROR: ATTRIBUTE NAME TOO LONG.,
if(
or(
t(setr(0, set(ulocal(f.find-first-free-parent, %0, first(%1, :)), %1))),
eq(strlen(%q0), 0)
),
@@(All good. Say nothing!),
%q0
)
)
@@ All done! Anything past this point is gravy.
@@ -----------------------------------------------------------------------------
@@ Set function up on #1 with:
@@ @startup #1=@fo me=@function/privilege/preserve setdata=#DBREF_OF_DSF_GOES_HERE/f.global.setdata
@@
@@ The function needs privilege to create its own objects and set data on things.
@@ It needs preserve because we use %q0 in there and don
@@ Because it
@@ Call function wherever you would call set(target, attrname:attrvalue) like so:
@@ setdata(target, attrname:attrvalue)
And some test code to use it...
@@ And here's a simple tester. All it does is let staff AND PLAYERS add data to
@@ a specified object. Because it has its own unique key, it can do that. If you
@@ want to restrict it to a specific bit type, @lock it. This is just a proof of
@@ concept.
@@
@@ Commands:
@@ +setdata <attr>=<value>
@@ +setdata/quiet <attr>=<value>
@@ +getdata <attr>
@@ +datastats
@create Basic Data Storage Object <BDSO>=10
@set BDSO=safe
@create Data Storage Command Tester <DSCT>=10
@set DSCT=safe inherit
@fo me=&vD DSCT=[num(BDSO)]
&vK DSCT=UNIQUE KEY HERE
@@ Get all the data objects.
@@
@@ Args:
@@ %0 - target object
@@ Results: Nothing, just like set().
&f.list-parents DSCT=if(t(parent(%0)), parent(%0) [u(f.list-parents, parent(%0))])
@@ Set the data loudly.
@@ Format: +setdata attrname=attrvalue
@@ Result: "Set." or error.
&cmd-+setdata DSCT=$+setdata *=*:@pemit %#=[if(or(t(setr(0, setdata(%vD, %0:%1, %vK))), eq(strlen(%q0), 0)), Set., %q0)];
@@ Set the data quietly.
@@ Format: +setdata attrname=attrvalue
@@ Result: Nothing unless there's an error.
&cmd-+setdata/quiet DSCT=$+setdata/quiet *=*:@switch [setr(0, setdata(%vD, %0:%1, %vK))]=,{},@pemit %#=%q0;
@@ Get the data.
@@ Format: +getdata attrname
@@ Result: Data or a list of possible attributes it could be on using lattrp.
&cmd-+getdata DSCT=$+getdata *:@pemit %#=[if(t(setr(0, xget(%vD, %0))), MSG: Your data is: %q0, MSG: Attribute not found. Did you mean: [lattrp(%vD/%0*)]?)];
@@ Show some data stats.
@@ Format: +datastats
@@ Result: The stats.
&cmd-+datastats DSCT=$+datastats:@pemit %#=[repeat(-, 80)]%RBase object: %vD%RParents: [setr(P, [u(f.list-parents, %vD)])]%R[iter(%vD %qP, if(t(itext(0)), [repeat(-, 80)]%R[itext(0)] Name: [name(itext(0))]%RAttributes: [attrcnt(itext(0))]%RObject memory used: [objmem(itext(0))]%R),,@@)][repeat(-, 80)]
I dunno how to solve the key issue, I have a headcold. You could solve just by making the function not-wizard, but then it fails at its primary requirement.
Throwing this up there in case someone besides me finds it useful and can solve the tricky bits.