corrade-lsl-templates – Blame information for rev 10

Subversion Repositories:
Rev:
Rev Author Line No. Line
8 office 1 ///////////////////////////////////////////////////////////////////////////
2 // Copyright (C) Wizardry and Steamworks 2016 - License: GNU GPLv3 //
3 ///////////////////////////////////////////////////////////////////////////
4 //
5 // A module that bans group members using fuzzy name matching.
6 //
7 ///////////////////////////////////////////////////////////////////////////
8  
9 ///////////////////////////////////////////////////////////////////////////
10 // Copyright (C) 2014 Wizardry and Steamworks - License: GNU GPLv3 //
11 ///////////////////////////////////////////////////////////////////////////
12 string wasKeyValueGet(string k, string data) {
13 if(llStringLength(data) == 0) return "";
14 if(llStringLength(k) == 0) return "";
15 list a = llParseString2List(data, ["&", "="], []);
16 integer i = llListFindList(a, [ k ]);
17 if(i != -1) return llList2String(a, i+1);
18 return "";
19 }
20  
21 ///////////////////////////////////////////////////////////////////////////
22 // Copyright (C) 2013 Wizardry and Steamworks - License: GNU GPLv3 //
23 ///////////////////////////////////////////////////////////////////////////
24 string wasKeyValueEncode(list data) {
25 list k = llList2ListStrided(data, 0, -1, 2);
26 list v = llList2ListStrided(llDeleteSubList(data, 0, 0), 0, -1, 2);
27 data = [];
28 do {
29 data += llList2String(k, 0) + "=" + llList2String(v, 0);
30 k = llDeleteSubList(k, 0, 0);
31 v = llDeleteSubList(v, 0, 0);
32 } while(llGetListLength(k) != 0);
33 return llDumpList2String(data, "&");
34 }
35  
36 ///////////////////////////////////////////////////////////////////////////
37 // Copyright (C) 2011 Wizardry and Steamworks - License: GNU GPLv3 //
38 ///////////////////////////////////////////////////////////////////////////
39 // http://was.fm/secondlife/wanderer
40 vector wasCirclePoint(float radius) {
41 float x = llPow(-1, 1 + (integer) llFrand(2)) * llFrand(radius*2);
42 float y = llPow(-1, 1 + (integer) llFrand(2)) * llFrand(radius*2);
43 if(llPow(x,2) + llPow(y,2) <= llPow(radius,2))
44 return <x, y, 0>;
45 return wasCirclePoint(radius);
46 }
47  
48 ///////////////////////////////////////////////////////////////////////////
49 // Copyright (C) 2015 Wizardry and Steamworks - License: GNU GPLv3 //
50 ///////////////////////////////////////////////////////////////////////////
51 // escapes a string in conformance with RFC1738
52 string wasURLEscape(string i) {
53 string o = "";
54 do {
55 string c = llGetSubString(i, 0, 0);
56 i = llDeleteSubString(i, 0, 0);
57 if(c == "") jump continue;
58 if(c == " ") {
59 o += "+";
60 jump continue;
61 }
62 if(c == "\n") {
63 o += "%0D" + llEscapeURL(c);
64 jump continue;
65 }
66 o += llEscapeURL(c);
67 @continue;
68 } while(i != "");
69 return o;
70 }
71  
72 ///////////////////////////////////////////////////////////////////////////
73 // Copyright (C) 2015 Wizardry and Steamworks - License: GNU GPLv3 //
74 ///////////////////////////////////////////////////////////////////////////
75 list wasCSVToList(string csv) {
76 list l = [];
77 list s = [];
78 string m = "";
79 do {
80 string a = llGetSubString(csv, 0, 0);
81 csv = llDeleteSubString(csv, 0, 0);
82 if(a == ",") {
83 if(llList2String(s, -1) != "\"") {
84 l += m;
85 m = "";
86 jump continue;
87 }
88 m += a;
89 jump continue;
90 }
91 if(a == "\"" && llGetSubString(csv, 0, 0) == a) {
92 m += a;
93 csv = llDeleteSubString(csv, 0, 0);
94 jump continue;
95 }
96 if(a == "\"") {
97 if(llList2String(s, -1) != a) {
98 s += a;
99 jump continue;
100 }
101 s = llDeleteSubList(s, -1, -1);
102 jump continue;
103 }
104 m += a;
105 @continue;
106 } while(csv != "");
107 // postcondition: length(s) = 0
108 return l + m;
109 }
110  
111 ///////////////////////////////////////////////////////////////////////////
112 // Copyright (C) 2015 Wizardry and Steamworks - License: GNU GPLv3 //
113 ///////////////////////////////////////////////////////////////////////////
114 string wasListToCSV(list l) {
115 list v = [];
116 do {
117 string a = llDumpList2String(
118 llParseStringKeepNulls(
119 llList2String(
120 l,
121  
122 ),
123 ["\""],
124 []
125 ),
126 "\"\""
127 );
128 if(llParseStringKeepNulls(
129 a,
130 [" ", ",", "\n", "\""], []
131 ) !=
132 (list) a
133 ) a = "\"" + a + "\"";
134 v += a;
135 l = llDeleteSubList(l, 0, 0);
136 } while(l != []);
137 return llDumpList2String(v, ",");
138 }
139  
140 ///////////////////////////////////////////////////////////////////////////
141 // Copyright (C) 2015 Wizardry and Steamworks - License: GNU GPLv3 //
142 ///////////////////////////////////////////////////////////////////////////
143 // unescapes a string in conformance with RFC1738
144 string wasURLUnescape(string i) {
145 return llUnescapeURL(
146 llDumpList2String(
147 llParseString2List(
148 llDumpList2String(
149 llParseString2List(
150 i,
151 ["+"],
152 []
153 ),
154 " "
155 ),
156 ["%0D%0A"],
157 []
158 ),
159 "\n"
160 )
161 );
162 }
163  
164 ///////////////////////////////////////////////////////////////////////////
165 // Copyright (C) 2017 Wizardry and Steamworks - License: GNU GPLv3 //
166 ///////////////////////////////////////////////////////////////////////////
167 list wasSetIntersect(list a, list b) {
168 if(llGetListLength(a) == 0) return [];
169 string i = llList2String(a, 0);
170 a = llDeleteSubList(a, 0, 0);
171 if(llListFindList(b, (list)i) == -1)
172 return wasSetIntersect(a, b);
173 return i + wasSetIntersect(a, b);
174 }
175  
176 // configuration data
177 string configuration = "";
178 // callback URL
179 string URL = "";
180 // store message over state.
181 string data = "";
182 // banee
183 string firstname = "";
184 string lastname = "";
185  
186 default {
187 state_entry() {
188 llOwnerSay("[Ban] Starting...");
189 llSetTimerEvent(10);
190 }
191 link_message(integer sender, integer num, string message, key id) {
192 if(id != "configuration") return;
193 llOwnerSay("[Ban] Got configuration...");
194 configuration = message;
195 state listen_group;
196 }
197 timer() {
198 llOwnerSay("[Ban] Requesting configuration...");
199 llMessageLinked(LINK_THIS, 0, "configuration", NULL_KEY);
200 }
201 on_rez(integer num) {
202 llResetScript();
203 }
204 changed(integer change) {
205 if((change & CHANGED_INVENTORY) ||
206 (change & CHANGED_REGION_START) ||
207 (change & CHANGED_OWNER)) {
208 llResetScript();
209 }
210 }
211 state_exit() {
212 llSetTimerEvent(0);
213 }
214 }
215  
216 state listen_group {
217 state_entry() {
218 // DEBUG
219 llOwnerSay("[Ban] Waiting for group messages...");
220 }
221 link_message(integer sender, integer num, string message, key id) {
222 // We only care about notifications now.
223 if(id != "notification")
224 return;
225  
226 // This script only processes group notifications.
227 if(wasKeyValueGet("type", message) != "group")
228 return;
229  
230 // Get the sent message.
231 data = wasURLUnescape(
232 wasKeyValueGet(
233 "message",
234 message
235 )
236 );
237  
238 if(llGetSubString(data, 0, 0) !=
239 wasKeyValueGet("command", configuration))
240 return;
241  
10 office 242 list command = llParseString2List(data,
243 [wasKeyValueGet("command", configuration), " "], []);
8 office 244 if(llList2String(command, 0) != "ban")
245 return;
246  
247 firstname = wasKeyValueGet("firstname", message);
248 lastname = wasKeyValueGet("lastname", message);
249  
250 if(firstname == "" || lastname == "") {
251 data = "And who would yarr be?";
252 state tell;
253 }
254  
255 // Remove command.
256 command = llDeleteSubList(command, 0, 0);
257  
258 // Dump the rest of the message.
259 data = llDumpList2String(command, " ");
260  
261 // Get an URL.
262 state url;
263 }
264 on_rez(integer num) {
265 llResetScript();
266 }
267 changed(integer change) {
268 if((change & CHANGED_INVENTORY) ||
269 (change & CHANGED_REGION_START) ||
270 (change & CHANGED_OWNER)) {
271 llResetScript();
272 }
273 }
274 }
275  
276 state url {
277 state_entry() {
278 // DEBUG
279 llOwnerSay("[Ban] Requesting URL...");
280 llRequestURL();
281 }
282 http_request(key id, string method, string body) {
283 if(method != URL_REQUEST_GRANTED) return;
284 URL = body;
285 // DEBUG
286 llOwnerSay("[Ban] Got URL...");
287 state get_caller_roles;
288 }
289 on_rez(integer num) {
290 llResetScript();
291 }
292 changed(integer change) {
293 if((change & CHANGED_INVENTORY) ||
294 (change & CHANGED_REGION_START) ||
295 (change & CHANGED_OWNER)) {
296 llResetScript();
297 }
298 }
299 }
300  
301 state get_caller_roles {
302 state_entry() {
303 // DEBUG
304 llOwnerSay("[Ban] Searching for caller...");
305 llInstantMessage(
306 wasKeyValueGet(
307 "corrade",
308 configuration
309 ),
310 wasKeyValueEncode(
311 [
312 "command", "getmemberroles",
313 "group", wasURLEscape(
314 wasKeyValueGet(
315 "group",
316 configuration
317 )
318 ),
319 "password", wasURLEscape(
320 wasKeyValueGet(
321 "password",
322 configuration
323 )
324 ),
325 "firstname", firstname,
326 "lastname", lastname,
327 "callback", wasURLEscape(URL)
328 ]
329 )
330 );
331 llSetTimerEvent(60);
332 }
333 http_request(key id, string method, string body) {
334 llHTTPResponse(id, 200, "OK");
335 if(wasKeyValueGet("command", body) != "getmemberroles" ||
336 wasKeyValueGet("success", body) != "True") {
337 // DEBUG
338 llOwnerSay("[Ban] Unable to get member roles: " +
339 wasURLUnescape(
340 wasKeyValueGet("error", body)
341 )
342 );
343 llReleaseURL(URL);
344 state listen_group;
345 }
346  
347 // Dump the roles to a list.
348 list roles = wasCSVToList(
349 wasURLUnescape(
350 wasKeyValueGet("data", body)
351 )
352 );
353  
354 if(llGetListLength(
355 wasSetIntersect(roles,
356 wasCSVToList(
357 wasKeyValueGet(
358 "admin roles", configuration
359 )
360 )
361 )
362 ) == 0) {
363 data = "You ain't got the cojones!";
364 llReleaseURL(URL);
365 state tell;
366 }
367  
368 list banee = llParseString2List(data, [" "], []);
369  
370 firstname = llList2String(banee, 0);
371 lastname = llList2String(banee, 1);
372  
373 if(firstname == "" || lastname == "") {
374 data = "Full name required.";
375 state tell;
376 }
377  
378 // GC
379 banee = [];
380 state get_banee_roles;
381 }
382 timer() {
383 llReleaseURL(URL);
384 state listen_group;
385 }
386 on_rez(integer num) {
387 llResetScript();
388 }
389 changed(integer change) {
390 if((change & CHANGED_INVENTORY) ||
391 (change & CHANGED_REGION_START) ||
392 (change & CHANGED_OWNER)) {
393 llResetScript();
394 }
395 }
396 state_exit() {
397 llSetTimerEvent(0);
398 }
399 }
400  
401 state get_banee_roles {
402 state_entry() {
403 // DEBUG
404 llOwnerSay("[Ban] Searching for banee...");
405 llInstantMessage(
406 wasKeyValueGet(
407 "corrade",
408 configuration
409 ),
410 wasKeyValueEncode(
411 [
412 "command", "getmemberroles",
413 "group", wasURLEscape(
414 wasKeyValueGet(
415 "group",
416 configuration
417 )
418 ),
419 "password", wasURLEscape(
420 wasKeyValueGet(
421 "password",
422 configuration
423 )
424 ),
425 "firstname", firstname,
426 "lastname", lastname,
427 "callback", wasURLEscape(URL)
428 ]
429 )
430 );
431 llSetTimerEvent(60);
432 }
433 http_request(key id, string method, string body) {
434 llHTTPResponse(id, 200, "OK");
435 if(wasKeyValueGet("command", body) != "getmemberroles" ||
436 wasKeyValueGet("success", body) != "True") {
437 // DEBUG
438 llOwnerSay("[Ban] Unable to get member roles: " +
439 wasURLUnescape(
440 wasKeyValueGet("error", body)
441 )
442 );
443 llReleaseURL(URL);
444 state listen_group;
445 }
446  
447 string result = wasURLUnescape(
448 wasKeyValueGet("data", body)
449 );
450  
451 if(result != "" && llListFindList(wasCSVToList(result), (list)"Owners") != -1) {
452 data = "Ejectee is an owner. I'm not gonna open the pod bay doors.";
453 llReleaseURL(URL);
454 state tell;
455 }
456  
457 state ban;
458 }
459 timer() {
460 llReleaseURL(URL);
461 state listen_group;
462 }
463 on_rez(integer num) {
464 llResetScript();
465 }
466 changed(integer change) {
467 if((change & CHANGED_INVENTORY) ||
468 (change & CHANGED_REGION_START) ||
469 (change & CHANGED_OWNER)) {
470 llResetScript();
471 }
472 }
473 state_exit() {
474 llSetTimerEvent(0);
475 }
476 }
477  
478 state ban {
479 state_entry() {
480 // DEBUG
481 llOwnerSay("[Ban] Banning...");
482 llInstantMessage(
483 wasKeyValueGet(
484 "corrade",
485 configuration
486 ),
487 wasKeyValueEncode(
488 [
489 "command", "ban",
490 "group", wasURLEscape(
491 wasKeyValueGet(
492 "group",
493 configuration
494 )
495 ),
496 "password", wasURLEscape(
497 wasKeyValueGet(
498 "password",
499 configuration
500 )
501 ),
502 "action", "ban",
503 "avatars", wasURLEscape(
504 wasListToCSV(
505 [
506 firstname + " " + lastname
507 ]
508 )
509 ),
510 "eject", "True",
511 "callback", wasURLEscape(URL)
512 ]
513 )
514 );
515 llSetTimerEvent(60);
516 }
517 http_request(key id, string method, string body) {
518 llHTTPResponse(id, 200, "OK");
519 llReleaseURL(URL);
520 if(wasKeyValueGet("command", body) != "ban" ||
521 wasKeyValueGet("success", body) != "True") {
522 // DEBUG
523 llOwnerSay("[Ban] Unable to ban member: " +
524 wasURLUnescape(
525 wasKeyValueGet("error", body)
526 )
527 );
528 state listen_group;
529 }
530  
531 data = "Hasta la vista, baby!";
532  
533 state tell;
534 }
535 timer() {
536 llReleaseURL(URL);
537 state listen_group;
538 }
539 on_rez(integer num) {
540 llResetScript();
541 }
542 changed(integer change) {
543 if((change & CHANGED_INVENTORY) ||
544 (change & CHANGED_REGION_START) ||
545 (change & CHANGED_OWNER)) {
546 llResetScript();
547 }
548 }
549 state_exit() {
550 llSetTimerEvent(0);
551 }
552 }
553  
554 state tell {
555 state_entry() {
556 // DEBUG
557 llOwnerSay("[Ban] Sending to group.");
558 llInstantMessage(
559 wasKeyValueGet(
560 "corrade",
561 configuration
562 ),
563 wasKeyValueEncode(
564 [
565 "command", "tell",
566 "group", wasURLEscape(
567 wasKeyValueGet(
568 "group",
569 configuration
570 )
571 ),
572 "password", wasURLEscape(
573 wasKeyValueGet(
574 "password",
575 configuration
576 )
577 ),
578 "entity", "group",
579 "message", wasURLEscape(data)
580 ]
581 )
582 );
583 state listen_group;
584 }
585 }