Annotation of ChivanetModcon/modcon, revision 1.4

1.1       snw         1: #!/usr/bin/env perl
                      2: 
                      3: #
                      4: # ChivaNet Moderator Console
                      5: #  Copyright (C) 2025 Coherent Logic Development LLC
                      6: #
1.4     ! snw         7: # $Id: modcon,v 1.3 2025/01/31 15:39:06 snw Exp $
1.1       snw         8: #
                      9: # Author: Serena Willis <snw@coherent-logic.com>
                     10: #
                     11: # Licensed AGPL-3.0
                     12: #
1.2       snw        13: # $Log: modcon,v $
1.4     ! snw        14: # Revision 1.3  2025/01/31 15:39:06  snw
        !            15: # Minor fixes
        !            16: #
1.3       snw        17: # Revision 1.2  2025/01/31 13:38:51  snw
                     18: # Initial basic functions working
                     19: #
1.2       snw        20: # Revision 1.1.1.1  2025/01/30 19:16:06  snw
                     21: # Initial commit
                     22: #
1.1       snw        23: #
                     24: 
                     25: use REST::Client;
                     26: use JSON;
                     27: use Data::Dumper;
                     28: use Term::ReadKey;
1.4     ! snw        29: use Getopt::Long;
1.1       snw        30: 
1.4     ! snw        31: my $sso_account = '';
1.1       snw        32: my $apikey = '';
                     33: my $online = 0;
                     34: my $modcon_version = '0.0.1';
                     35: my $cnclient = '';
                     36: my $wchar = '';
                     37: my $hchar = '';
                     38: my $wpixels = '';
                     39: my $hpixels = '';
1.2       snw        40: my $account = '';
1.4     ! snw        41: my $mode = '---';
        !            42: my $user = '---';
        !            43: $site = '';
        !            44: 
        !            45: sub list_directories {
        !            46:     return ('sso', 'ras');
        !            47: }
1.2       snw        48: 
1.1       snw        49: sub list_sso_users {
1.4     ! snw        50:     $cnclient->GET("/chivanet/users");       
1.1       snw        51:     my $json = $cnclient->responseContent();
                     52:     my $hashref = decode_json($json);
                     53: 
                     54:     if($hashref->{ok} == 1) {
                     55:         my $arrayref = $hashref->{users};
1.4     ! snw        56:         my @users = @{$arrayref};
        !            57: 
        !            58:         return @users;
1.1       snw        59:     }
                     60:     else {
                     61:         print "RPC error\n";
1.4     ! snw        62:         return ();
1.1       snw        63:     }
                     64: }
                     65: 
                     66: 
1.2       snw        67: sub select_sso_user {
                     68:     my($id) = @_;
                     69: 
                     70:     $cnclient->GET("/chivanet/validate_user?id=$id");
                     71:     my $json = $cnclient->responseContent();
                     72:     my $hashref = decode_json($json);
                     73:     
                     74:     if($hashref->{exists} == 1) {
                     75: 
                     76:         $cnclient->GET("/chivanet/user?id=$id");
                     77:         my $json = $cnclient->responseContent();
                     78:         $account = decode_json($json);
1.4     ! snw        79: 
1.2       snw        80:         return $id;
                     81:     }
                     82:     else {
                     83:         return "---";
1.3       snw        84:     }            
1.2       snw        85: }
                     86: 
                     87: sub select_ras_user {
                     88:     my($id) = @_;
                     89: 
                     90:     $cnclient->GET("/chivanet/validate_sn?id=$id");
                     91:     my $json = $cnclient->responseContent();
                     92:     my $hashref = decode_json($json);
                     93: 
                     94:     if($hashref->{exists} == 1) {
                     95:         return $id;
                     96:     }
                     97:     else {
                     98:         return "---";
                     99:     }
                    100: }
                    101: 
                    102: sub trace_ras_sn {
                    103:     my($user) = @_;
                    104:     
                    105:     $cnclient->GET("/chivanet/trace_sn?id=$user");
                    106:     my $json = $cnclient->responseContent();
                    107:     my $hashref = decode_json($json);
                    108:     my $record = $hashref->{account};
                    109:     my $result = $record->{id};
                    110:     
                    111:     print "RAS screen name $user belongs to SSO account $result; switching to SSO mode\n";
1.3       snw       112:     return $result;   
1.2       snw       113: }
                    114: 
                    115: sub list_ras_sessions {
                    116:     $cnclient->GET("/chivanet/ras_sessions");
                    117:     my $json = $cnclient->responseContent();
                    118:     my $hashref = decode_json($json);
                    119:     my $sessions = $hashref->{sessions};
                    120:     my $arrayref = $sessions->{sessions};
                    121:     my $ct = 0;
                    122:     
                    123:     foreach my $session (@{$arrayref}) {
                    124:         print "$session->{screen_name}\n";
                    125: 
                    126:         $ct = $ct + 1;
                    127: 
                    128:         if($ct > $hchar - 2) {
                    129:             print "ENTER to continue, Q to quit...";
                    130:             my $resp = <STDIN>;
                    131:             chomp($resp);
                    132:             if($resp eq "Q") {
                    133:                 return;
                    134:             }
                    135:             elsif($resp eq "q") {
                    136:                 return;
                    137:             }
                    138:             $ct = 0;
                    139:         }
                    140:     }
                    141:     
1.3       snw       142:     print "$sessions->{count} current sessions\n";    
1.2       snw       143: }
                    144: 
                    145: sub list_ras_screennames {
                    146:     my($id) = @_;
                    147:     
                    148:     $cnclient->GET("/chivanet/ras_screen_names?id=$id");
                    149:     my $json = $cnclient->responseContent();
                    150:     my $hashref = decode_json($json);
                    151:     my $arrayref = $hashref->{screen_names};
                    152:     
1.4     ! snw       153:     return @{$arrayref};   
        !           154: }
1.2       snw       155: 
1.4     ! snw       156: sub list_ras_users {
1.3       snw       157: 
1.4     ! snw       158:     return ();
        !           159:     
1.2       snw       160: }
                    161: 
                    162: sub print_sso_user {
                    163:     my $act = $account->{account};
                    164: 
                    165:     print "\n";
                    166:     print "Username          :  $act->{id}\n";
                    167:     print "Real Name         :  $act->{last_name}, $act->{first_name}\n";
                    168:     print "Display Name      :  $act->{display_name}\n";
                    169:     print "Pronouns          :  $act->{pronouns}\n";
                    170:     print "Profile Image     :  https://chivanet.org$act->{profile_photo}\n";        
                    171:     print "E-Mail Address    :  $act->{email}\n";
                    172:     print "Permission Level  :  $act->{perm_level}\n";
                    173:     print "Created           :  $act->{create_ts}\n";
                    174:     print "Banned            :  $act->{mod_banned}\n\n";
                    175: }
                    176: 
                    177: sub print_ras_user {
                    178:     my($id) = @_;
                    179: 
                    180:     $cnclient->GET("/chivanet/sn_status?id=$id");
                    181:     my $json = $cnclient->responseContent();
                    182:     my $hashref = decode_json($json);
                    183:     my $status = $hashref->{status};
                    184:     my $online = "offline";
                    185:     
                    186:     if($status->{online} == 1) {        
                    187:         $online = "online";        
                    188:     }
                    189:     
                    190:     print "\n";
                    191: 
                    192: 
                    193:     if($online eq "online") {
                    194:         my $sess = $status->{session};
                    195:         my $service = 'AIM';
                    196:         
                    197:         if($sess->{is_icq} == 1) {
                    198:             $service = 'ICQ';
                    199:         }
                    200: 
                    201:         if($sess->{idle_seconds} > 0) {
                    202:             $online = "idle";
                    203:         }
                    204:         
                    205:         print "Screen Name       :  $id [$online]\n";
                    206:         print "Service           :  $service\n";
                    207:         if($online eq "online") {
                    208:             print "Time Online (secs):  $sess->{online_seconds}\n";
                    209:         }
                    210:         else {
                    211:             print "Time Idle (secs)  :  $sess->{idle_seconds}\n";
                    212:         }
                    213:         print "Away Message      :  $sess->{away_message}\n\n";
                    214:     }
                    215:     else {
                    216:         print "Screen Name       :  $id [$online]\n\n";
                    217:     }
                    218:     
                    219: #    print Dumper($status);
                    220:     
                    221: }
                    222: 
                    223: sub send_im {
                    224:     my($user, $msgbody) = @_;
                    225: 
                    226:     print "sending message \"$msgbody\" to $user...";   
                    227:     
                    228:     my $msg = "<font color=red><strong>[ChivaNet Support]:</strong></font> $msgbody";
                    229:     
                    230:     $cnclient->GET("/chivanet/send_im?from=ChivaNet&to=$user&message=$msg");
                    231:     my $json = $cnclient->responseContent();
                    232:     my $hashref = decode_json($json);
                    233:     
                    234:     if($hashref->{status} == 1) {
                    235:         print "[OK]\n";
                    236:     }
                    237:     else {
                    238:         print "[FAIL]\n";
                    239:     }
                    240: }
                    241: 
1.1       snw       242: sub prompt {
                    243:     
                    244:     my $rawcmd = '';
1.4     ! snw       245:     my $pmode = $mode;    
        !           246:     my @path = ();
        !           247:     
1.1       snw       248:     while (1) {
1.4     ! snw       249:         $pmode = lc $mode;
        !           250:         my $prompt = '';
        !           251:         if($pmode ne "---") {
        !           252:             if($user eq '---') {
        !           253:                 $prompt = "$sso_account\@$site:[/$pmode]\$ ";
        !           254:             }
        !           255:             else {
        !           256:                 $prompt = "$sso_account\@$site:[/$pmode/$user]\$ ";
        !           257:             }
        !           258:         }
        !           259:         else {
        !           260:             $prompt = "$sso_account\@$site:[/]\$ ";
        !           261:         }
        !           262:         print $prompt;
1.1       snw       263:         my $line = <STDIN>;
                    264:         chomp($line);
                    265:         $rawcmd = $line;
                    266:         
                    267:         my @cmd = split(' ', $rawcmd);
                    268:         
1.4     ! snw       269:         if ($cmd[0] eq "exit" || $cmd[0] eq "logout" || $cmd[0] eq "quit" || $cmd[0] eq "bye") {
1.1       snw       270:             return;
                    271:         }
1.4     ! snw       272:         elsif ($cmd[0] eq "pwd") {
        !           273:             my $pstr = join('/', @path);
        !           274:             print "/$pstr\n";        
        !           275:         }        
        !           276:         elsif ($cmd[0] eq "cd") {
        !           277:             my $abspath = false;
        !           278: 
        !           279:             if($cmd[1] eq "..") {
        !           280:                 if(@path) {
        !           281:                     if($#path == 1) {
        !           282:                         $user = "---";
        !           283:                         @path = ($pmode);
        !           284:                     }
        !           285:                     elsif($#path == 0) {
        !           286:                         $mode = "---";
        !           287:                         @path = ();
        !           288:                     }
        !           289:                 }
        !           290:                 else {
        !           291:                     print "already in root directory\n";
        !           292:                 }
1.2       snw       293:             }
                    294:             else {
1.4     ! snw       295:                 my @oldpath = @path;
        !           296:                 @path = split('/', $cmd[1]);
        !           297:             
        !           298:                 if(substr($cmd[1], 0, 1) eq '/') {
        !           299:                     $abspath = true;
        !           300:                     shift @path;                
        !           301:                 }
        !           302:                 
        !           303:                 if($abspath eq true || !@oldpath) {
        !           304:                     if($#path == 0) {
        !           305:                         # mode, no user
        !           306:                         if($path[0] eq "sso") {
        !           307:                             $mode = "SSO";
        !           308:                             $user = "---";
        !           309:                             @path = ('sso');
        !           310:                         }
        !           311:                         elsif($path[0] eq "ras") {
        !           312:                             $mode = "RAS";
        !           313:                             $user = "---";
        !           314:                             @path = ('ras');
        !           315:                         }
        !           316:                         else {
        !           317:                             print "$path[0]:  no such directory exists\n";
        !           318:                             @path = @oldpath;
        !           319:                         }
        !           320:                     }
        !           321:                     elsif($#path == 1) {
        !           322:                         # mode and user
        !           323:                         if($path[0] eq "sso") {
        !           324:                             $user = select_sso_user($path[1]);                          
        !           325:                             if($user eq "---") {
        !           326:                                 print "$path[1]:  no such file exists in $pmode\n";
        !           327:                                 @path = @oldpath;
        !           328:                             }
        !           329:                             else {
        !           330:                                 $mode = "SSO";
        !           331:                                 @path = ('sso', $user);
        !           332:                             }
        !           333:                         }
        !           334:                         elsif($path[0] eq "ras") {
        !           335:                             $user = select_ras_user($path[1]);
        !           336:                             if($user eq "---") {
        !           337:                                 print "$path[1]:  no such file exists in $pmode\n";
        !           338:                                 @path = @oldpath;
        !           339:                             }
        !           340:                             else {
        !           341:                                 $mode = "RAS";
        !           342:                                 @path = ('ras', $user);
        !           343:                             }
        !           344:                         }
        !           345:                         else {
        !           346:                             print "$path[0]:  no such directory exists\n";
        !           347:                             @path = @oldpath;
        !           348:                         }
        !           349:                     }
        !           350:                     else {
        !           351:                         if($cmd[1] eq '/') {
        !           352:                             $mode = '---';
        !           353:                             $user = '---';
        !           354:                             @path = ();
        !           355:                         }
        !           356:                         else {
        !           357:                             print "cd:  invalid path specification\n";
        !           358:                             @path = @oldpath;
        !           359:                         }
        !           360:                     }
1.2       snw       361:                 }
                    362:                 else {
1.4     ! snw       363:                     if($#oldpath == 1) {
        !           364:                         print "invalid path specification\n";
        !           365:                         @path = @oldpath;
        !           366:                     }
        !           367:                     else {
        !           368:                         if($#path == 0) {
        !           369:                             if($pmode eq "sso") {
        !           370:                                 $user = select_sso_user($path[0]);                          
        !           371:                                 if($user eq "---") {
        !           372:                                     print "$path[0]:  no such file exists in $pmode\n";
        !           373:                                     @path = @oldpath;
        !           374:                                 }
        !           375:                                 else {
        !           376:                                     $mode = "SSO";
        !           377:                                     @path = ('sso', $user);
        !           378:                                 }
        !           379:                             }
        !           380:                             elsif($pmode eq "ras") {
        !           381:                                 $user = select_ras_user($path[0]);                          
        !           382:                                 if($user eq "---") {
        !           383:                                     print "$path[0]:  no such file exists in $pmode\n";
        !           384:                                     @path = @oldpath;
        !           385:                                 }
        !           386:                                 else {
        !           387:                                     $mode = "SSO";
        !           388:                                     @path = ('sso', $user);
        !           389:                                 }
        !           390:                             }
        !           391:                             else {
        !           392:                                 print "$path[0]:  no such directory exists\n";
        !           393:                                 @path = @oldpath;                            
        !           394:                             }
        !           395:                         }
        !           396:                         else {
        !           397:                             print "invalid path specification\n";
        !           398:                             @path = @oldpath;
        !           399:                         }
        !           400:                     }
        !           401:                 }                                         
        !           402:             } # if ..                                    
        !           403:             
1.2       snw       404:         }
                    405:         elsif ($cmd[0] eq "im") {
                    406:             if($mode eq "RAS") {
                    407:                 if($user ne "---") {
                    408:                     my @msga = @cmd[1..$#cmd];
                    409:                     my $msgbody = join(' ', @msga);                    
                    410:                     send_im($user, $msgbody);
                    411:                 }
                    412:                 else {
1.4     ! snw       413:                     print "im:  no user selected\n";
1.2       snw       414:                 }
                    415:             }
                    416:             else {
1.4     ! snw       417:                 print "im:  invalid command outside of ras directory\n";
1.2       snw       418:             }
                    419:         }    
                    420:         elsif ($cmd[0] eq "trace") {
                    421:             if($mode eq "RAS") {
                    422:                 if($user ne "---") {                    
                    423:                     my $id = trace_ras_sn($user);
                    424:                     $mode = "SSO";
                    425:                     $user = select_sso_user($id);
                    426:                     print_sso_user;
                    427:                 }
                    428:                 else {
1.4     ! snw       429:                     print "trace:  no user select\n";
1.2       snw       430:                 }
                    431:             }
                    432:             else {
1.4     ! snw       433:                 print "trace:  invalid command outside of ras directory\n";
1.2       snw       434:             }
                    435:         }
                    436:         elsif ($cmd[0] eq "field") {
1.4     ! snw       437:             if($mode eq "SSO") {
        !           438:                 if($user ne "---") {
        !           439:                     my $act = $account->{account};
        !           440:                     print "$user\-\>$cmd[1]:  $act->{$cmd[1]}\n";
        !           441:                 }
        !           442:                 else {
        !           443:                     print "field:  no user selected\n";
        !           444:                 }
1.2       snw       445:             }
                    446:             else {
1.4     ! snw       447:                 print "field:  command invalid outside of ras directory\n";
1.2       snw       448:             }
                    449:         }
1.4     ! snw       450:         elsif ($cmd[0] eq "ls") {
        !           451:             my @entries = ();
        !           452:             
        !           453:             if($mode eq "---") {
        !           454:                 @entries = list_directories();
1.1       snw       455:             }
                    456:             else {
1.4     ! snw       457:                 if($user ne "---") {
        !           458:                     if($mode eq "SSO") {
        !           459:                         print_sso_user;
        !           460:                     }
        !           461:                     elsif($mode eq "RAS") {
        !           462:                         print_ras_user($user);
        !           463:                     }                                
1.2       snw       464:                 }
                    465:                 else {
1.4     ! snw       466:                     if($mode eq "SSO") {
        !           467:                         @entries = list_sso_users();
1.2       snw       468:                     }
                    469:                     else {
1.4     ! snw       470:                         @entries = list_ras_users();
1.2       snw       471:                     }
                    472:                 }
1.4     ! snw       473:             }
        !           474: 
        !           475:             if(@entries) {
        !           476:                 my $col = 0;
        !           477:                 my $pstr = join('/', @path);
        !           478:                 my $pfin = "/$pstr";
        !           479:                 my @sorted = sort(@entries);
        !           480:                 my $maxlen = 0;
        !           481:                 foreach my $entry (@sorted) {
        !           482:                     my $len = length($entry);
        !           483:                     if ($len > $maxlen) {
        !           484:                         $maxlen = $len;
        !           485:                     }
1.2       snw       486:                 }
1.4     ! snw       487:                 $maxlen = $maxlen + 2;
        !           488:                 my $ct = $#sorted + 1;
        !           489:                 foreach my $entry (@sorted) {
        !           490:                     if($col + $maxlen >= $wchar) {
        !           491:                         print "\n";
        !           492:                         $col = 0;
        !           493:                     }
        !           494:                     else {
        !           495:                         printf("%-$maxlen\s", $entry);
        !           496:                         $col = $col + $maxlen;
        !           497:                     }                    
        !           498:                 }                
        !           499:                 print "\n\t$ct items in $pfin\n";
1.2       snw       500:             }
1.4     ! snw       501: 
1.1       snw       502:         }
                    503:         else {
1.4     ! snw       504:             print "$cmd[0]: command not found\n";
1.1       snw       505:         }
                    506:     }
                    507: 
                    508: }
                    509: 
                    510: sub main {  
1.4     ! snw       511:     ($wchar, $hchar, $wpixels, $hpixels) = GetTerminalSize();    
1.1       snw       512: 
1.4     ! snw       513:     GetOptions("site=s" => \$site) or die "error in command line arguments";
        !           514:     
        !           515:     if($site eq "") {
        !           516:         print "modcon: must supply -site command-line argument\n";
        !           517:         return;
        !           518:     }
1.1       snw       519:     
                    520:     $cnclient = REST::Client->new({
1.4     ! snw       521:         host => "https://$site/rest/api",
1.1       snw       522:         timeout => 10});
                    523:     
                    524:     print "ChivaNet MODCON $modcon_version\n";
                    525:     print " Copyright (C) 2025 Coherent Logic Development LLC\n\n";
1.4     ! snw       526: 
        !           527:     print "username: ";
        !           528:     $sso_account = <STDIN>;
        !           529:     chomp($sso_account);
        !           530: 
        !           531:     print "password: ";
        !           532:     ReadMode('noecho');
        !           533:     my $password = <STDIN>;
        !           534:     chomp($password);
        !           535:     ReadMode('normal');
        !           536: 
        !           537:     
        !           538:     my $params = $cnclient->buildQuery([username => $sso_account, password => $password]);
        !           539:     my $result = $cnclient->POST("/chivanet/modcon_auth", substr($params, 1), {'Content-type' => 'application/x-www-form-urlencoded'});
        !           540:     my $http_response = $result->{_res};
        !           541:     my $json = $http_response->{_content};
        !           542:     my $apiresult = decode_json($json);
1.1       snw       543:     
1.4     ! snw       544:     if($apiresult->{ok} == 1) {
        !           545:         $cnclient->addHeader('Authorization', "Apikey $apiresult->{token}");
        !           546: 
        !           547:         select_sso_user $sso_account;
        !           548:         my $act = $account->{account};
        !           549:         
        !           550:         print "\n\nWelcome to MODCON, $act->{display_name}!\n\n";
        !           551:         
        !           552:         prompt();
        !           553:     }
        !           554:     else {
        !           555:         print "\nerror: $apiresult->{error}\n";
        !           556:         return;
        !           557:     }
        !           558: 
1.1       snw       559:     print "Goodbye.\n"
                    560: }
                    561: 
                    562: main();

FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>