/[sudobot]/trunk/deploy-commands.js
ViewVC logotype

Diff of /trunk/deploy-commands.js

Parent Directory Parent Directory | Revision Log Revision Log | View Patch Patch

revision 63 by rakin, Mon Jul 29 17:28:26 2024 UTC revision 268 by rakin, Mon Jul 29 17:29:17 2024 UTC
# Line 1  Line 1 
1  #!/bin/node  #!/bin/node
2    
3  const { SlashCommandBuilder } = require('@discordjs/builders');  const { SlashCommandBuilder, ContextMenuCommandBuilder } = require('@discordjs/builders');
4  const { REST } = require('@discordjs/rest');  const { REST } = require('@discordjs/rest');
5  const { Routes } = require('discord-api-types/v9');  const { Routes } = require('discord-api-types/v9');
6  const { config } = require('dotenv');  const { config } = require('dotenv');
7  const { existsSync } = require('fs');  const { existsSync } = require('fs');
8  const { Permissions } = require('discord.js');  const { Permissions, ApplicationCommand } = require('discord.js');
9  const path = require('path');  const path = require('path');
10    const { ActivityType, ApplicationCommandType } = require('discord-api-types/v10');
11    
12  if (existsSync(path.join(__dirname, '.env'))) {  if (existsSync(path.join(__dirname, '.env'))) {
13      config();      config();
# Line 17  else { Line 18  else {
18    
19  const { CLIENT_ID, GUILD_ID, TOKEN } = process.env;  const { CLIENT_ID, GUILD_ID, TOKEN } = process.env;
20    
21  const commands = [  let commands = [
22      // SETTINGS      // SETTINGS
23      new SlashCommandBuilder().setName('help').setDescription('A short documentation about the commands')      new SlashCommandBuilder().setName('help').setDescription('A short documentation about the commands')
24          .addStringOption(option => option.setName('command').setDescription("The command")),          .addStringOption(option => option.setName('command').setDescription("The command")),
25      new SlashCommandBuilder().setName('about').setDescription('Show information about the bot'),      new SlashCommandBuilder().setName('about').setDescription('Show information about the bot'),
26        new SlashCommandBuilder().setName('eval').setDescription('Execute raw code in the runtime environment')
27            .addStringOption(option => option.setName('code').setDescription('The code to be executed').setRequired(true)),
28      new SlashCommandBuilder().setName('system').setDescription('Show the system status'),      new SlashCommandBuilder().setName('system').setDescription('Show the system status'),
29      new SlashCommandBuilder().setName('restart').setDescription('Restart the system'),      new SlashCommandBuilder().setName('restart').setDescription('Restart the system'),
30        new SlashCommandBuilder().setName('setstatus').setDescription('Set status for the bot system')
31            .addStringOption(option => option.setName('activity').setDescription('The activity').setRequired(true))
32            .addStringOption(option => option.setName('status').setDescription('The status').setChoices(...[
33                {
34                    name: 'Online',
35                    value: 'online'
36                },
37                {
38                    name: 'Idle',
39                    value: 'idle'
40                },
41                {
42                    name: 'DND',
43                    value: 'dnd'
44                },
45                {
46                    name: 'Invisible',
47                    value: 'invisible'
48                }
49            ]))
50            .addStringOption(option => option.setName('type').setDescription('The activity type').setChoices(...[
51                {
52                    name: 'Playing',
53                    value: 'PLAYING'
54                },
55                {
56                    name: 'Watching',
57                    value: 'WATCHING'
58                },
59                {
60                    name: 'Competing',
61                    value: 'COMPETING'
62                }
63            ])),
64        new SlashCommandBuilder().setName('config').setDescription('View/change the system settings for this server')
65            .addStringOption(option => option.setName('key').setDescription('The setting key (e.g. spam_filter.enabled)').setRequired(true))
66            .addStringOption(option => option.setName('value').setDescription('New value for the setting')),
67    
68      // INFORMATION      // INFORMATION
69      new SlashCommandBuilder().setName('stats').setDescription('Show the server statistics'),      new SlashCommandBuilder().setName('stats').setDescription('Show the server statistics'),
# Line 32  const commands = [ Line 72  const commands = [
72      new SlashCommandBuilder().setName('avatar').setDescription('Show someone\'s avatar')      new SlashCommandBuilder().setName('avatar').setDescription('Show someone\'s avatar')
73          .addUserOption(option => option.setName('user').setDescription('The user')),          .addUserOption(option => option.setName('user').setDescription('The user')),
74      new SlashCommandBuilder().setName('rolelist').setDescription('List all roles or show info about a role')      new SlashCommandBuilder().setName('rolelist').setDescription('List all roles or show info about a role')
75          .addUserOption(option => option.setName('role').setDescription('The role')),          .addRoleOption(option => option.setName('role').setDescription('The role'))
76            .addIntegerOption(option => option.setName('page').setDescription('The page number')),
77    
78      // AUTOMATION      // AUTOMATION
79      new SlashCommandBuilder().setName('ballot').setDescription('Ballot engine')      new SlashCommandBuilder().setName('ballot').setDescription('Ballot engine')
# Line 49  const commands = [ Line 90  const commands = [
90                  .setDescription('Get information/stats about a ballot')                  .setDescription('Get information/stats about a ballot')
91                  .addIntegerOption(option => option.setName('id').setDescription('The ballot ID'))),                  .addIntegerOption(option => option.setName('id').setDescription('The ballot ID'))),
92                    
93        new SlashCommandBuilder().setName('embed').setDescription('Make an embed')
94            .addSubcommand(subcmd =>
95                 subcmd.setName("send").setDescription("Make and send an embed")
96                    .addStringOption(option => option.setName('author_name').setDescription('The embed author name'))
97                    .addStringOption(option => option.setName('author_iconurl').setDescription('The embed author icon URL'))
98                    .addStringOption(option => option.setName('title').setDescription('The embed title'))
99                    .addStringOption(option => option.setName('description').setDescription('The embed description'))
100                    .addStringOption(option => option.setName('thumbnail').setDescription('The embed thumbnail URL'))
101                    .addStringOption(option => option.setName('image').setDescription('The embed image attachment URL'))
102                    .addStringOption(option => option.setName('video').setDescription('The embed video attachment URL'))
103                    .addStringOption(option => option.setName('footer_text').setDescription('The embed footer text'))
104                    .addStringOption(option => option.setName('footer_iconurl').setDescription('The embed footer icon URL'))
105                    .addStringOption(option => option.setName('timestamp').setDescription('The embed timestamp, use \'current\' to set current date'))
106                    .addStringOption(option => option.setName('color').setDescription('The embed color (default is #007bff)'))
107                    .addStringOption(option => option.setName('url').setDescription('The embed URL'))
108                    .addStringOption(option => option.setName('fields').setDescription('The embed fields, should be in `Field 1: Value 1, Field 2: Value 2` format'))
109            )
110            .addSubcommand(subcmd =>
111                 subcmd.setName("schema").setDescription("Make and send an embed schema representation")
112                    .addStringOption(option => option.setName('author_name').setDescription('The embed author name'))
113                    .addStringOption(option => option.setName('author_iconurl').setDescription('The embed author icon URL'))
114                    .addStringOption(option => option.setName('title').setDescription('The embed title'))
115                    .addStringOption(option => option.setName('description').setDescription('The embed description'))
116                    .addStringOption(option => option.setName('thumbnail').setDescription('The embed thumbnail URL'))
117                    .addStringOption(option => option.setName('image').setDescription('The embed image attachment URL'))
118                    .addStringOption(option => option.setName('video').setDescription('The embed video attachment URL'))
119                    .addStringOption(option => option.setName('footer_text').setDescription('The embed footer text'))
120                    .addStringOption(option => option.setName('footer_iconurl').setDescription('The embed footer icon URL'))
121                    .addStringOption(option => option.setName('timestamp').setDescription('The embed timestamp, use \'current\' to set current date'))
122                    .addStringOption(option => option.setName('color').setDescription('The embed color (default is #007bff)'))
123                    .addStringOption(option => option.setName('url').setDescription('The embed URL'))
124                    .addStringOption(option => option.setName('fields').setDescription('The embed fields, should be in `Field 1: Value 1, Field 2: Value 2` format'))
125            )
126            .addSubcommand(subcmd =>
127                 subcmd.setName("build").setDescription("Build an embed from schema")
128                    .addStringOption(option => option.setName('json_schema').setDescription('The embed JSON schema'))
129            ),
130    
131      new SlashCommandBuilder().setName('queues').setDescription('List all queued jobs'),      new SlashCommandBuilder().setName('queues').setDescription('List all queued jobs'),
132    
133      new SlashCommandBuilder().setName('schedule').setDescription('Schedule a message for sending later')      new SlashCommandBuilder().setName('schedule').setDescription('Schedule a message for sending later')
# Line 131  const commands = [ Line 210  const commands = [
210      new SlashCommandBuilder().setName('afk').setDescription('Set your AFK status')      new SlashCommandBuilder().setName('afk').setDescription('Set your AFK status')
211          .addStringOption(option => option.setName('reason').setDescription("The reason for going AFK")),          .addStringOption(option => option.setName('reason').setDescription("The reason for going AFK")),
212            
213        new SlashCommandBuilder().setName('hash').setDescription('Generate hash for a string (text) data')
214            .addStringOption(option => option.setName('content').setDescription("The content to be hashed").setRequired(true))
215            .addStringOption(option =>
216                option
217                .setName('algorithm')
218                .setDescription("Hash algorithm")
219                .setChoices(
220                    {
221                        name: 'SHA1',
222                        value: 'sha1'
223                    },
224                    {
225                        name: 'SHA256',
226                        value: 'sha256'
227                    },
228                    {
229                        name: 'SHA512',
230                        value: 'sha512'
231                    },
232                    {
233                        name: 'MD5',
234                        value: 'md5'
235                    },
236                )
237            )
238            .addStringOption(option =>
239                option
240                .setName('digest')
241                .setDescription("Digest mode")
242                .setChoices(
243                    {
244                        name: 'HEX',
245                        value: 'hex'
246                    },
247                    {
248                        name: 'Base64',
249                        value: 'base64'
250                    },
251                    {
252                        name: 'Base64 URL',
253                        value: 'base64url'
254                    },
255                )
256            ),
257        
258      new SlashCommandBuilder().setName('announce').setDescription('Announce something')      new SlashCommandBuilder().setName('announce').setDescription('Announce something')
259          .addStringOption(option => option.setName('content').setDescription("The announcemnt message content")),          .addStringOption(option => option.setName('content').setDescription("The announcemnt message content")),
260    
261      // MODERATION      // MODERATION
262        new SlashCommandBuilder().setName('antijoin').setDescription('Enable antijoin system which will kick any new users joining the server'),
263    
264      new SlashCommandBuilder().setName('ban').setDescription('Ban a user')      new SlashCommandBuilder().setName('ban').setDescription('Ban a user')
265          .addUserOption(option => option.setName('user').setDescription("The user").setRequired(true))          .addUserOption(option => option.setName('user').setDescription("The user").setRequired(true))
266          .addStringOption(option => option.setName('reason').setDescription("The reason for banning this user"))          .addStringOption(option => option.setName('reason').setDescription("The reason for banning this user"))
267          .addIntegerOption(option => option.setName('days').setDescription("The days old messages to delete of this user").setMinValue(0).setMaxValue(7)),          .addIntegerOption(option => option.setName('days').setDescription("The days old messages to delete of this user").setMinValue(0).setMaxValue(7)),
268    
269        new SlashCommandBuilder().setName('softban').setDescription('Softban a user')
270            .addUserOption(option => option.setName('user').setDescription("The user").setRequired(true))
271            .addStringOption(option => option.setName('reason').setDescription("The reason for softbanning this user"))
272            .addIntegerOption(option => option.setName('days').setDescription("The days old messages to delete of this user (default is 7)").setMinValue(0).setMaxValue(7)),
273    
274        new SlashCommandBuilder().setName('tempban').setDescription('Temporarily ban a user')
275            .addUserOption(option => option.setName('user').setDescription("The user").setRequired(true))
276            .addStringOption(option => option.setName('time').setDescription("TBan duration").setRequired(true))
277            .addStringOption(option => option.setName('reason').setDescription("The reason for softbanning this user"))
278            .addIntegerOption(option => option.setName('days').setDescription("The days old messages to delete of this user (default is 7)").setMinValue(0).setMaxValue(7)),
279    
280      new SlashCommandBuilder().setName('massban').setDescription('Ban multiple users')      new SlashCommandBuilder().setName('massban').setDescription('Ban multiple users')
281          .addStringOption(option => option.setName('users').setDescription("The user IDs (separated by spaces)").setRequired(true))          .addStringOption(option => option.setName('users').setDescription("The user IDs (separated by spaces)").setRequired(true))
282          .addStringOption(option => option.setName('reason').setDescription("The reason for banning"))          .addStringOption(option => option.setName('reason').setDescription("The reason for banning"))
# Line 149  const commands = [ Line 286  const commands = [
286          .addUserOption(option => option.setName('member').setDescription("The member").setRequired(true))          .addUserOption(option => option.setName('member').setDescription("The member").setRequired(true))
287          .addStringOption(option => option.setName('reason').setDescription("The reason for kicking this user")),          .addStringOption(option => option.setName('reason').setDescription("The reason for kicking this user")),
288    
289      new SlashCommandBuilder().setName('bean').setDescription('Bean a member')      new SlashCommandBuilder().setName('shot').setDescription('Give a shot to a member')
290          .addUserOption(option => option.setName('member').setDescription("The member").setRequired(true))          .addUserOption(option => option.setName('member').setDescription("The member").setRequired(true))
291          .addStringOption(option => option.setName('reason').setDescription("The reason for beaning this user")),          .addStringOption(option => option.setName('reason').setDescription("The reason for giving shot to this user")),
292    
293      new SlashCommandBuilder().setName('warn').setDescription('Warn a member')      new SlashCommandBuilder().setName('warn').setDescription('Warn a member')
294          .addUserOption(option => option.setName('member').setDescription("The member").setRequired(true))          .addUserOption(option => option.setName('member').setDescription("The member").setRequired(true))
295          .addStringOption(option => option.setName('reason').setDescription("The reason for warning this user")),          .addStringOption(option => option.setName('reason').setDescription("The reason for warning this user")),
296    
297      new SlashCommandBuilder().setName('note').setDescription('Take a note for a member')      new SlashCommandBuilder().setName('note').setDescription('Take a note for a user')
298          .addUserOption(option => option.setName('member').setDescription("The member").setRequired(true))          .addUserOption(option => option.setName('user').setDescription("The user").setRequired(true))
299          .addStringOption(option => option.setName('note').setDescription("The note content").setRequired(true)),          .addStringOption(option => option.setName('note').setDescription("The note content").setRequired(true)),
300    
301      new SlashCommandBuilder().setName('mute').setDescription('Mute a member')      new SlashCommandBuilder().setName('mute').setDescription('Mute a member')
302          .addUserOption(option => option.setName('member').setDescription("The member").setRequired(true))          .addUserOption(option => option.setName('member').setDescription("The member").setRequired(true))
303          .addStringOption(option => option.setName('reason').setDescription("The reason for muting this user"))          .addStringOption(option => option.setName('reason').setDescription("The reason for muting this user"))
304          .addStringOption(option => option.setName('time').setDescription("Mute duration")),          .addStringOption(option => option.setName('time').setDescription("Mute duration"))
305            .addBooleanOption(option => option.setName('hardmute').setDescription("Specify if the system should take out all roles of the user during the mute")),
306    
307      new SlashCommandBuilder().setName('unmute').setDescription('Unmute a member')      new SlashCommandBuilder().setName('unmute').setDescription('Unmute a member')
308          .addUserOption(option => option.setName('member').setDescription("The member").setRequired(true)),          .addUserOption(option => option.setName('member').setDescription("The member").setRequired(true)),
# Line 172  const commands = [ Line 310  const commands = [
310      new SlashCommandBuilder().setName('unban').setDescription('Unban a user')      new SlashCommandBuilder().setName('unban').setDescription('Unban a user')
311          .addUserOption(option => option.setName('user').setDescription("The user").setRequired(true)),          .addUserOption(option => option.setName('user').setDescription("The user").setRequired(true)),
312    
313      new SlashCommandBuilder().setName('warndel').setDescription('Delete a warning')      new SlashCommandBuilder().setName('warning').setDescription('Clear, remove or view warnings')
314          .addNumberOption(option => option.setName('id').setDescription("The warning ID").setRequired(true)),          .addSubcommand(subcmd => {
315                return subcmd.setName('view').setDescription('View information about a warning').addNumberOption(option => option.setName('id').setDescription("The warning ID").setRequired(true));
316      new SlashCommandBuilder().setName('warning').setDescription('Get information about a warning')          })
317          .addNumberOption(option => option.setName('id').setDescription("The warning ID").setRequired(true)),          .addSubcommand(subcmd => {
318                return subcmd.setName('remove').setDescription('Remove a warning').addNumberOption(option => option.setName('id').setDescription("The warning ID").setRequired(true));
319            })
320            .addSubcommand(subcmd => {
321                return subcmd.setName('list').setDescription('List warnings for a user').addUserOption(option => option.setName('user').setDescription("The user").setRequired(true));
322            })
323            .addSubcommand(subcmd => {
324                return subcmd.setName('clear').setDescription('Clear all warnings for a user').addUserOption(option => option.setName('user').setDescription("The user").setRequired(true));
325            }),
326    
327      new SlashCommandBuilder().setName('noteget').setDescription('Get information about a note')      new SlashCommandBuilder().setName('noteget').setDescription('Get information about a note')
328          .addNumberOption(option => option.setName('id').setDescription("The note ID").setRequired(true)),          .addNumberOption(option => option.setName('id').setDescription("The note ID").setRequired(true)),
# Line 184  const commands = [ Line 330  const commands = [
330      new SlashCommandBuilder().setName('notedel').setDescription('Delete a note')      new SlashCommandBuilder().setName('notedel').setDescription('Delete a note')
331          .addNumberOption(option => option.setName('id').setDescription("The note ID").setRequired(true)),          .addNumberOption(option => option.setName('id').setDescription("The note ID").setRequired(true)),
332    
     new SlashCommandBuilder().setName('warnings').setDescription('Fetch all warnings')  
         .addUserOption(option => option.setName('member').setDescription("Show warnings for only this member")),  
   
333      new SlashCommandBuilder().setName('notes').setDescription('Fetch all notes for a user')      new SlashCommandBuilder().setName('notes').setDescription('Fetch all notes for a user')
334          .addUserOption(option => option.setName('member').setDescription("The member").setRequired(true)),          .addUserOption(option => option.setName('user').setDescription("The user").setRequired(true)),
335    
336      new SlashCommandBuilder().setName('history').setDescription('Fetch all moderation history for a user')      new SlashCommandBuilder().setName('history').setDescription('Fetch all moderation history for a user')
337          .addUserOption(option => option.setName('user').setDescription("The user").setRequired(true)),          .addUserOption(option => option.setName('user').setDescription("The user").setRequired(true)),
338    
339      new SlashCommandBuilder().setName('clear').setDescription('Clear all messages in the current channel for a user')      new SlashCommandBuilder().setName('clear').setDescription('Clear messages in bulk')
340          .addUserOption(option => option.setName('member').setDescription("The member").setRequired(true)),          .addUserOption(option => option.setName('user').setDescription("The user"))
341            .addIntegerOption(option => option.setName('count').setDescription("The amount of messages to delete").setMaxValue(400).setMinValue(0))
342            .addChannelOption(option => option.setName('channel').setDescription("The channel where the messages will be deleted")),
343    
344      new SlashCommandBuilder().setName('echo').setDescription('Re-send a message from the bot system')      new SlashCommandBuilder().setName('echo').setDescription('Re-send a message from the bot system')
345          .addStringOption(option => option.setName('content').setDescription("The message content").setRequired(true))          .addStringOption(option => option.setName('content').setDescription("The message content").setRequired(true))
# Line 207  const commands = [ Line 352  const commands = [
352      new SlashCommandBuilder().setName('setchperms').setDescription('Set permissions for channels')      new SlashCommandBuilder().setName('setchperms').setDescription('Set permissions for channels')
353          .addChannelOption(option => option.setName('channel').setDescription("The channel that (or its children) will be updated").setRequired(true))          .addChannelOption(option => option.setName('channel').setDescription("The channel that (or its children) will be updated").setRequired(true))
354          .addRoleOption(option => option.setName('role').setDescription("Lock channel for the given role.").setRequired(true))          .addRoleOption(option => option.setName('role').setDescription("Lock channel for the given role.").setRequired(true))
355          .addStringOption(option => option.setName('permission').setDescription("The permission codename").setRequired(true))          .addStringOption(option => option.setName('permission').setDescription("The permission codename").setRequired(true).setAutocomplete(true))
356          .addStringOption(option => option.setName('value').setDescription("The permission value").addChoices(...[          .addStringOption(option => option.setName('value').setDescription("The permission value").addChoices(...[
357              {              {
358                  name: 'Allow',                  name: 'Allow',
# Line 224  const commands = [ Line 369  const commands = [
369          ]).setRequired(true)),          ]).setRequired(true)),
370    
371      new SlashCommandBuilder().setName('lockall').setDescription('Lock multiple channels')      new SlashCommandBuilder().setName('lockall').setDescription('Lock multiple channels')
372            .addStringOption(option => option.setName('channels').setDescription("The channels, must be separated by spaces"))
373          .addRoleOption(option => option.setName('role').setDescription("Lock channels for the given role. Default is @everyone"))          .addRoleOption(option => option.setName('role').setDescription("Lock channels for the given role. Default is @everyone"))
374          .addBooleanOption(option => option.setName('raid').setDescription("The raid protected channels will be locked. Default is `false`")),          .addBooleanOption(option => option.setName('raid').setDescription("The raid protected channels will be locked. Default is `false`")),
375    
376      new SlashCommandBuilder().setName('unlockall').setDescription('Unlock multiple channels')      new SlashCommandBuilder().setName('unlockall').setDescription('Unlock multiple channels')
377            .addStringOption(option => option.setName('channels').setDescription("The channels, must be separated by spaces"))
378          .addRoleOption(option => option.setName('role').setDescription("Unlock channels for the given role. Default is @everyone"))          .addRoleOption(option => option.setName('role').setDescription("Unlock channels for the given role. Default is @everyone"))
379          .addBooleanOption(option => option.setName('force').setDescription("Force set the channel permissions to `true`"))          .addBooleanOption(option => option.setName('force').setDescription("Force set the channel permissions to `true`"))
380          .addBooleanOption(option => option.setName('raid').setDescription("The raid protected channels will be unlocked. Default is `false`")),          .addBooleanOption(option => option.setName('raid').setDescription("The raid protected channels will be unlocked. Default is `false`")),
# Line 240  const commands = [ Line 387  const commands = [
387      new SlashCommandBuilder().setName('send').setDescription('Send a DM to a user')      new SlashCommandBuilder().setName('send').setDescription('Send a DM to a user')
388          .addStringOption(option => option.setName('content').setDescription("The message content").setRequired(true))          .addStringOption(option => option.setName('content').setDescription("The message content").setRequired(true))
389          .addUserOption(option => option.setName('member').setDescription("The member").setRequired(true)),          .addUserOption(option => option.setName('member').setDescription("The member").setRequired(true)),
390    
391    
392        new SlashCommandBuilder().setName('appeal').setDescription('Send us a messages about a punishment appeal')
393    ].map(command => command.toJSON());
394    
395    let contextMenuCommands = [
396        new ContextMenuCommandBuilder().setName('Moderation History').setType(ApplicationCommandType.User),
397        new ContextMenuCommandBuilder().setName('Ban').setType(ApplicationCommandType.User),
398        new ContextMenuCommandBuilder().setName('Shot').setType(ApplicationCommandType.User),
399        new ContextMenuCommandBuilder().setName('Kick').setType(ApplicationCommandType.User),
400  ].map(command => command.toJSON());  ].map(command => command.toJSON());
401    
402    commands = commands.concat(contextMenuCommands);
403    
404    if (process.argv.includes('--clear')) {
405        commands = [];
406        contextMenuCommands = [];
407    }
408    
409  const rest = new REST({ version: '9' }).setToken(TOKEN);  const rest = new REST({ version: '9' }).setToken(TOKEN);
410    
 rest.put(Routes.applicationGuildCommands(CLIENT_ID, GUILD_ID), { body: commands })  
     .then(() => console.log('Successfully registered application commands.'))  
     .catch(console.error);  
411    rest.put(Routes[process.argv.includes('--guild') ? 'applicationGuildCommands' : 'applicationCommands'](CLIENT_ID, GUILD_ID), { body: commands })
412        .then(() => console.log('Successfully registered application ' + (process.argv.includes('--guild') ? 'guild ' : '') + 'commands.'))
413        .catch(console.error);

Legend:
Removed from v.63  
changed lines
  Added in v.268

[email protected]
ViewVC Help
Powered by ViewVC 1.1.26