Compare commits
No commits in common. "v5" and "5.1.11" have entirely different histories.
304 changed files with 7674 additions and 73296 deletions
398
CHANGELOG.md
398
CHANGELOG.md
|
@ -2,322 +2,14 @@
|
||||||
|
|
||||||
Mostly based on [keepachangelog](https://keepachangelog.com/en/1.1.0/) except date format. a-c-f-r-o
|
Mostly based on [keepachangelog](https://keepachangelog.com/en/1.1.0/) except date format. a-c-f-r-o
|
||||||
|
|
||||||
## [5.3.3] - 16.12.2024
|
|
||||||
|
|
||||||
## Fixed
|
|
||||||
|
|
||||||
- `.notify` commands are no longer owner only, they now require Admin permissions
|
|
||||||
- `.notify` messages can now mention anyone
|
|
||||||
|
|
||||||
## [5.3.2] - 14.12.2024
|
|
||||||
|
|
||||||
## Fixed
|
|
||||||
|
|
||||||
- `.banner` should be working properly now with both server and global user banners
|
|
||||||
|
|
||||||
## [5.3.1] - 13.12.2024
|
|
||||||
|
|
||||||
## Changed
|
|
||||||
|
|
||||||
- `.translate` will now use 2 embeds, to allow for longer messages
|
|
||||||
- Added role icon to `.inrole`, if it exists
|
|
||||||
- `.honeypot` will now add a 'Honeypot' as a ban reason.
|
|
||||||
|
|
||||||
## Fixed
|
|
||||||
|
|
||||||
- `.winlb` looks better, has a title, shows 9 entries now
|
|
||||||
- `.sar ex` help updated
|
|
||||||
- `.banner` partially fixed, it still can't show global banners, but it will show guild ones correctly, in a good enough size
|
|
||||||
- `.sclr` will now show correct color hexes without alpha
|
|
||||||
- `.dmcmd` will now correctly block commands in dms, not globally
|
|
||||||
|
|
||||||
## [5.3.0] - 10.12.2024
|
|
||||||
|
|
||||||
## Added
|
|
||||||
|
|
||||||
- Added `.minesweeper` / `.mw` command - spoiler-based minesweeper minigame. Just for fun
|
|
||||||
- Added `.temprole` command - add a role to a user for a certain amount of time, after which the role will be removed
|
|
||||||
- Added `.xplevelset` - you can now set a level for a user in your server
|
|
||||||
- Added `.winlb` command - leaderboard of top gambling wins
|
|
||||||
- Added `.notify` command
|
|
||||||
- Specify an event to be notified about, and the bot will post the specified message in the current channel when the
|
|
||||||
event occurs
|
|
||||||
- A few events supported right now:
|
|
||||||
- `UserLevelUp` when user levels up in the server
|
|
||||||
- `AddRoleReward` when a role is added to a user through .xpreward system
|
|
||||||
- `RemoveRoleReward` when a role is removed from a user through .xpreward system
|
|
||||||
- `Protection` when antialt, antiraid or antispam protection is triggered
|
|
||||||
- Added `.banner` command to see someone's banner
|
|
||||||
- Selfhosters:
|
|
||||||
- Added `.dmmod` and `.dmcmd` - you can now disable or enable whether commands or modules can be executed in bot's
|
|
||||||
DMs
|
|
||||||
|
|
||||||
## Changed
|
|
||||||
|
|
||||||
- Giveaway improvements
|
|
||||||
- Now mentions winners in a separate message
|
|
||||||
- Shows the timestamp of when the giveaway ends
|
|
||||||
- Xp Changes
|
|
||||||
- Removed awarded xp (the number in the brackets on the xp card)
|
|
||||||
- Awarded xp, (or the new level set) now directly apply to user's real xp
|
|
||||||
- Server xp notifications are now set by the server admin/manager in a specified channel
|
|
||||||
- `.sclr show` will now show hex code of the current color
|
|
||||||
- Queueing a song will now restart the playback if the queue is on the last track and stopped (there were no more tracks
|
|
||||||
to play)
|
|
||||||
- `.translate` will now use 2 embeds instead of 1
|
|
||||||
|
|
||||||
## Fixed
|
|
||||||
|
|
||||||
- .setstream and .setactivity will now pause .ropl (rotating statuses)
|
|
||||||
- Fixed `.sar ex` help description
|
|
||||||
|
|
||||||
## Removed
|
|
||||||
|
|
||||||
- `.xpnotify` command, superseded by `.notify`, although as of right now you can't post user's level up in the same
|
|
||||||
channel user last typed, because you have to specify a channel where the notify messages will be posted
|
|
||||||
|
|
||||||
## [5.2.4] - 29.11.2024
|
|
||||||
|
|
||||||
## Fixed
|
|
||||||
|
|
||||||
- More fixes for .sclr
|
|
||||||
- `.iamn` fixed
|
|
||||||
|
|
||||||
## [5.2.3] - 29.11.2024
|
|
||||||
|
|
||||||
### Fixed
|
|
||||||
|
|
||||||
- `.iam` Fixed
|
|
||||||
- `.sclr` will now properly change color on many commands it didn't work previously
|
|
||||||
|
|
||||||
### Changed
|
|
||||||
|
|
||||||
- `.rps` now also has bet amount in the result, like other gambling commands
|
|
||||||
|
|
||||||
## [5.2.2] - 29.11.2024
|
|
||||||
|
|
||||||
### Changed
|
|
||||||
|
|
||||||
- Button roles are now non-exclusive by default
|
|
||||||
|
|
||||||
### Fixed
|
|
||||||
|
|
||||||
- Fixed sar migration, again (this time correctly)
|
|
||||||
- Fixed `.sclr` not updating unless bot is restarted, the changes should be immediate now for warn and error
|
|
||||||
- Fixed group buttons exclusivity message always saying groups are exclusive
|
|
||||||
|
|
||||||
|
|
||||||
## [5.2.1] - 28.11.2024
|
|
||||||
|
|
||||||
### Fixed
|
|
||||||
|
|
||||||
- Fixed old self assigned missing
|
|
||||||
|
|
||||||
## [5.2.0] - 28.11.2024
|
|
||||||
|
|
||||||
### Added
|
|
||||||
|
|
||||||
- Added `.todo undone` command to unmark a todo as done
|
|
||||||
- Added Button Roles!
|
|
||||||
- `.btr a` to add a button role to the specified message
|
|
||||||
- `.btr list` to list all button roles on the server
|
|
||||||
- `.btr rm` to remove a button role from the specified message
|
|
||||||
- `.btr rma` to remove all button roles on the specified message
|
|
||||||
- `.btr excl` to toggle exclusive button roles (only 1 role per message or any number)
|
|
||||||
- Use `.h btr` for more info
|
|
||||||
- Added `.wrongsong` which will delete the last queued song.
|
|
||||||
- Useful in case you made a mistake, or the bot queued a wrong song
|
|
||||||
- It will reset after a shuffle or fairplay toggle, or similar events.
|
|
||||||
- Added Server color Commands!
|
|
||||||
- Every Server can now set their own colors for ok/error/pending embed (the default green/red/yellow color on the
|
|
||||||
left side of the message the bot sends)
|
|
||||||
- Use `.h .sclr` to see the list of commands
|
|
||||||
- `.sclr show` will show the current server colors
|
|
||||||
- `.sclr ok <color hex>` to set ok color
|
|
||||||
- `.sclr warn <color hex>` to set warn color
|
|
||||||
- `.sclr error <color hex>` to set error color
|
|
||||||
|
|
||||||
### Changed
|
|
||||||
|
|
||||||
- Self Assigned Roles reworked! Use `.h .sar` for the list of commands
|
|
||||||
- `.sar autodel`
|
|
||||||
- Toggles the automatic deletion of the user's message and Nadeko's confirmations for .iam and .iamn commands.
|
|
||||||
- `.sar ad`
|
|
||||||
- Adds a role to the list of self-assignable roles. You can also specify a group.
|
|
||||||
- If 'Exclusive self-assignable roles' feature is enabled (.sar exclusive), users will be able to pick one role
|
|
||||||
per group.
|
|
||||||
- `.sar groupname`
|
|
||||||
- Sets a self assignable role group name. Provide no name to remove.
|
|
||||||
- `.sar remove`
|
|
||||||
- Removes a specified role from the list of self-assignable roles.
|
|
||||||
- `.sar list`
|
|
||||||
- Lists self-assignable roles. Shows 20 roles per page.
|
|
||||||
- `.sar exclusive`
|
|
||||||
- Toggles whether self-assigned roles are exclusive. While enabled, users can only have one self-assignable role
|
|
||||||
per group.
|
|
||||||
- `.sar rolelvlreq`
|
|
||||||
- Set a level requirement on a self-assignable role.
|
|
||||||
- `.sar grouprolereq`
|
|
||||||
- Set a role that users have to have in order to assign a self-assignable role from the specified group.
|
|
||||||
- `.sar groupdelete`
|
|
||||||
- Deletes a self-assignable role group
|
|
||||||
- `.iam` and `.iamn` are unchanged
|
|
||||||
- Removed patron limits from Reaction Roles. Anyone can have as many reros as they like.
|
|
||||||
- `.timely` captcha made stronger and cached per user.
|
|
||||||
- `.bsreset` price reduced by 90%
|
|
||||||
|
|
||||||
### Fixed
|
|
||||||
|
|
||||||
- Fixed `.sinfo` for servers on other shard
|
|
||||||
|
|
||||||
## [5.1.20] - 13.11.2024
|
|
||||||
|
|
||||||
### Added
|
|
||||||
|
|
||||||
- Added `.rakeback` command, get a % of house edge back as claimable currency
|
|
||||||
- Added `.snipe` command to quickly get a copy of a posted message as an embed
|
|
||||||
- You can reply to a message to snipe that message
|
|
||||||
- Or just type .snipe and the bot will snipe the last message in the channel with content or image
|
|
||||||
- Added `.betstatsreset` / `.bsreset` command to reset your stats for a fee
|
|
||||||
- Added `.gamblestatsreset` / `.gsreset` owner-only command to reset bot stats for all games
|
|
||||||
- Added `.waifuclaims` command which lists all of your claimed waifus
|
|
||||||
- Added and changed `%bot.time%` and `%bot.date%` placeholders. They use timestamp tags now
|
|
||||||
|
|
||||||
### Changed
|
|
||||||
|
|
||||||
- `.divorce` no longer has a cooldown
|
|
||||||
- `.betroll` has a 2% better payout
|
|
||||||
- `.slot` payout balanced out (less volatile), reduced jackpot win but increased other wins,
|
|
||||||
- now has a new symbol, wheat
|
|
||||||
- worse around 1% in total (now shares the top spot with .bf)
|
|
||||||
|
|
||||||
## [5.1.19] - 05.11.2024
|
|
||||||
|
|
||||||
### Added
|
|
||||||
|
|
||||||
- Added `.betstats`
|
|
||||||
- See your own stats with .betstats
|
|
||||||
- Target someone else: .betstats @mai_lanfiel
|
|
||||||
- You can also specify a game .betstats lula
|
|
||||||
- Or both! .betstats mai_lanfiel br
|
|
||||||
- `.timely` can now have a server boost bonus
|
|
||||||
- Configure server ids and reward amount in data/gambling.yml
|
|
||||||
- anyone who boosts one of the sepcified servers gets the amount as base timely bonus
|
|
||||||
|
|
||||||
### Changed
|
|
||||||
|
|
||||||
- `.plant/pick` password font size will be slightly bigger
|
|
||||||
- `.race` will now have 82-94% payout rate based on the number of players playing (1-12, x0.01 per player).
|
|
||||||
- Any player over 12 won't increase payout
|
|
||||||
|
|
||||||
### Fixed
|
|
||||||
|
|
||||||
- `.xplb` and `.xpglb` now have proper ranks after page 1
|
|
||||||
- Fixed boost bonus on shards different than the specified servers' shard
|
|
||||||
|
|
||||||
## [5.1.18] - 04.11.2024
|
|
||||||
|
|
||||||
### Added
|
|
||||||
|
|
||||||
- Added `.translateflags` / `.trfl` command.
|
|
||||||
- Enable on a per-channel basis.
|
|
||||||
- Reacting on any message in that channel with a flag emoji will post the translation of that message in the
|
|
||||||
language of that country
|
|
||||||
- 5 second cooldown per user
|
|
||||||
- The message can only be translated once per language (counter resets every 24h)
|
|
||||||
- `.timely` now has a button. Togglable via `.conf gambling` it's called pass because previously it was a captcha, but captchas are too annoying
|
|
||||||
|
|
||||||
## Changed
|
|
||||||
|
|
||||||
- [public bot] Patreon reward bonus for flowers reduced. Timely bonuses stay the same
|
|
||||||
- discriminators removed from the databases. All users who had ???? as discriminator have been renamed to ??username.
|
|
||||||
- all new unknown users will have ??Unknown as their name
|
|
||||||
- Flower currency generation will now have a strikeout to try combat the pickbots. This is the weakest but easiest protection to implement. There may be more options in the future
|
|
||||||
|
|
||||||
## Fixed
|
|
||||||
|
|
||||||
- nunchi join game message is now ok color instead of error color
|
|
||||||
|
|
||||||
## [5.1.17] - 29.10.2024
|
|
||||||
|
|
||||||
### Fixed
|
|
||||||
|
|
||||||
- fix: Bot will now not accept .aar Role if that Role is higher than or equal to bot's role. Previously bot would just
|
|
||||||
fail silently, now there is a proper error message.
|
|
||||||
|
|
||||||
## [5.1.16] - 28.10.2024
|
|
||||||
|
|
||||||
## Added
|
|
||||||
|
|
||||||
- Added .ncanvas and related commands.
|
|
||||||
- You can set pixel colors (and text) on a 500x350 canvas, pepega version of r/place
|
|
||||||
- You use currency to set pixels.
|
|
||||||
- Commands:
|
|
||||||
- see the entire canvas: `.nc`
|
|
||||||
- zoom: `.ncz <pos>` or `.ncz x y`
|
|
||||||
- set pixel: `.ncsp <pos> <color> <text?>`
|
|
||||||
- get pixel: `.ncp <pos>`
|
|
||||||
- Owners can use .ncsetimg to set a starting image, use `.h .setimg` for instructions
|
|
||||||
- Owners can reset the whole canvas via `.ncreset`
|
|
||||||
|
|
||||||
## [5.1.15] - 21.10.2024
|
|
||||||
|
|
||||||
## Added
|
|
||||||
|
|
||||||
- Added -c option for `.xpglb`
|
|
||||||
|
|
||||||
## Change
|
|
||||||
|
|
||||||
- Leaderboards will now show 10 users per page
|
|
||||||
- A lot of internal changes and improvements
|
|
||||||
|
|
||||||
## Fixed
|
|
||||||
|
|
||||||
- Fixed a big issue which caused several features to not get loaded on bot restart
|
|
||||||
- Alias collision fix `.qse` is now quotesearch, `.qs` will stay `.queuesearch`
|
|
||||||
- Fixed some migrations which would prevent users from updating from ancient versions
|
|
||||||
- Waifulb will no longer show #0000 discrims
|
|
||||||
- More `.greet` command fixes
|
|
||||||
- Author name will now be counted as content in embeds. Embeds can now only have author fields and still be valid
|
|
||||||
- Grpc api fixes, and additions
|
|
||||||
|
|
||||||
## [5.1.14] - 03.10.2024
|
|
||||||
|
|
||||||
## Changed
|
|
||||||
|
|
||||||
- Improved `.xplb -c`, it will now correctly only show users who are still in the server with no count limit
|
|
||||||
|
|
||||||
## Fixed
|
|
||||||
|
|
||||||
- Fixed marmalade load error on startup
|
|
||||||
|
|
||||||
## [5.1.13] - 03.10.2024
|
|
||||||
|
|
||||||
### Fixed
|
|
||||||
|
|
||||||
- Grpc api server will no longer start unless enabled in creds
|
|
||||||
- Seq comment in creds fixed
|
|
||||||
|
|
||||||
## [5.1.12] - 03.10.2024
|
|
||||||
|
|
||||||
### Added
|
|
||||||
|
|
||||||
- Added support for `seq` for logging. If you fill in seq url and apiKey in creds.yml, bot will sends logs to it
|
|
||||||
|
|
||||||
### Fixed
|
|
||||||
|
|
||||||
- Fixed the Check for updates service not using the right URL and spitting an error in the console.
|
|
||||||
- Fixed another bug in `.greet` / `.bye` system, which caused it to show wrong message on a wrong server occasionally
|
|
||||||
|
|
||||||
## [5.1.11] - 03.10.2024
|
## [5.1.11] - 03.10.2024
|
||||||
|
|
||||||
### Added
|
### Added
|
||||||
|
|
||||||
- Added `%user.displayname%` placeholder. It will show users nickname, if there is one, otherwise it will show the
|
- Added `%user.displayname%` placeholder. It will show users nickname, if there is one, otherwise it will show the username.
|
||||||
username.
|
- Nickname won't be shown in bye messages.
|
||||||
- Nickname won't be shown in bye messages.
|
|
||||||
- Added initial version of grpc api. Beta
|
- Added initial version of grpc api. Beta
|
||||||
|
|
||||||
### Fixed
|
### Fixed
|
||||||
|
|
||||||
- Fixed a bug which caused `.bye` and `.greet` messages to be randomly disabled
|
- Fixed a bug which caused `.bye` and `.greet` messages to be randomly disabled
|
||||||
|
@ -326,9 +18,9 @@ Mostly based on [keepachangelog](https://keepachangelog.com/en/1.1.0/) except da
|
||||||
### Changed
|
### Changed
|
||||||
|
|
||||||
- Youtube now always uses `yt-dlp`. Dropped support for `youtube-dl`
|
- Youtube now always uses `yt-dlp`. Dropped support for `youtube-dl`
|
||||||
- If you've previously renamed your yt-dlp file to youtube-dl, please rename it back.
|
- If you've previously renamed your yt-dlp file to youtube-dl, please rename it back.
|
||||||
- ytProvider in data/searches.yml now also controls where you're getting your song streams from.
|
- ytProvider in data/searches.yml now also controls where you're getting your song streams from.
|
||||||
- (Invidious support added for .q)
|
- (Invidious support added for .q)
|
||||||
|
|
||||||
## [5.1.10] - 24.09.2024
|
## [5.1.10] - 24.09.2024
|
||||||
|
|
||||||
|
@ -347,28 +39,26 @@ Mostly based on [keepachangelog](https://keepachangelog.com/en/1.1.0/) except da
|
||||||
- Fixed `.greettest`, and other `.*test` commands if you didn't have them enabled.
|
- Fixed `.greettest`, and other `.*test` commands if you didn't have them enabled.
|
||||||
- Fixed `.greetdmtest` sending messages twice.
|
- Fixed `.greetdmtest` sending messages twice.
|
||||||
- Fixed a serious bug which caused greet messages to be jumbled up, and wrong ones to be sent for the wrong events.
|
- Fixed a serious bug which caused greet messages to be jumbled up, and wrong ones to be sent for the wrong events.
|
||||||
- There is no database issue, all greet messages are safe, the cache was caching any setting every 3 seconds with no
|
- There is no database issue, all greet messages are safe, the cache was caching any setting every 3 seconds with no regard for the type of the event
|
||||||
regard for the type of the event
|
- This also caused `.greetdm` messages to not be sent if `.greet` is enabled
|
||||||
- This also caused `.greetdm` messages to not be sent if `.greet` is enabled
|
- This bug was introduced in 5.1.8. PLEASE UPDATE if you are on 5.1.8
|
||||||
- This bug was introduced in 5.1.8. PLEASE UPDATE if you are on 5.1.8
|
|
||||||
- Selfhosters only: Fixed marmalade dependency loading
|
- Selfhosters only: Fixed marmalade dependency loading
|
||||||
- Note: Make sure to not publish any other DLLs besides the ones you are sure you will need, as there can be version
|
- Note: Make sure to not publish any other DLLs besides the ones you are sure you will need, as there can be version conflicts which didn't happen before.
|
||||||
conflicts which didn't happen before.
|
|
||||||
|
|
||||||
## [5.1.8] - 20.09.2024
|
## [5.1.8] - 20.09.2024
|
||||||
|
|
||||||
### Added
|
### Added
|
||||||
|
|
||||||
- Added `.leaveunkeptservers` which will make the bot leave all servers on all shards whose owners didn't run `.keep` command.
|
- Added `.leaveunkeptservers` which will make the bot leave all servers on all shards whose owners didn't run `.keep` command.
|
||||||
- This is a dangerous and irreversible command, don't use it. Meant for use on the public bot.
|
- This is a dangerous and irreversible command, don't use it. Meant for use on the public bot.
|
||||||
- `.adpl` now supports custom statuses (you no longer need to specify Playing, Watching, etc...)
|
- `.adpl` now supports custom statuses (you no longer need to specify Playing, Watching, etc...)
|
||||||
|
|
||||||
### Changed
|
### Changed
|
||||||
|
|
||||||
- `.quote` commands cleaned up and improved
|
- `.quote` commands cleaned up and improved
|
||||||
- All quote commands now start with `.q<whatever>` and follow the same naming pattern as Expression commands
|
- All quote commands now start with `.q<whatever>` and follow the same naming pattern as Expression commands
|
||||||
- `.liqu` renamed to `.qli`
|
- `.liqu` renamed to `.qli`
|
||||||
- `.quotesearch` / `.qse` is now paginated for easier searching
|
- `.quotesearch` / `.qse` is now paginated for easier searching
|
||||||
- `.whosplaying` is now paginated
|
- `.whosplaying` is now paginated
|
||||||
- `.img` is now paginated
|
- `.img` is now paginated
|
||||||
- `.setgame` renamed to`.setactivity` and now supports custom text activity. You don't have to specify playing, listening etc before the activity
|
- `.setgame` renamed to`.setactivity` and now supports custom text activity. You don't have to specify playing, listening etc before the activity
|
||||||
|
@ -388,7 +78,7 @@ Mostly based on [keepachangelog](https://keepachangelog.com/en/1.1.0/) except da
|
||||||
### Removed
|
### Removed
|
||||||
|
|
||||||
- Removed mysql support as it didn't work for a while, and requires some special handling/maintenance
|
- Removed mysql support as it didn't work for a while, and requires some special handling/maintenance
|
||||||
- Sqlite and Postgres support stays
|
- Sqlite and Postgres support stays
|
||||||
|
|
||||||
## [5.1.7] - 09.08.2024
|
## [5.1.7] - 09.08.2024
|
||||||
|
|
||||||
|
@ -410,18 +100,17 @@ Mostly based on [keepachangelog](https://keepachangelog.com/en/1.1.0/) except da
|
||||||
|
|
||||||
- `.afk` messages can no longer ping, and the response is moved to DMs to avoid abuse
|
- `.afk` messages can no longer ping, and the response is moved to DMs to avoid abuse
|
||||||
- Possible fix for `.remind` timestamp
|
- Possible fix for `.remind` timestamp
|
||||||
|
|
||||||
### Removed
|
### Removed
|
||||||
|
- Removed old bloat / semi broken / dumb commands
|
||||||
- Removed old bloat / semi broken / dumb commands
|
- `.memelist` / `.memegen` (too inconvenient to use)
|
||||||
- `.memelist` / `.memegen` (too inconvenient to use)
|
- `.activity` (useless owner-only command)
|
||||||
- `.activity` (useless owner-only command)
|
- `.rafflecur` (Just use raffle and then award manually instead)
|
||||||
- `.rafflecur` (Just use raffle and then award manually instead)
|
- `.rollduel` (we had this command?)
|
||||||
- `.rollduel` (we had this command?)
|
|
||||||
- You can no longer bet on `.connect4`
|
- You can no longer bet on `.connect4`
|
||||||
- `.economy` Removed.
|
- `.economy` Removed.
|
||||||
- Was buggy and didn't really show the real state of the economy.
|
- Was buggy and didn.t really show the real state of the economy.
|
||||||
- It might come back improved in the future
|
- It might come back improved in the future
|
||||||
- `.mal` Removed. Useless information / semi broken
|
- `.mal` Removed. Useless information / semi broken
|
||||||
|
|
||||||
## [5.1.5] - 01.08.2024
|
## [5.1.5] - 01.08.2024
|
||||||
|
@ -429,9 +118,9 @@ Mostly based on [keepachangelog](https://keepachangelog.com/en/1.1.0/) except da
|
||||||
### Added
|
### Added
|
||||||
|
|
||||||
- Added: Added a `.afk <msg>?` command which sets an afk message which will trigger whenever someone pings you
|
- Added: Added a `.afk <msg>?` command which sets an afk message which will trigger whenever someone pings you
|
||||||
- Message will when you type a message in any channel that the bot sees, or after 8 hours, whichever comes first
|
- Message will when you type a message in any channel that the bot sees, or after 8 hours, whichever comes first
|
||||||
- The specified message will be prefixed with "The user is afk: "
|
- The specified message will be prefixed with "The user is afk: "
|
||||||
- The afk message will disappear 30 seconds after being triggered
|
- The afk message will disappear 30 seconds after being triggered
|
||||||
|
|
||||||
### Changed
|
### Changed
|
||||||
|
|
||||||
|
@ -439,7 +128,7 @@ Mostly based on [keepachangelog](https://keepachangelog.com/en/1.1.0/) except da
|
||||||
- Updated some bet descriptions to include 'all' 'half' usage instructions
|
- Updated some bet descriptions to include 'all' 'half' usage instructions
|
||||||
- Updated some command strings
|
- Updated some command strings
|
||||||
- dev: Vastly simplified marmalade creation using dotnet templates, docs updated
|
- dev: Vastly simplified marmalade creation using dotnet templates, docs updated
|
||||||
- Slight refactor of .wiki, time, .catfact, .wikia, .define, .bible and .quran commands, no significant change in functionality
|
- Slight refactor of .wiki, .time, .catfact, .wikia, .define, .bible and .quran commands, no significant change in functionality
|
||||||
|
|
||||||
### Fixed
|
### Fixed
|
||||||
|
|
||||||
|
@ -458,10 +147,8 @@ Mostly based on [keepachangelog](https://keepachangelog.com/en/1.1.0/) except da
|
||||||
- Added Clubs rank in the leaderboard to `.clubinfo`
|
- Added Clubs rank in the leaderboard to `.clubinfo`
|
||||||
- Bot owners can now check other people's bank balance (Not server owners, only bot owner, the person who is hosting the bot)
|
- Bot owners can now check other people's bank balance (Not server owners, only bot owner, the person who is hosting the bot)
|
||||||
- You can now send multiple waifu gifts at once to waifus. For example `.waifugift 3xRose @user` will give that user 3 roses
|
- You can now send multiple waifu gifts at once to waifus. For example `.waifugift 3xRose @user` will give that user 3 roses
|
||||||
- The format is `<NUMBER>x<ITEM>`, no spaces
|
- The format is `<NUMBER>x<ITEM>`, no spaces
|
||||||
- Added `.boosttest` command
|
- Added `.boosttest` command
|
||||||
- Added support for any openai compatible api for the chatterbot feature change:
|
|
||||||
- Changed games.yml to allow input of the apiUrl (needs to be openai compatible) and modelName as a string.
|
|
||||||
|
|
||||||
### Changed
|
### Changed
|
||||||
|
|
||||||
|
@ -503,9 +190,9 @@ Mostly based on [keepachangelog](https://keepachangelog.com/en/1.1.0/) except da
|
||||||
### Added
|
### Added
|
||||||
|
|
||||||
- Added `.honeypot` command, which automatically softbans (ban and immediate unban) any user who posts in that channel.
|
- Added `.honeypot` command, which automatically softbans (ban and immediate unban) any user who posts in that channel.
|
||||||
- Useful to auto softban bots who spam every channel upon joining
|
- Useful to auto softban bots who spam every channel upon joining
|
||||||
- Users who run commands or expressions won't be softbanned.
|
- Users who run commands or expressions won't be softbanned.
|
||||||
- Users who have ban member permissions are also excluded.
|
- Users who have ban member permissions are also excluded.
|
||||||
|
|
||||||
### Fixed
|
### Fixed
|
||||||
|
|
||||||
|
@ -519,6 +206,7 @@ Mostly based on [keepachangelog](https://keepachangelog.com/en/1.1.0/) except da
|
||||||
- Added support for `gpt-4o` in `data/games.yml`
|
- Added support for `gpt-4o` in `data/games.yml`
|
||||||
- Added EllieAiToken to `creds.yml`
|
- Added EllieAiToken to `creds.yml`
|
||||||
|
|
||||||
|
|
||||||
### Changed
|
### Changed
|
||||||
|
|
||||||
- Remind will now show a timestamp tag for durations
|
- Remind will now show a timestamp tag for durations
|
||||||
|
@ -531,8 +219,8 @@ Mostly based on [keepachangelog](https://keepachangelog.com/en/1.1.0/) except da
|
||||||
|
|
||||||
- Fixed xp bg buy button not working, and possibly some other buttons too
|
- Fixed xp bg buy button not working, and possibly some other buttons too
|
||||||
- Fixed shopbuy %user% placeholders and updated help text
|
- Fixed shopbuy %user% placeholders and updated help text
|
||||||
- All .feed overloads should now work"
|
- All 'feed overloads should now work"
|
||||||
- `.xpexclude` should will now work with forums too. If you exclude a forum you won't be able to gain xp in any of the threads.
|
- `'xpexclude` should will now work with forums too. If you exclude a forum you won't be able to gain xp in any of the threads.
|
||||||
- Fixed remind not showing correct time
|
- Fixed remind not showing correct time
|
||||||
|
|
||||||
### Removed
|
### Removed
|
||||||
|
@ -544,14 +232,12 @@ Mostly based on [keepachangelog](https://keepachangelog.com/en/1.1.0/) except da
|
||||||
|
|
||||||
### Added
|
### Added
|
||||||
|
|
||||||
- Added `.setserverbanner` and `.setservericon` commands
|
- Added `'setserverbanner` and `'setservericon` commands
|
||||||
- Added overloads section to `.h command` which will show you all versions of command usage with param names
|
- Added overloads section to `'h command` which will show you all versions of command usage with param names
|
||||||
- You can now check commands for submodules, for example `.cmds SelfAssignedRoles` will show brief help for each of the
|
- You can now check commands for submodules, for example `'cmds SelfAssignedRoles` will show brief help for each of the commands in that submodule
|
||||||
commands in that submodule
|
- Added dropdown menus for 'mdls and 'cmds (both module and group versions) which will give you the option to see more detailed help for each specific module, group or command respectively
|
||||||
- Added dropdown menus for .mdls and .cmds (both module and group versions) which will give you the option to see more
|
|
||||||
detailed help for each specific module, group or command respectively
|
|
||||||
- Self-Hosters only:
|
- Self-Hosters only:
|
||||||
- Added a dangerous cleanup command that you don't have to know about
|
- Added a dangerous cleanup command that you don't have to know about
|
||||||
|
|
||||||
### Changed
|
### Changed
|
||||||
|
|
||||||
|
@ -559,7 +245,7 @@ Mostly based on [keepachangelog](https://keepachangelog.com/en/1.1.0/) except da
|
||||||
|
|
||||||
### Fixed
|
### Fixed
|
||||||
|
|
||||||
- `.verbose` will now be respected for expression errors
|
- `'verbose` will now be respected for expression errors
|
||||||
- Using `.pick` will now correctly show the name of the user who picked the currency
|
- Using `'pick` will now correctly show the name of the user who picked the currency
|
||||||
- Fixed `.h` not working on some commands
|
- Fixed `'h` not working on some commands
|
||||||
- `.langset` and `.langsetd` should no longer allow unsupported languages and nonsense to be typed in
|
- `'langset` and `'langsetd` should no longer allow unsupported languages and nonsense to be typed in
|
|
@ -11,7 +11,9 @@ Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "Solution Items", "Solution
|
||||||
Dockerfile = Dockerfile
|
Dockerfile = Dockerfile
|
||||||
ellie-menu.ps1 = ellie-menu.ps1
|
ellie-menu.ps1 = ellie-menu.ps1
|
||||||
LICENSE = LICENSE
|
LICENSE = LICENSE
|
||||||
|
migrate.ps1 = migrate.ps1
|
||||||
README.md = README.md
|
README.md = README.md
|
||||||
|
remove-migrations.ps1 = remove-migrations.ps1
|
||||||
EndProjectSection
|
EndProjectSection
|
||||||
EndProject
|
EndProject
|
||||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "EllieBot", "src\EllieBot\EllieBot.csproj", "{4D9001F7-B3E8-48FE-97AA-CFD36DA65A64}"
|
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "EllieBot", "src\EllieBot\EllieBot.csproj", "{4D9001F7-B3E8-48FE-97AA-CFD36DA65A64}"
|
||||||
|
@ -28,7 +30,7 @@ Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "Ellie.Marmalade", "src\Elli
|
||||||
EndProject
|
EndProject
|
||||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "EllieBot.Voice", "src\EllieBot.Voice\EllieBot.Voice.csproj", "{1D93CE3C-80B4-49C7-A9A2-99988920AAEC}"
|
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "EllieBot.Voice", "src\EllieBot.Voice\EllieBot.Voice.csproj", "{1D93CE3C-80B4-49C7-A9A2-99988920AAEC}"
|
||||||
EndProject
|
EndProject
|
||||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "EllieBot.GrpcApiBase", "src\EllieBot.GrpcApiBase\EllieBot.GrpcApiBase.csproj", "{3B71F0BF-AE6C-480C-AB88-FCE23EDC7D91}"
|
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "EllieBot.GrpcApiBase", "src\EllieBot.GrpcApiBase\EllieBot.GrpcApiBase.csproj", "{3B71F0BF-AE6C-480C-AB88-FCE23EDC7D91}"
|
||||||
EndProject
|
EndProject
|
||||||
Global
|
Global
|
||||||
GlobalSection(SolutionConfigurationPlatforms) = preSolution
|
GlobalSection(SolutionConfigurationPlatforms) = preSolution
|
||||||
|
|
8
migrate.ps1
Normal file
8
migrate.ps1
Normal file
|
@ -0,0 +1,8 @@
|
||||||
|
if ($args.Length -eq 0) {
|
||||||
|
Write-Host "Please provide a migration name." -ForegroundColor Red
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
$migrationName = $args[0]
|
||||||
|
dotnet ef migrations add $migrationName -o Migrations/Mysql -c SqliteContext -p src/EllieBot/EllieBot.csproj
|
||||||
|
dotnet ef migrations add $migrationName -o Migrations/PostgreSql -c PostgreSqlContext -p src/EllieBot/EllieBot.csproj
|
||||||
|
}
|
|
@ -9,7 +9,7 @@
|
||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<PackageReference Include="Discord.Net.Core" Version="3.16.0" />
|
<PackageReference Include="Discord.Net.Core" Version="3.15.3" />
|
||||||
<PackageReference Include="Serilog" Version="3.1.1" />
|
<PackageReference Include="Serilog" Version="3.1.1" />
|
||||||
<PackageReference Include="YamlDotNet" Version="15.1.4" />
|
<PackageReference Include="YamlDotNet" Version="15.1.4" />
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
|
|
@ -12,14 +12,13 @@ namespace EllieBot.Generators
|
||||||
{
|
{
|
||||||
public readonly record struct MethodPermData
|
public readonly record struct MethodPermData
|
||||||
{
|
{
|
||||||
public readonly ImmutableArray<(string Name, string Value)> MethodPerms;
|
public readonly string Name;
|
||||||
public readonly ImmutableArray<string> NoAuthRequired;
|
public readonly string Value;
|
||||||
|
|
||||||
public MethodPermData(ImmutableArray<(string Name, string Value)> methodPerms,
|
public MethodPermData(string name, string value)
|
||||||
ImmutableArray<string> noAuthRequired)
|
|
||||||
{
|
{
|
||||||
MethodPerms = methodPerms;
|
Name = name;
|
||||||
NoAuthRequired = noAuthRequired;
|
Value = value;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -27,7 +26,7 @@ namespace EllieBot.Generators
|
||||||
[Generator]
|
[Generator]
|
||||||
public class GrpcApiPermGenerator : IIncrementalGenerator
|
public class GrpcApiPermGenerator : IIncrementalGenerator
|
||||||
{
|
{
|
||||||
public const string GRPC_API_PERM_ATTRIBUTE =
|
public const string Attribute =
|
||||||
"""
|
"""
|
||||||
namespace EllieBot.GrpcApi;
|
namespace EllieBot.GrpcApi;
|
||||||
|
|
||||||
|
@ -39,25 +38,12 @@ namespace EllieBot.Generators
|
||||||
}
|
}
|
||||||
""";
|
""";
|
||||||
|
|
||||||
public const string GRPC_NO_AUTH_REQUIRED_ATTRIBUTE =
|
|
||||||
"""
|
|
||||||
namespace EllieBot.GrpcApi;
|
|
||||||
|
|
||||||
[System.AttributeUsage(System.AttributeTargets.Method)]
|
|
||||||
public class GrpcNoAuthRequiredAttribute : System.Attribute
|
|
||||||
{
|
|
||||||
}
|
|
||||||
""";
|
|
||||||
|
|
||||||
public void Initialize(IncrementalGeneratorInitializationContext context)
|
public void Initialize(IncrementalGeneratorInitializationContext context)
|
||||||
{
|
{
|
||||||
context.RegisterPostInitializationOutput(ctx => ctx.AddSource("GrpcApiPermAttribute.cs",
|
context.RegisterPostInitializationOutput(ctx => ctx.AddSource("GrpcApiPermAttribute.cs",
|
||||||
SourceText.From(GRPC_API_PERM_ATTRIBUTE, Encoding.UTF8)));
|
SourceText.From(Attribute, Encoding.UTF8)));
|
||||||
|
|
||||||
context.RegisterPostInitializationOutput(ctx => ctx.AddSource("GrpcNoAuthRequiredAttribute.cs",
|
var enumsToGenerate = context.SyntaxProvider
|
||||||
SourceText.From(GRPC_NO_AUTH_REQUIRED_ATTRIBUTE, Encoding.UTF8)));
|
|
||||||
|
|
||||||
var perms = context.SyntaxProvider
|
|
||||||
.ForAttributeWithMetadataName(
|
.ForAttributeWithMetadataName(
|
||||||
"EllieBot.GrpcApi.GrpcApiPermAttribute",
|
"EllieBot.GrpcApi.GrpcApiPermAttribute",
|
||||||
predicate: static (s, _) => s is MethodDeclarationSyntax,
|
predicate: static (s, _) => s is MethodDeclarationSyntax,
|
||||||
|
@ -66,24 +52,11 @@ namespace EllieBot.Generators
|
||||||
.Select(static (x, _) => x!.Value)
|
.Select(static (x, _) => x!.Value)
|
||||||
.Collect();
|
.Collect();
|
||||||
|
|
||||||
|
context.RegisterSourceOutput(enumsToGenerate,
|
||||||
var all = context.SyntaxProvider
|
|
||||||
.ForAttributeWithMetadataName(
|
|
||||||
"EllieBot.GrpcApi.GrpcNoAuthRequiredAttribute",
|
|
||||||
predicate: static (s, _) => s is MethodDeclarationSyntax,
|
|
||||||
transform: static (ctx, _) => GetNoAuthMethodName(ctx.SemanticModel, ctx.TargetNode))
|
|
||||||
.Collect()
|
|
||||||
.Combine(perms)
|
|
||||||
.Select((x, _) => new MethodPermData(x.Right, x.Left));
|
|
||||||
|
|
||||||
context.RegisterSourceOutput(all,
|
|
||||||
static (spc, source) => Execute(source, spc));
|
static (spc, source) => Execute(source, spc));
|
||||||
}
|
}
|
||||||
|
|
||||||
private static string GetNoAuthMethodName(SemanticModel model, SyntaxNode node)
|
private static MethodPermData? GetMethodSemanticTargets(SemanticModel model, SyntaxNode node)
|
||||||
=> ((MethodDeclarationSyntax)node).Identifier.Text;
|
|
||||||
|
|
||||||
private static (string Name, string Value)? GetMethodSemanticTargets(SemanticModel model, SyntaxNode node)
|
|
||||||
{
|
{
|
||||||
var method = (MethodDeclarationSyntax)node;
|
var method = (MethodDeclarationSyntax)node;
|
||||||
|
|
||||||
|
@ -91,14 +64,20 @@ namespace EllieBot.Generators
|
||||||
var attr = method.AttributeLists
|
var attr = method.AttributeLists
|
||||||
.SelectMany(x => x.Attributes)
|
.SelectMany(x => x.Attributes)
|
||||||
.FirstOrDefault();
|
.FirstOrDefault();
|
||||||
|
// .FirstOrDefault(x => x.Name.ToString() == "GrpcApiPermAttribute");
|
||||||
|
|
||||||
|
|
||||||
if (attr is null)
|
if (attr is null)
|
||||||
return null;
|
return null;
|
||||||
|
|
||||||
return (name, attr.ArgumentList?.Arguments[0].ToString() ?? "__missing_perm__");
|
// if (model.GetSymbolInfo(attr).Symbol is not IMethodSymbol attrSymbol)
|
||||||
|
// return null;
|
||||||
|
|
||||||
|
return new MethodPermData(name, attr.ArgumentList.Arguments[0].ToString() ?? "__missing_perm__");
|
||||||
|
// return new MethodPermData(name, attrSymbol.Parameters[0].ContainingType.ToDisplayString() + "." + attrSymbol.Parameters[0].Name);
|
||||||
}
|
}
|
||||||
|
|
||||||
private static void Execute(MethodPermData data, SourceProductionContext ctx)
|
private static void Execute(ImmutableArray<MethodPermData> fields, SourceProductionContext ctx)
|
||||||
{
|
{
|
||||||
using (var stringWriter = new StringWriter())
|
using (var stringWriter = new StringWriter())
|
||||||
using (var sw = new IndentedTextWriter(stringWriter))
|
using (var sw = new IndentedTextWriter(stringWriter))
|
||||||
|
@ -108,17 +87,16 @@ namespace EllieBot.Generators
|
||||||
sw.WriteLine("namespace EllieBot.GrpcApi;");
|
sw.WriteLine("namespace EllieBot.GrpcApi;");
|
||||||
sw.WriteLine();
|
sw.WriteLine();
|
||||||
|
|
||||||
sw.WriteLine("public partial class GrpcApiPermsInterceptor");
|
sw.WriteLine("public partial class PermsInterceptor");
|
||||||
sw.WriteLine("{");
|
sw.WriteLine("{");
|
||||||
|
|
||||||
sw.Indent++;
|
sw.Indent++;
|
||||||
|
|
||||||
sw.WriteLine(
|
sw.WriteLine("public static FrozenDictionary<string, GuildPerm> perms = new Dictionary<string, GuildPerm>()");
|
||||||
"private static FrozenDictionary<string, GuildPerm> _perms = new Dictionary<string, GuildPerm>()");
|
|
||||||
sw.WriteLine("{");
|
sw.WriteLine("{");
|
||||||
|
|
||||||
sw.Indent++;
|
sw.Indent++;
|
||||||
foreach (var field in data.MethodPerms)
|
foreach (var field in fields)
|
||||||
{
|
{
|
||||||
sw.WriteLine("{{ \"{0}\", {1} }},", field.Name, field.Value);
|
sw.WriteLine("{{ \"{0}\", {1} }},", field.Name, field.Value);
|
||||||
}
|
}
|
||||||
|
@ -126,21 +104,6 @@ namespace EllieBot.Generators
|
||||||
sw.Indent--;
|
sw.Indent--;
|
||||||
sw.WriteLine("}.ToFrozenDictionary();");
|
sw.WriteLine("}.ToFrozenDictionary();");
|
||||||
|
|
||||||
sw.WriteLine();
|
|
||||||
sw.WriteLine("private static FrozenSet<string> _noAuthRequired = new HashSet<string>()");
|
|
||||||
sw.WriteLine("{");
|
|
||||||
|
|
||||||
sw.Indent++;
|
|
||||||
foreach (var noauth in data.NoAuthRequired)
|
|
||||||
{
|
|
||||||
sw.WriteLine("{{ \"{0}\" }},", noauth);
|
|
||||||
}
|
|
||||||
|
|
||||||
sw.WriteLine("");
|
|
||||||
|
|
||||||
sw.Indent--;
|
|
||||||
sw.WriteLine("}.ToFrozenSet();");
|
|
||||||
|
|
||||||
sw.Indent--;
|
sw.Indent--;
|
||||||
sw.WriteLine("}");
|
sw.WriteLine("}");
|
||||||
|
|
||||||
|
|
|
@ -1,47 +0,0 @@
|
||||||
syntax = "proto3";
|
|
||||||
|
|
||||||
option csharp_namespace = "EllieBot.GrpcApi";
|
|
||||||
|
|
||||||
import "google/protobuf/empty.proto";
|
|
||||||
|
|
||||||
package ncanvas;
|
|
||||||
|
|
||||||
service GrpcNCanvas {
|
|
||||||
rpc GetCanvas(google.protobuf.Empty) returns (CanvasReply);
|
|
||||||
rpc GetPixel(GetPixelRequest) returns (GetPixelReply);
|
|
||||||
rpc SetPixel(SetPixelRequest) returns (SetPixelReply);
|
|
||||||
}
|
|
||||||
|
|
||||||
message CanvasReply {
|
|
||||||
repeated uint32 pixels = 1;
|
|
||||||
int32 width = 2;
|
|
||||||
int32 height = 3;
|
|
||||||
}
|
|
||||||
|
|
||||||
message GetPixelRequest {
|
|
||||||
int32 x = 1;
|
|
||||||
int32 y = 2;
|
|
||||||
}
|
|
||||||
|
|
||||||
message GetPixelReply {
|
|
||||||
string color = 1;
|
|
||||||
uint32 packedColor = 2;
|
|
||||||
int32 positionX = 3;
|
|
||||||
int32 positionY = 4;
|
|
||||||
int64 price = 5;
|
|
||||||
string text = 6;
|
|
||||||
string position = 7;
|
|
||||||
}
|
|
||||||
|
|
||||||
message SetPixelRequest {
|
|
||||||
string position = 1;
|
|
||||||
string color = 2;
|
|
||||||
string text = 3;
|
|
||||||
int64 price = 4;
|
|
||||||
}
|
|
||||||
|
|
||||||
message SetPixelReply {
|
|
||||||
string error = 1;
|
|
||||||
bool success = 2;
|
|
||||||
optional GetPixelReply pixel = 3;
|
|
||||||
}
|
|
26
src/EllieBot.GrpcApiBase/protos/econ.proto
Normal file
26
src/EllieBot.GrpcApiBase/protos/econ.proto
Normal file
|
@ -0,0 +1,26 @@
|
||||||
|
syntax = "proto3";
|
||||||
|
|
||||||
|
option csharp_namespace = "EllieBot.GrpcApi";
|
||||||
|
|
||||||
|
package econ;
|
||||||
|
|
||||||
|
service GrpcEcon {
|
||||||
|
rpc GetEconomy(EconomyRequest) returns (EconomyReply);
|
||||||
|
}
|
||||||
|
|
||||||
|
message EconomyRequest {
|
||||||
|
string guildId = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
message EconomyReply {
|
||||||
|
uint64 totalOwned = 1;
|
||||||
|
uint64 byTopOnePercent = 2;
|
||||||
|
uint64 plantedAmount = 3;
|
||||||
|
uint64 ownedByTheBot = 4;
|
||||||
|
uint64 inTheBank = 5;
|
||||||
|
uint64 totalEconomy = 6;
|
||||||
|
}
|
||||||
|
|
||||||
|
message CurrencyLbRequest {
|
||||||
|
int32 page = 1;
|
||||||
|
}
|
|
@ -2,7 +2,7 @@
|
||||||
|
|
||||||
option csharp_namespace = "EllieBot.GrpcApi";
|
option csharp_namespace = "EllieBot.GrpcApi";
|
||||||
|
|
||||||
import "google/protobuf/empty.proto";
|
import "google/protobuf/empty.proto";
|
||||||
|
|
||||||
package exprs;
|
package exprs;
|
||||||
|
|
||||||
|
@ -10,10 +10,6 @@ service GrpcExprs {
|
||||||
rpc GetExprs(GetExprsRequest) returns (GetExprsReply);
|
rpc GetExprs(GetExprsRequest) returns (GetExprsReply);
|
||||||
rpc AddExpr(AddExprRequest) returns (AddExprReply);
|
rpc AddExpr(AddExprRequest) returns (AddExprReply);
|
||||||
rpc DeleteExpr(DeleteExprRequest) returns (google.protobuf.Empty);
|
rpc DeleteExpr(DeleteExprRequest) returns (google.protobuf.Empty);
|
||||||
|
|
||||||
rpc GetQuotes(GetQuotesRequest) returns (GetQuotesReply);
|
|
||||||
rpc AddQuote(AddQuoteRequest) returns (AddQuoteReply);
|
|
||||||
rpc DeleteQuote(DeleteQuoteRequest) returns (google.protobuf.Empty);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
message DeleteExprRequest {
|
message DeleteExprRequest {
|
||||||
|
@ -52,38 +48,3 @@ message AddExprReply {
|
||||||
string id = 1;
|
string id = 1;
|
||||||
bool success = 2;
|
bool success = 2;
|
||||||
}
|
}
|
||||||
|
|
||||||
message GetQuotesRequest {
|
|
||||||
uint64 guildId = 1;
|
|
||||||
string query = 2;
|
|
||||||
int32 page = 3;
|
|
||||||
}
|
|
||||||
|
|
||||||
message GetQuotesReply {
|
|
||||||
repeated QuoteDto quotes = 1;
|
|
||||||
int32 totalCount = 2;
|
|
||||||
}
|
|
||||||
|
|
||||||
message QuoteDto {
|
|
||||||
string id = 1;
|
|
||||||
string trigger = 2;
|
|
||||||
string response = 3;
|
|
||||||
|
|
||||||
uint64 authorId = 4;
|
|
||||||
string authorName = 5;
|
|
||||||
}
|
|
||||||
|
|
||||||
message AddQuoteRequest {
|
|
||||||
uint64 guildId = 1;
|
|
||||||
QuoteDto quote = 2;
|
|
||||||
}
|
|
||||||
|
|
||||||
message AddQuoteReply {
|
|
||||||
string id = 1;
|
|
||||||
bool success = 2;
|
|
||||||
}
|
|
||||||
|
|
||||||
message DeleteQuoteRequest {
|
|
||||||
string id = 1;
|
|
||||||
uint64 guildId = 2;
|
|
||||||
}
|
|
|
@ -1,60 +0,0 @@
|
||||||
syntax = "proto3";
|
|
||||||
|
|
||||||
option csharp_namespace = "EllieBot.GrpcApi";
|
|
||||||
|
|
||||||
import "google/protobuf/timestamp.proto";
|
|
||||||
|
|
||||||
package fin;
|
|
||||||
|
|
||||||
service GrpcFin {
|
|
||||||
rpc GetTransactions(GetTransactionsRequest) returns (GetTransactionsReply);
|
|
||||||
rpc GetHoldings(GetHoldingsRequest) returns (GetHoldingsReply);
|
|
||||||
rpc Withdraw(WithdrawRequest) returns (WithdrawReply);
|
|
||||||
rpc Deposit(DepositRequest) returns (DepositReply);
|
|
||||||
}
|
|
||||||
|
|
||||||
message GetTransactionsRequest {
|
|
||||||
int32 page = 1;
|
|
||||||
uint64 userId = 2;
|
|
||||||
}
|
|
||||||
|
|
||||||
message GetTransactionsReply {
|
|
||||||
repeated TransactionReply transactions = 1;
|
|
||||||
int32 total = 2;
|
|
||||||
}
|
|
||||||
|
|
||||||
message TransactionReply {
|
|
||||||
int64 amount = 1;
|
|
||||||
string note = 2;
|
|
||||||
string type = 3;
|
|
||||||
string extra = 4;
|
|
||||||
google.protobuf.Timestamp timestamp = 5;
|
|
||||||
string id = 6;
|
|
||||||
}
|
|
||||||
|
|
||||||
message GetHoldingsRequest {
|
|
||||||
uint64 userId = 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
message GetHoldingsReply {
|
|
||||||
int64 cash = 1;
|
|
||||||
int64 bank = 2;
|
|
||||||
}
|
|
||||||
|
|
||||||
message WithdrawRequest {
|
|
||||||
uint64 userId = 1;
|
|
||||||
int64 amount = 2;
|
|
||||||
}
|
|
||||||
|
|
||||||
message WithdrawReply {
|
|
||||||
bool success = 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
message DepositRequest {
|
|
||||||
uint64 userId = 1;
|
|
||||||
int64 amount = 2;
|
|
||||||
}
|
|
||||||
|
|
||||||
message DepositReply {
|
|
||||||
bool success = 1;
|
|
||||||
}
|
|
|
@ -5,13 +5,20 @@ option csharp_namespace = "EllieBot.GrpcApi";
|
||||||
package greet;
|
package greet;
|
||||||
|
|
||||||
service GrpcGreet {
|
service GrpcGreet {
|
||||||
rpc GetGreetSettings (GetGreetRequest) returns (GrpcGreetSettings);
|
rpc GetGreetSettings (GetGreetRequest) returns (GetGreetReply);
|
||||||
rpc UpdateGreet (UpdateGreetRequest) returns (UpdateGreetReply);
|
rpc UpdateGreet (UpdateGreetRequest) returns (UpdateGreetReply);
|
||||||
rpc TestGreet (TestGreetRequest) returns (TestGreetReply);
|
rpc TestGreet (TestGreetRequest) returns (TestGreetReply);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
message GetGreetReply {
|
||||||
|
GrpcGreetSettings greet = 1;
|
||||||
|
GrpcGreetSettings greetDm = 2;
|
||||||
|
GrpcGreetSettings bye = 3;
|
||||||
|
GrpcGreetSettings boost = 4;
|
||||||
|
}
|
||||||
|
|
||||||
message GrpcGreetSettings {
|
message GrpcGreetSettings {
|
||||||
string channelId = 1;
|
optional uint64 channelId = 1;
|
||||||
string message = 2;
|
string message = 2;
|
||||||
bool isEnabled = 3;
|
bool isEnabled = 3;
|
||||||
GrpcGreetType type = 4;
|
GrpcGreetType type = 4;
|
||||||
|
@ -19,7 +26,6 @@ message GrpcGreetSettings {
|
||||||
|
|
||||||
message GetGreetRequest {
|
message GetGreetRequest {
|
||||||
uint64 guildId = 1;
|
uint64 guildId = 1;
|
||||||
GrpcGreetType type = 2;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
message UpdateGreetRequest {
|
message UpdateGreetRequest {
|
||||||
|
@ -35,7 +41,7 @@ enum GrpcGreetType {
|
||||||
}
|
}
|
||||||
|
|
||||||
message UpdateGreetReply {
|
message UpdateGreetReply {
|
||||||
bool Success = 1;
|
bool success = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
message TestGreetRequest {
|
message TestGreetRequest {
|
||||||
|
|
|
@ -3,50 +3,43 @@
|
||||||
option csharp_namespace = "EllieBot.GrpcApi";
|
option csharp_namespace = "EllieBot.GrpcApi";
|
||||||
|
|
||||||
import "google/protobuf/empty.proto";
|
import "google/protobuf/empty.proto";
|
||||||
|
import "google/protobuf/timestamp.proto";
|
||||||
|
|
||||||
package other;
|
package other;
|
||||||
|
|
||||||
service GrpcOther {
|
service GrpcOther {
|
||||||
rpc BotOnGuild(BotOnGuildRequest) returns (BotOnGuildReply);
|
|
||||||
|
rpc GetGuilds(google.protobuf.Empty) returns (GetGuildsReply);
|
||||||
rpc GetTextChannels(GetTextChannelsRequest) returns (GetTextChannelsReply);
|
rpc GetTextChannels(GetTextChannelsRequest) returns (GetTextChannelsReply);
|
||||||
rpc GetRoles(GetRolesRequest) returns (GetRolesReply);
|
|
||||||
|
|
||||||
rpc GetCurrencyLb(GetLbRequest) returns (CurrencyLbReply);
|
rpc GetCurrencyLb(GetLbRequest) returns (CurrencyLbReply);
|
||||||
rpc GetXpLb(GetLbRequest) returns (XpLbReply);
|
rpc GetXpLb(GetLbRequest) returns (XpLbReply);
|
||||||
rpc GetWaifuLb(GetLbRequest) returns (WaifuLbReply);
|
rpc GetWaifuLb(GetLbRequest) returns (WaifuLbReply);
|
||||||
|
|
||||||
rpc GetShardStats(google.protobuf.Empty) returns (stream ShardStatsReply);
|
rpc GetShardStatuses(google.protobuf.Empty) returns (GetShardStatusesReply);
|
||||||
rpc GetCommandFeed(google.protobuf.Empty) returns (stream CommandFeedEntry);
|
|
||||||
rpc GetServerInfo(ServerInfoRequest) returns (GetServerInfoReply);
|
rpc GetServerInfo(ServerInfoRequest) returns (GetServerInfoReply);
|
||||||
}
|
}
|
||||||
|
|
||||||
message CommandFeedEntry {
|
message GetGuildsReply {
|
||||||
string command = 1;
|
repeated GuildReply guilds = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
message GetRolesRequest {
|
message GuildReply {
|
||||||
uint64 guildId = 1;
|
uint64 id = 1;
|
||||||
|
string name = 2;
|
||||||
|
string iconUrl = 3;
|
||||||
}
|
}
|
||||||
|
|
||||||
message GetRolesReply {
|
message GetShardStatusesReply {
|
||||||
repeated RoleReply roles = 1;
|
repeated ShardStatusReply shards = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
message BotOnGuildRequest {
|
message ShardStatusReply {
|
||||||
uint64 guildId = 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
message BotOnGuildReply {
|
|
||||||
bool success = 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
message ShardStatsReply {
|
|
||||||
int32 id = 1;
|
int32 id = 1;
|
||||||
string status = 2;
|
string status = 2;
|
||||||
|
|
||||||
int32 guildCount = 3;
|
int32 guildCount = 3;
|
||||||
string uptime = 4;
|
google.protobuf.Timestamp lastUpdate = 4;
|
||||||
int64 commands = 5;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
message GetTextChannelsRequest{
|
message GetTextChannelsRequest{
|
||||||
|
|
|
@ -6,17 +6,11 @@ package warn;
|
||||||
|
|
||||||
service GrpcWarn {
|
service GrpcWarn {
|
||||||
rpc GetWarnSettings (WarnSettingsRequest) returns (WarnSettingsReply);
|
rpc GetWarnSettings (WarnSettingsRequest) returns (WarnSettingsReply);
|
||||||
|
|
||||||
rpc SetWarnExpiry(SetWarnExpiryRequest) returns (SetWarnExpiryReply);
|
|
||||||
rpc AddWarnp (AddWarnpRequest) returns (AddWarnpReply);
|
rpc AddWarnp (AddWarnpRequest) returns (AddWarnpReply);
|
||||||
rpc DeleteWarnp (DeleteWarnpRequest) returns (DeleteWarnpReply);
|
rpc DeleteWarnp (DeleteWarnpRequest) returns (DeleteWarnpReply);
|
||||||
|
|
||||||
rpc GetLatestWarnings(GetLatestWarningsRequest) returns (GetLatestWarningsReply);
|
|
||||||
rpc GetUserWarnings(GetUserWarningsRequest) returns (GetUserWarningsReply);
|
rpc GetUserWarnings(GetUserWarningsRequest) returns (GetUserWarningsReply);
|
||||||
|
rpc ClearWarning(ClearWarningRequest) returns (ClearWarningReply);
|
||||||
rpc ForgiveWarning(ForgiveWarningRequest) returns (ForgiveWarningReply);
|
rpc SetWarnExpiry(SetWarnExpiryRequest) returns (SetWarnExpiryReply);
|
||||||
rpc DeleteWarning(ForgiveWarningRequest) returns (ForgiveWarningReply);
|
|
||||||
|
|
||||||
}
|
}
|
||||||
message WarnSettingsRequest {
|
message WarnSettingsRequest {
|
||||||
uint64 guildId = 1;
|
uint64 guildId = 1;
|
||||||
|
@ -25,14 +19,12 @@ message WarnSettingsRequest {
|
||||||
message WarnPunishment {
|
message WarnPunishment {
|
||||||
int32 threshold = 1;
|
int32 threshold = 1;
|
||||||
string action = 2;
|
string action = 2;
|
||||||
int32 duration = 3;
|
int64 duration = 3;
|
||||||
string role = 4;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
message WarnSettingsReply {
|
message WarnSettingsReply {
|
||||||
repeated WarnPunishment punishments = 1;
|
repeated WarnPunishment punishments = 1;
|
||||||
int32 expiryDays = 2;
|
int32 expiryDays = 2;
|
||||||
bool deleteOnExpire = 3;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
message AddWarnpRequest {
|
message AddWarnpRequest {
|
||||||
|
@ -46,7 +38,7 @@ message AddWarnpReply {
|
||||||
|
|
||||||
message DeleteWarnpRequest {
|
message DeleteWarnpRequest {
|
||||||
uint64 guildId = 1;
|
uint64 guildId = 1;
|
||||||
int32 threshold = 2;
|
int32 warnpIndex = 2;
|
||||||
}
|
}
|
||||||
|
|
||||||
message DeleteWarnpReply {
|
message DeleteWarnpReply {
|
||||||
|
@ -55,53 +47,37 @@ message DeleteWarnpReply {
|
||||||
|
|
||||||
message GetUserWarningsRequest {
|
message GetUserWarningsRequest {
|
||||||
uint64 guildId = 1;
|
uint64 guildId = 1;
|
||||||
string user = 2;
|
uint64 user_id = 2;
|
||||||
int32 page = 3;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
message GetUserWarningsReply {
|
message GetUserWarningsReply {
|
||||||
repeated Warning warnings = 1;
|
repeated Warning warnings = 1;
|
||||||
int32 totalCount = 2;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
message Warning {
|
message Warning {
|
||||||
string id = 1;
|
int32 id = 1;
|
||||||
string reason = 2;
|
string reason = 2;
|
||||||
int64 timestamp = 3;
|
int64 timestamp = 3;
|
||||||
int64 weight = 4;
|
int64 expiry_timestamp = 4;
|
||||||
bool forgiven = 5;
|
bool cleared = 5;
|
||||||
string forgivenBy = 6;
|
string clearedBy = 6;
|
||||||
string user = 7;
|
|
||||||
uint64 userId = 8;
|
|
||||||
string moderator = 9;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
message ForgiveWarningRequest {
|
message ClearWarningRequest {
|
||||||
uint64 guildId = 1;
|
uint64 guildId = 1;
|
||||||
string warnId = 2;
|
uint64 userId = 2;
|
||||||
string modName = 3;
|
optional int32 warnId = 3;
|
||||||
}
|
}
|
||||||
|
|
||||||
message ForgiveWarningReply {
|
message ClearWarningReply {
|
||||||
bool success = 1;
|
bool success = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
message SetWarnExpiryRequest {
|
message SetWarnExpiryRequest {
|
||||||
uint64 guildId = 1;
|
uint64 guildId = 1;
|
||||||
int32 expiryDays = 2;
|
int32 expiryDays = 2;
|
||||||
bool deleteOnExpire = 3;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
message SetWarnExpiryReply {
|
message SetWarnExpiryReply {
|
||||||
bool success = 1;
|
bool success = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
message GetLatestWarningsRequest {
|
|
||||||
uint64 guildId = 1;
|
|
||||||
int32 page = 2;
|
|
||||||
}
|
|
||||||
|
|
||||||
message GetLatestWarningsReply {
|
|
||||||
repeated Warning warnings = 1;
|
|
||||||
int32 totalCount = 2;
|
|
||||||
}
|
|
|
@ -1,120 +0,0 @@
|
||||||
syntax = "proto3";
|
|
||||||
|
|
||||||
option csharp_namespace = "EllieBot.GrpcApi";
|
|
||||||
|
|
||||||
package xp;
|
|
||||||
|
|
||||||
service GrpcXp {
|
|
||||||
rpc GetXpLb(GetXpLbRequest) returns (GetXpLbReply);
|
|
||||||
rpc ResetUserXp(ResetUserXpRequest) returns (ResetUserXpReply);
|
|
||||||
|
|
||||||
rpc GetXpSettings(GetXpSettingsRequest) returns (GetXpSettingsReply);
|
|
||||||
|
|
||||||
rpc AddExclusion(AddExclusionRequest) returns (AddExclusionReply);
|
|
||||||
rpc DeleteExclusion(DeleteExclusionRequest) returns (DeleteExclusionReply);
|
|
||||||
|
|
||||||
rpc AddReward(AddRewardRequest) returns (AddRewardReply);
|
|
||||||
rpc DeleteReward(DeleteRewardRequest) returns (DeleteRewardReply);
|
|
||||||
|
|
||||||
rpc SetServerExclusion(SetServerExclusionRequest) returns (SetServerExclusionReply);
|
|
||||||
}
|
|
||||||
|
|
||||||
message SetServerExclusionRequest {
|
|
||||||
uint64 guildId = 1;
|
|
||||||
bool serverExcluded = 2;
|
|
||||||
}
|
|
||||||
|
|
||||||
message SetServerExclusionReply {
|
|
||||||
bool success = 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
message GetXpLbRequest {
|
|
||||||
uint64 guildId = 1;
|
|
||||||
int32 page = 2;
|
|
||||||
}
|
|
||||||
|
|
||||||
message GetXpLbReply {
|
|
||||||
repeated XpLbUserReply users = 1;
|
|
||||||
int32 total = 2;
|
|
||||||
}
|
|
||||||
|
|
||||||
message XpLbUserReply {
|
|
||||||
uint64 userId = 1;
|
|
||||||
string username = 2;
|
|
||||||
int64 xp = 3;
|
|
||||||
int64 level = 4;
|
|
||||||
int64 levelPercent = 5;
|
|
||||||
string avatar = 6;
|
|
||||||
}
|
|
||||||
|
|
||||||
message ResetUserXpRequest {
|
|
||||||
uint64 guildId = 1;
|
|
||||||
uint64 userId = 2;
|
|
||||||
}
|
|
||||||
|
|
||||||
message ResetUserXpReply {
|
|
||||||
bool success = 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
message GetXpSettingsReply {
|
|
||||||
repeated ExclItemReply exclusions = 1;
|
|
||||||
repeated RewItemReply rewards = 2;
|
|
||||||
bool serverExcluded = 3;
|
|
||||||
}
|
|
||||||
|
|
||||||
message GetXpSettingsRequest {
|
|
||||||
uint64 guildId = 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
message ExclItemReply {
|
|
||||||
string type = 1;
|
|
||||||
uint64 id = 2;
|
|
||||||
string name = 3;
|
|
||||||
}
|
|
||||||
|
|
||||||
message RewItemReply {
|
|
||||||
int32 level = 1;
|
|
||||||
string type = 2;
|
|
||||||
string value = 3;
|
|
||||||
}
|
|
||||||
|
|
||||||
message AddExclusionRequest {
|
|
||||||
uint64 guildId = 1;
|
|
||||||
string type = 2;
|
|
||||||
uint64 id = 3;
|
|
||||||
}
|
|
||||||
|
|
||||||
message AddExclusionReply {
|
|
||||||
bool success = 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
message DeleteExclusionRequest {
|
|
||||||
uint64 guildId = 1;
|
|
||||||
string type = 2;
|
|
||||||
uint64 id = 3;
|
|
||||||
}
|
|
||||||
|
|
||||||
message DeleteExclusionReply {
|
|
||||||
bool success = 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
message AddRewardRequest {
|
|
||||||
uint64 guildId = 1;
|
|
||||||
int32 level = 2;
|
|
||||||
string type = 3;
|
|
||||||
string value = 4;
|
|
||||||
}
|
|
||||||
|
|
||||||
message AddRewardReply {
|
|
||||||
bool success = 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
message DeleteRewardRequest {
|
|
||||||
uint64 guildId = 1;
|
|
||||||
int32 level = 2;
|
|
||||||
string type = 3;
|
|
||||||
}
|
|
||||||
|
|
||||||
message DeleteRewardReply {
|
|
||||||
bool success = 1;
|
|
||||||
}
|
|
|
@ -356,5 +356,3 @@ resharper_arrange_redundant_parentheses_highlighting = hint
|
||||||
|
|
||||||
# IDE0011: Add braces
|
# IDE0011: Add braces
|
||||||
dotnet_diagnostic.IDE0011.severity = warning
|
dotnet_diagnostic.IDE0011.severity = warning
|
||||||
|
|
||||||
resharper_arrange_type_member_modifiers_highlighting = hint
|
|
|
@ -25,7 +25,7 @@ public sealed class Bot : IBot
|
||||||
public bool IsReady { get; private set; }
|
public bool IsReady { get; private set; }
|
||||||
public int ShardId { get; set; }
|
public int ShardId { get; set; }
|
||||||
|
|
||||||
private readonly IBotCreds _creds;
|
private readonly IBotCredentials _creds;
|
||||||
private readonly CommandService _commandService;
|
private readonly CommandService _commandService;
|
||||||
private readonly DbService _db;
|
private readonly DbService _db;
|
||||||
|
|
||||||
|
@ -42,9 +42,6 @@ public sealed class Bot : IBot
|
||||||
_credsProvider = new BotCredsProvider(totalShards, credPath);
|
_credsProvider = new BotCredsProvider(totalShards, credPath);
|
||||||
_creds = _credsProvider.GetCreds();
|
_creds = _credsProvider.GetCreds();
|
||||||
|
|
||||||
LogSetup.SetupLogger(shardId, _creds);
|
|
||||||
Log.Information("Pid: {ProcessId}", Environment.ProcessId);
|
|
||||||
|
|
||||||
_db = new EllieDbService(_credsProvider);
|
_db = new EllieDbService(_credsProvider);
|
||||||
|
|
||||||
var messageCacheSize =
|
var messageCacheSize =
|
||||||
|
@ -118,7 +115,7 @@ public sealed class Bot : IBot
|
||||||
// svcs.Components.Remove<IPlanner, Planner>();
|
// svcs.Components.Remove<IPlanner, Planner>();
|
||||||
// svcs.Components.Add<IPlanner, RemovablePlanner>();
|
// svcs.Components.Add<IPlanner, RemovablePlanner>();
|
||||||
|
|
||||||
svcs.AddSingleton<IBotCreds>(_ => _credsProvider.GetCreds());
|
svcs.AddSingleton<IBotCredentials>(_ => _credsProvider.GetCreds());
|
||||||
svcs.AddSingleton<DbService, DbService>(_db);
|
svcs.AddSingleton<DbService, DbService>(_db);
|
||||||
svcs.AddSingleton<IBotCredsProvider>(_credsProvider);
|
svcs.AddSingleton<IBotCredsProvider>(_credsProvider);
|
||||||
svcs.AddSingleton<DiscordSocketClient>(Client);
|
svcs.AddSingleton<DiscordSocketClient>(Client);
|
||||||
|
|
|
@ -2,7 +2,6 @@
|
||||||
using Microsoft.EntityFrameworkCore;
|
using Microsoft.EntityFrameworkCore;
|
||||||
using Microsoft.Extensions.Logging;
|
using Microsoft.Extensions.Logging;
|
||||||
using EllieBot.Db.Models;
|
using EllieBot.Db.Models;
|
||||||
using EllieBot.Modules.Administration.Services;
|
|
||||||
|
|
||||||
// ReSharper disable UnusedAutoPropertyAccessor.Global
|
// ReSharper disable UnusedAutoPropertyAccessor.Global
|
||||||
|
|
||||||
|
@ -15,6 +14,7 @@ public abstract class EllieContext : DbContext
|
||||||
|
|
||||||
public DbSet<Quote> Quotes { get; set; }
|
public DbSet<Quote> Quotes { get; set; }
|
||||||
public DbSet<Reminder> Reminders { get; set; }
|
public DbSet<Reminder> Reminders { get; set; }
|
||||||
|
public DbSet<SelfAssignedRole> SelfAssignableRoles { get; set; }
|
||||||
public DbSet<MusicPlaylist> MusicPlaylists { get; set; }
|
public DbSet<MusicPlaylist> MusicPlaylists { get; set; }
|
||||||
public DbSet<EllieExpression> Expressions { get; set; }
|
public DbSet<EllieExpression> Expressions { get; set; }
|
||||||
public DbSet<CurrencyTransaction> CurrencyTransactions { get; set; }
|
public DbSet<CurrencyTransaction> CurrencyTransactions { get; set; }
|
||||||
|
@ -62,7 +62,7 @@ public abstract class EllieContext : DbContext
|
||||||
public DbSet<ArchivedTodoListModel> TodosArchive { get; set; }
|
public DbSet<ArchivedTodoListModel> TodosArchive { get; set; }
|
||||||
public DbSet<HoneypotChannel> HoneyPotChannels { get; set; }
|
public DbSet<HoneypotChannel> HoneyPotChannels { get; set; }
|
||||||
|
|
||||||
|
// todo add guild colors
|
||||||
// public DbSet<GuildColors> GuildColors { get; set; }
|
// public DbSet<GuildColors> GuildColors { get; set; }
|
||||||
|
|
||||||
|
|
||||||
|
@ -74,133 +74,6 @@ public abstract class EllieContext : DbContext
|
||||||
|
|
||||||
protected override void OnModelCreating(ModelBuilder modelBuilder)
|
protected override void OnModelCreating(ModelBuilder modelBuilder)
|
||||||
{
|
{
|
||||||
#region Notify
|
|
||||||
|
|
||||||
modelBuilder.Entity<Notify>(e =>
|
|
||||||
{
|
|
||||||
e.HasAlternateKey(x => new
|
|
||||||
{
|
|
||||||
x.GuildId,
|
|
||||||
Event = x.Type
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
#endregion
|
|
||||||
|
|
||||||
#region TempRoles
|
|
||||||
|
|
||||||
modelBuilder.Entity<TempRole>(e =>
|
|
||||||
{
|
|
||||||
e.HasAlternateKey(x => new
|
|
||||||
{
|
|
||||||
x.GuildId,
|
|
||||||
x.UserId,
|
|
||||||
x.RoleId
|
|
||||||
});
|
|
||||||
|
|
||||||
e.HasIndex(x => x.ExpiresAt);
|
|
||||||
});
|
|
||||||
|
|
||||||
#endregion
|
|
||||||
|
|
||||||
#region GuildColors
|
|
||||||
|
|
||||||
modelBuilder.Entity<GuildColors>()
|
|
||||||
.HasIndex(x => x.GuildId)
|
|
||||||
.IsUnique(true);
|
|
||||||
|
|
||||||
#endregion
|
|
||||||
|
|
||||||
#region Button Roles
|
|
||||||
|
|
||||||
modelBuilder.Entity<ButtonRole>(br =>
|
|
||||||
{
|
|
||||||
br.HasIndex(x => x.GuildId)
|
|
||||||
.IsUnique(false);
|
|
||||||
|
|
||||||
br.HasAlternateKey(x => new
|
|
||||||
{
|
|
||||||
x.RoleId,
|
|
||||||
x.MessageId,
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
#endregion
|
|
||||||
|
|
||||||
#region New Sar
|
|
||||||
|
|
||||||
modelBuilder.Entity<SarGroup>(sg =>
|
|
||||||
{
|
|
||||||
sg.HasAlternateKey(x => new
|
|
||||||
{
|
|
||||||
x.GuildId,
|
|
||||||
x.GroupNumber
|
|
||||||
});
|
|
||||||
|
|
||||||
sg.HasMany(x => x.Roles)
|
|
||||||
.WithOne()
|
|
||||||
.OnDelete(DeleteBehavior.Cascade);
|
|
||||||
});
|
|
||||||
|
|
||||||
modelBuilder.Entity<Sar>()
|
|
||||||
.HasAlternateKey(x => new
|
|
||||||
{
|
|
||||||
x.GuildId,
|
|
||||||
x.RoleId
|
|
||||||
});
|
|
||||||
|
|
||||||
modelBuilder.Entity<SarAutoDelete>()
|
|
||||||
.HasIndex(x => x.GuildId)
|
|
||||||
.IsUnique();
|
|
||||||
|
|
||||||
#endregion
|
|
||||||
|
|
||||||
#region Rakeback
|
|
||||||
|
|
||||||
modelBuilder.Entity<Rakeback>()
|
|
||||||
.HasKey(x => x.UserId);
|
|
||||||
|
|
||||||
#endregion
|
|
||||||
|
|
||||||
#region UserBetStats
|
|
||||||
|
|
||||||
modelBuilder.Entity<UserBetStats>(ubs =>
|
|
||||||
{
|
|
||||||
ubs.HasIndex(x => new
|
|
||||||
{
|
|
||||||
x.UserId,
|
|
||||||
x.Game
|
|
||||||
})
|
|
||||||
.IsUnique();
|
|
||||||
|
|
||||||
ubs.HasIndex(x => x.MaxWin)
|
|
||||||
.IsUnique(false);
|
|
||||||
});
|
|
||||||
|
|
||||||
#endregion
|
|
||||||
|
|
||||||
#region Flag Translate
|
|
||||||
|
|
||||||
modelBuilder.Entity<FlagTranslateChannel>()
|
|
||||||
.HasIndex(x => new
|
|
||||||
{
|
|
||||||
x.GuildId,
|
|
||||||
x.ChannelId
|
|
||||||
})
|
|
||||||
.IsUnique();
|
|
||||||
|
|
||||||
#endregion
|
|
||||||
|
|
||||||
#region NCanvas
|
|
||||||
|
|
||||||
modelBuilder.Entity<NCPixel>()
|
|
||||||
.HasAlternateKey(x => x.Position);
|
|
||||||
|
|
||||||
modelBuilder.Entity<NCPixel>()
|
|
||||||
.HasIndex(x => x.OwnerId);
|
|
||||||
|
|
||||||
#endregion
|
|
||||||
|
|
||||||
#region QUOTES
|
#region QUOTES
|
||||||
|
|
||||||
var quoteEntity = modelBuilder.Entity<Quote>();
|
var quoteEntity = modelBuilder.Entity<Quote>();
|
||||||
|
@ -322,6 +195,11 @@ public abstract class EllieContext : DbContext
|
||||||
.WithOne()
|
.WithOne()
|
||||||
.OnDelete(DeleteBehavior.Cascade);
|
.OnDelete(DeleteBehavior.Cascade);
|
||||||
|
|
||||||
|
modelBuilder.Entity<GuildConfig>()
|
||||||
|
.HasMany(x => x.WarnPunishments)
|
||||||
|
.WithOne()
|
||||||
|
.OnDelete(DeleteBehavior.Cascade);
|
||||||
|
|
||||||
modelBuilder.Entity<GuildConfig>()
|
modelBuilder.Entity<GuildConfig>()
|
||||||
.HasMany(x => x.SlowmodeIgnoredRoles)
|
.HasMany(x => x.SlowmodeIgnoredRoles)
|
||||||
.WithOne()
|
.WithOne()
|
||||||
|
@ -379,6 +257,11 @@ public abstract class EllieContext : DbContext
|
||||||
.HasForeignKey(x => x.GuildConfigId)
|
.HasForeignKey(x => x.GuildConfigId)
|
||||||
.OnDelete(DeleteBehavior.Cascade);
|
.OnDelete(DeleteBehavior.Cascade);
|
||||||
|
|
||||||
|
modelBuilder.Entity<GuildConfig>()
|
||||||
|
.HasMany(x => x.SelfAssignableRoleGroupNames)
|
||||||
|
.WithOne()
|
||||||
|
.OnDelete(DeleteBehavior.Cascade);
|
||||||
|
|
||||||
modelBuilder.Entity<FeedSub>()
|
modelBuilder.Entity<FeedSub>()
|
||||||
.HasAlternateKey(x => new
|
.HasAlternateKey(x => new
|
||||||
{
|
{
|
||||||
|
@ -394,16 +277,19 @@ public abstract class EllieContext : DbContext
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
#region WarningPunishments
|
|
||||||
|
|
||||||
var warnpunishmentEntity = modelBuilder.Entity<WarningPunishment>(b =>
|
#region Self Assignable Roles
|
||||||
{
|
|
||||||
b.HasAlternateKey(x => new
|
var selfassignableRolesEntity = modelBuilder.Entity<SelfAssignedRole>();
|
||||||
{
|
|
||||||
x.GuildId,
|
selfassignableRolesEntity.HasIndex(s => new
|
||||||
x.Count
|
{
|
||||||
});
|
s.GuildId,
|
||||||
});
|
s.RoleId
|
||||||
|
})
|
||||||
|
.IsUnique();
|
||||||
|
|
||||||
|
selfassignableRolesEntity.Property(x => x.Group).HasDefaultValue(0);
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
|
@ -453,7 +339,6 @@ public abstract class EllieContext : DbContext
|
||||||
du.HasIndex(x => x.TotalXp);
|
du.HasIndex(x => x.TotalXp);
|
||||||
du.HasIndex(x => x.CurrencyAmount);
|
du.HasIndex(x => x.CurrencyAmount);
|
||||||
du.HasIndex(x => x.UserId);
|
du.HasIndex(x => x.UserId);
|
||||||
du.HasIndex(x => x.Username);
|
|
||||||
});
|
});
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
@ -483,6 +368,7 @@ public abstract class EllieContext : DbContext
|
||||||
xps.HasIndex(x => x.UserId);
|
xps.HasIndex(x => x.UserId);
|
||||||
xps.HasIndex(x => x.GuildId);
|
xps.HasIndex(x => x.GuildId);
|
||||||
xps.HasIndex(x => x.Xp);
|
xps.HasIndex(x => x.Xp);
|
||||||
|
xps.HasIndex(x => x.AwardedXp);
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
|
@ -588,6 +474,23 @@ public abstract class EllieContext : DbContext
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
|
#region GroupName
|
||||||
|
|
||||||
|
modelBuilder.Entity<GroupName>()
|
||||||
|
.HasIndex(x => new
|
||||||
|
{
|
||||||
|
x.GuildConfigId,
|
||||||
|
x.Number
|
||||||
|
})
|
||||||
|
.IsUnique();
|
||||||
|
|
||||||
|
modelBuilder.Entity<GroupName>()
|
||||||
|
.HasOne(x => x.GuildConfig)
|
||||||
|
.WithMany(x => x.SelfAssignableRoleGroupNames)
|
||||||
|
.IsRequired();
|
||||||
|
|
||||||
|
#endregion
|
||||||
|
|
||||||
#region BanTemplate
|
#region BanTemplate
|
||||||
|
|
||||||
modelBuilder.Entity<BanTemplate>().HasIndex(x => x.GuildId).IsUnique();
|
modelBuilder.Entity<BanTemplate>().HasIndex(x => x.GuildId).IsUnique();
|
||||||
|
@ -792,7 +695,7 @@ public abstract class EllieContext : DbContext
|
||||||
gs
|
gs
|
||||||
.Property(x => x.IsEnabled)
|
.Property(x => x.IsEnabled)
|
||||||
.HasDefaultValue(false);
|
.HasDefaultValue(false);
|
||||||
|
|
||||||
gs
|
gs
|
||||||
.Property(x => x.AutoDeleteTimer)
|
.Property(x => x.AutoDeleteTimer)
|
||||||
.HasDefaultValue(0);
|
.HasDefaultValue(0);
|
||||||
|
|
|
@ -25,6 +25,7 @@ public static class DiscordUserExtensions
|
||||||
{
|
{
|
||||||
UserId = userId,
|
UserId = userId,
|
||||||
Username = username,
|
Username = username,
|
||||||
|
Discriminator = discrim,
|
||||||
AvatarId = avatarId,
|
AvatarId = avatarId,
|
||||||
TotalXp = 0,
|
TotalXp = 0,
|
||||||
CurrencyAmount = 0
|
CurrencyAmount = 0
|
||||||
|
@ -32,6 +33,7 @@ public static class DiscordUserExtensions
|
||||||
old => new()
|
old => new()
|
||||||
{
|
{
|
||||||
Username = username,
|
Username = username,
|
||||||
|
Discriminator = discrim,
|
||||||
AvatarId = avatarId
|
AvatarId = avatarId
|
||||||
},
|
},
|
||||||
() => new()
|
() => new()
|
||||||
|
@ -47,7 +49,8 @@ public static class DiscordUserExtensions
|
||||||
() => new()
|
() => new()
|
||||||
{
|
{
|
||||||
UserId = userId,
|
UserId = userId,
|
||||||
Username = "??Unknown",
|
Username = "Unknown",
|
||||||
|
Discriminator = "????",
|
||||||
AvatarId = string.Empty,
|
AvatarId = string.Empty,
|
||||||
TotalXp = 0,
|
TotalXp = 0,
|
||||||
CurrencyAmount = 0
|
CurrencyAmount = 0
|
||||||
|
@ -85,6 +88,13 @@ public static class DiscordUserExtensions
|
||||||
.Count()
|
.Count()
|
||||||
+ 1;
|
+ 1;
|
||||||
|
|
||||||
|
public static async Task<IReadOnlyCollection<DiscordUser>> GetUsersXpLeaderboardFor(this DbSet<DiscordUser> users, int page, int perPage)
|
||||||
|
=> await users.ToLinqToDBTable()
|
||||||
|
.OrderByDescending(x => x.TotalXp)
|
||||||
|
.Skip(page * perPage)
|
||||||
|
.Take(perPage)
|
||||||
|
.ToArrayAsyncLinqToDB();
|
||||||
|
|
||||||
public static Task<List<DiscordUser>> GetTopRichest(
|
public static Task<List<DiscordUser>> GetTopRichest(
|
||||||
this DbSet<DiscordUser> users,
|
this DbSet<DiscordUser> users,
|
||||||
ulong botId,
|
ulong botId,
|
||||||
|
|
|
@ -57,7 +57,8 @@ public static class GuildConfigExtensions
|
||||||
List<ulong> availableGuilds)
|
List<ulong> availableGuilds)
|
||||||
{
|
{
|
||||||
var result = await configs
|
var result = await configs
|
||||||
.IncludeEverything()
|
.AsQueryable()
|
||||||
|
.Include(x => x.CommandCooldowns)
|
||||||
.Where(x => availableGuilds.Contains(x.GuildId))
|
.Where(x => availableGuilds.Contains(x.GuildId))
|
||||||
.AsNoTracking()
|
.AsNoTracking()
|
||||||
.ToArrayAsync();
|
.ToArrayAsync();
|
||||||
|
@ -95,6 +96,7 @@ public static class GuildConfigExtensions
|
||||||
GuildId = guildId,
|
GuildId = guildId,
|
||||||
Permissions = Permissionv2.GetDefaultPermlist,
|
Permissions = Permissionv2.GetDefaultPermlist,
|
||||||
WarningsInitialized = true,
|
WarningsInitialized = true,
|
||||||
|
WarnPunishments = DefaultWarnPunishments
|
||||||
});
|
});
|
||||||
ctx.SaveChanges();
|
ctx.SaveChanges();
|
||||||
}
|
}
|
||||||
|
@ -102,6 +104,7 @@ public static class GuildConfigExtensions
|
||||||
if (!config.WarningsInitialized)
|
if (!config.WarningsInitialized)
|
||||||
{
|
{
|
||||||
config.WarningsInitialized = true;
|
config.WarningsInitialized = true;
|
||||||
|
config.WarnPunishments = DefaultWarnPunishments;
|
||||||
}
|
}
|
||||||
|
|
||||||
return config;
|
return config;
|
||||||
|
|
22
src/EllieBot/Db/Extensions/SelfAssignableRolesExtensions.cs
Normal file
22
src/EllieBot/Db/Extensions/SelfAssignableRolesExtensions.cs
Normal file
|
@ -0,0 +1,22 @@
|
||||||
|
#nullable disable
|
||||||
|
using Microsoft.EntityFrameworkCore;
|
||||||
|
using EllieBot.Db.Models;
|
||||||
|
|
||||||
|
namespace EllieBot.Db;
|
||||||
|
|
||||||
|
public static class SelfAssignableRolesExtensions
|
||||||
|
{
|
||||||
|
public static bool DeleteByGuildAndRoleId(this DbSet<SelfAssignedRole> roles, ulong guildId, ulong roleId)
|
||||||
|
{
|
||||||
|
var role = roles.FirstOrDefault(s => s.GuildId == guildId && s.RoleId == roleId);
|
||||||
|
|
||||||
|
if (role is null)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
roles.Remove(role);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static IReadOnlyCollection<SelfAssignedRole> GetFromGuild(this DbSet<SelfAssignedRole> roles, ulong guildId)
|
||||||
|
=> roles.AsQueryable().Where(s => s.GuildId == guildId).ToArray();
|
||||||
|
}
|
|
@ -1,4 +1,5 @@
|
||||||
using LinqToDB;
|
#nullable disable
|
||||||
|
using LinqToDB;
|
||||||
using LinqToDB.EntityFrameworkCore;
|
using LinqToDB.EntityFrameworkCore;
|
||||||
using Microsoft.EntityFrameworkCore;
|
using Microsoft.EntityFrameworkCore;
|
||||||
using EllieBot.Db.Models;
|
using EllieBot.Db.Models;
|
||||||
|
@ -7,9 +8,6 @@ namespace EllieBot.Db;
|
||||||
|
|
||||||
public static class UserXpExtensions
|
public static class UserXpExtensions
|
||||||
{
|
{
|
||||||
public static async Task<UserXpStats?> GetGuildUserXp(this ITable<UserXpStats> table, ulong guildId, ulong userId)
|
|
||||||
=> await table.FirstOrDefaultAsyncLinqToDB(x => x.GuildId == guildId && x.UserId == userId);
|
|
||||||
|
|
||||||
public static UserXpStats GetOrCreateUserXpStats(this DbContext ctx, ulong guildId, ulong userId)
|
public static UserXpStats GetOrCreateUserXpStats(this DbContext ctx, ulong guildId, ulong userId)
|
||||||
{
|
{
|
||||||
var usr = ctx.Set<UserXpStats>().FirstOrDefault(x => x.UserId == userId && x.GuildId == guildId);
|
var usr = ctx.Set<UserXpStats>().FirstOrDefault(x => x.UserId == userId && x.GuildId == guildId);
|
||||||
|
@ -20,6 +18,7 @@ public static class UserXpExtensions
|
||||||
{
|
{
|
||||||
Xp = 0,
|
Xp = 0,
|
||||||
UserId = userId,
|
UserId = userId,
|
||||||
|
NotifyOnLevelUp = XpNotificationLocation.None,
|
||||||
GuildId = guildId
|
GuildId = guildId
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@ -27,24 +26,38 @@ public static class UserXpExtensions
|
||||||
return usr;
|
return usr;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static async Task<IReadOnlyCollection<UserXpStats>> GetUsersFor(
|
||||||
|
this DbSet<UserXpStats> xps,
|
||||||
|
ulong guildId,
|
||||||
|
int page)
|
||||||
|
=> await xps.ToLinqToDBTable()
|
||||||
|
.Where(x => x.GuildId == guildId)
|
||||||
|
.OrderByDescending(x => x.Xp + x.AwardedXp)
|
||||||
|
.Skip(page * 9)
|
||||||
|
.Take(9)
|
||||||
|
.ToArrayAsyncLinqToDB();
|
||||||
|
|
||||||
public static async Task<List<UserXpStats>> GetTopUserXps(this DbSet<UserXpStats> xps, ulong guildId, int count)
|
public static async Task<List<UserXpStats>> GetTopUserXps(this DbSet<UserXpStats> xps, ulong guildId, int count)
|
||||||
=> await xps.ToLinqToDBTable()
|
=> await xps.ToLinqToDBTable()
|
||||||
.Where(x => x.GuildId == guildId)
|
.Where(x => x.GuildId == guildId)
|
||||||
.OrderByDescending(x => x.Xp)
|
.OrderByDescending(x => x.Xp + x.AwardedXp)
|
||||||
.Take(count)
|
.Take(count)
|
||||||
.ToListAsyncLinqToDB();
|
.ToListAsyncLinqToDB();
|
||||||
|
|
||||||
public static async Task<int> GetUserGuildRanking(this DbSet<UserXpStats> xps, ulong userId, ulong guildId)
|
public static async Task<int> GetUserGuildRanking(this DbSet<UserXpStats> xps, ulong userId, ulong guildId)
|
||||||
=> await xps.ToLinqToDBTable()
|
=> await xps.ToLinqToDBTable()
|
||||||
.Where(x => x.GuildId == guildId
|
.Where(x => x.GuildId == guildId
|
||||||
&& x.Xp
|
&& x.Xp + x.AwardedXp
|
||||||
> xps.AsQueryable()
|
> xps.AsQueryable()
|
||||||
.Where(y => y.UserId == userId && y.GuildId == guildId)
|
.Where(y => y.UserId == userId && y.GuildId == guildId)
|
||||||
.Select(y => y.Xp)
|
.Select(y => y.Xp + y.AwardedXp)
|
||||||
.FirstOrDefault())
|
.FirstOrDefault())
|
||||||
.CountAsyncLinqToDB()
|
.CountAsyncLinqToDB()
|
||||||
+ 1;
|
+ 1;
|
||||||
|
|
||||||
|
public static void ResetGuildUserXp(this DbSet<UserXpStats> xps, ulong userId, ulong guildId)
|
||||||
|
=> xps.Delete(x => x.UserId == userId && x.GuildId == guildId);
|
||||||
|
|
||||||
public static void ResetGuildXp(this DbSet<UserXpStats> xps, ulong guildId)
|
public static void ResetGuildXp(this DbSet<UserXpStats> xps, ulong guildId)
|
||||||
=> xps.Delete(x => x.GuildId == guildId);
|
=> xps.Delete(x => x.GuildId == guildId);
|
||||||
|
|
||||||
|
@ -52,6 +65,6 @@ public static class UserXpExtensions
|
||||||
=> await userXp
|
=> await userXp
|
||||||
.Where(x => x.GuildId == guildId && x.UserId == userId)
|
.Where(x => x.GuildId == guildId && x.UserId == userId)
|
||||||
.FirstOrDefaultAsyncLinqToDB() is UserXpStats uxs
|
.FirstOrDefaultAsyncLinqToDB() is UserXpStats uxs
|
||||||
? new(uxs.Xp)
|
? new(uxs.Xp + uxs.AwardedXp)
|
||||||
: new(0);
|
: new(0);
|
||||||
}
|
}
|
|
@ -3,28 +3,38 @@ namespace EllieBot.Db;
|
||||||
|
|
||||||
public readonly struct LevelStats
|
public readonly struct LevelStats
|
||||||
{
|
{
|
||||||
|
public const int XP_REQUIRED_LVL_1 = 36;
|
||||||
|
|
||||||
public long Level { get; }
|
public long Level { get; }
|
||||||
public long LevelXp { get; }
|
public long LevelXp { get; }
|
||||||
public long RequiredXp { get; }
|
public long RequiredXp { get; }
|
||||||
public long TotalXp { get; }
|
public long TotalXp { get; }
|
||||||
|
|
||||||
public LevelStats(long totalXp)
|
public LevelStats(long xp)
|
||||||
{
|
{
|
||||||
if (totalXp < 0)
|
if (xp < 0)
|
||||||
totalXp = 0;
|
xp = 0;
|
||||||
|
|
||||||
TotalXp = totalXp;
|
TotalXp = xp;
|
||||||
Level = GetLevelByTotalXp(totalXp);
|
|
||||||
LevelXp = totalXp - GetTotalXpReqForLevel(Level);
|
const int baseXp = XP_REQUIRED_LVL_1;
|
||||||
RequiredXp = (9 * (Level + 1)) + 27;
|
|
||||||
|
var required = baseXp;
|
||||||
|
var totalXp = 0;
|
||||||
|
var lvl = 1;
|
||||||
|
while (true)
|
||||||
|
{
|
||||||
|
required = (int)(baseXp + (baseXp / 4.0 * (lvl - 1)));
|
||||||
|
|
||||||
|
if (required + totalXp > xp)
|
||||||
|
break;
|
||||||
|
|
||||||
|
totalXp += required;
|
||||||
|
lvl++;
|
||||||
|
}
|
||||||
|
|
||||||
|
Level = lvl - 1;
|
||||||
|
LevelXp = xp - totalXp;
|
||||||
|
RequiredXp = required;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static LevelStats CreateForLevel(long level)
|
|
||||||
=> new(GetTotalXpReqForLevel(level));
|
|
||||||
|
|
||||||
public static long GetTotalXpReqForLevel(long level)
|
|
||||||
=> ((9 * level * level) + (63 * level)) / 2;
|
|
||||||
|
|
||||||
public static long GetLevelByTotalXp(long totalXp)
|
|
||||||
=> (long)((-7.0 / 2) + (1 / 6.0 * Math.Sqrt((8 * totalXp) + 441)));
|
|
||||||
}
|
}
|
|
@ -6,7 +6,7 @@ public class DiscordUser : DbEntity
|
||||||
{
|
{
|
||||||
public ulong UserId { get; set; }
|
public ulong UserId { get; set; }
|
||||||
public string Username { get; set; }
|
public string Username { get; set; }
|
||||||
// public string Discriminator { get; set; }
|
public string Discriminator { get; set; }
|
||||||
public string AvatarId { get; set; }
|
public string AvatarId { get; set; }
|
||||||
|
|
||||||
public int? ClubId { get; set; }
|
public int? ClubId { get; set; }
|
||||||
|
@ -26,6 +26,9 @@ public class DiscordUser : DbEntity
|
||||||
|
|
||||||
public override string ToString()
|
public override string ToString()
|
||||||
{
|
{
|
||||||
return Username;
|
if (string.IsNullOrWhiteSpace(Discriminator) || Discriminator == "0000")
|
||||||
|
return Username;
|
||||||
|
|
||||||
|
return Username + "#" + Discriminator;
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -1,8 +0,0 @@
|
||||||
#nullable disable
|
|
||||||
namespace EllieBot.Db.Models;
|
|
||||||
|
|
||||||
public class FlagTranslateChannel : DbEntity
|
|
||||||
{
|
|
||||||
public ulong GuildId { get; set; }
|
|
||||||
public ulong ChannelId { get; set; }
|
|
||||||
}
|
|
|
@ -9,8 +9,7 @@ public class FollowedStream : DbEntity
|
||||||
Picarto = 3,
|
Picarto = 3,
|
||||||
Youtube = 4,
|
Youtube = 4,
|
||||||
Facebook = 5,
|
Facebook = 5,
|
||||||
Trovo = 6,
|
Trovo = 6
|
||||||
Kick = 7,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public ulong GuildId { get; set; }
|
public ulong GuildId { get; set; }
|
||||||
|
|
11
src/EllieBot/Db/Models/GroupName.cs
Normal file
11
src/EllieBot/Db/Models/GroupName.cs
Normal file
|
@ -0,0 +1,11 @@
|
||||||
|
#nullable disable
|
||||||
|
namespace EllieBot.Db.Models;
|
||||||
|
|
||||||
|
public class GroupName : DbEntity
|
||||||
|
{
|
||||||
|
public int GuildConfigId { get; set; }
|
||||||
|
public GuildConfig GuildConfig { get; set; }
|
||||||
|
|
||||||
|
public int Number { get; set; }
|
||||||
|
public string Name { get; set; }
|
||||||
|
}
|
|
@ -5,15 +5,14 @@ namespace EllieBot.Db.Models;
|
||||||
public class GuildColors
|
public class GuildColors
|
||||||
{
|
{
|
||||||
[Key]
|
[Key]
|
||||||
public int Id { get; set; }
|
|
||||||
public ulong GuildId { get; set; }
|
public ulong GuildId { get; set; }
|
||||||
|
|
||||||
[MaxLength(9)]
|
[Length(0, 9)]
|
||||||
public string? OkColor { get; set; }
|
public string? OkColor { get; set; }
|
||||||
|
|
||||||
[MaxLength(9)]
|
[Length(0, 9)]
|
||||||
public string? ErrorColor { get; set; }
|
public string? ErrorColor { get; set; }
|
||||||
|
|
||||||
[MaxLength(9)]
|
[Length(0, 9)]
|
||||||
public string? PendingColor { get; set; }
|
public string? PendingColor { get; set; }
|
||||||
}
|
}
|
|
@ -13,11 +13,28 @@ public class GuildConfig : DbEntity
|
||||||
|
|
||||||
public string AutoAssignRoleIds { get; set; }
|
public string AutoAssignRoleIds { get; set; }
|
||||||
|
|
||||||
//todo FUTURE: DELETE, UNUSED
|
// //greet stuff
|
||||||
|
// public int AutoDeleteGreetMessagesTimer { get; set; } = 30;
|
||||||
|
// public int AutoDeleteByeMessagesTimer { get; set; } = 30;
|
||||||
|
//
|
||||||
|
// public ulong GreetMessageChannelId { get; set; }
|
||||||
|
// public ulong ByeMessageChannelId { get; set; }
|
||||||
|
//
|
||||||
|
// public bool SendDmGreetMessage { get; set; }
|
||||||
|
// public string DmGreetMessageText { get; set; } = "Welcome to the %server% server, %user%!";
|
||||||
|
//
|
||||||
|
// public bool SendChannelGreetMessage { get; set; }
|
||||||
|
// public string ChannelGreetMessageText { get; set; } = "Welcome to the %server% server, %user%!";
|
||||||
|
//
|
||||||
|
// public bool SendChannelByeMessage { get; set; }
|
||||||
|
// public string ChannelByeMessageText { get; set; } = "%user% has left!";
|
||||||
|
// public bool SendBoostMessage { get; set; }
|
||||||
|
// pulic int BoostMessageDeleteAfter { get; set; }
|
||||||
|
|
||||||
|
//self assignable roles
|
||||||
public bool ExclusiveSelfAssignedRoles { get; set; }
|
public bool ExclusiveSelfAssignedRoles { get; set; }
|
||||||
public bool AutoDeleteSelfAssignedRoleMessages { get; set; }
|
public bool AutoDeleteSelfAssignedRoleMessages { get; set; }
|
||||||
|
|
||||||
|
|
||||||
//stream notifications
|
//stream notifications
|
||||||
public HashSet<FollowedStream> FollowedStreams { get; set; } = new();
|
public HashSet<FollowedStream> FollowedStreams { get; set; } = new();
|
||||||
|
|
||||||
|
@ -36,38 +53,31 @@ public class GuildConfig : DbEntity
|
||||||
public HashSet<FilterChannelId> FilterInvitesChannelIds { get; set; } = new();
|
public HashSet<FilterChannelId> FilterInvitesChannelIds { get; set; } = new();
|
||||||
public HashSet<FilterLinksChannelId> FilterLinksChannelIds { get; set; } = new();
|
public HashSet<FilterLinksChannelId> FilterLinksChannelIds { get; set; } = new();
|
||||||
|
|
||||||
|
//public bool FilterLinks { get; set; }
|
||||||
|
//public HashSet<FilterLinksChannelId> FilterLinksChannels { get; set; } = new HashSet<FilterLinksChannelId>();
|
||||||
|
|
||||||
public bool FilterWords { get; set; }
|
public bool FilterWords { get; set; }
|
||||||
public HashSet<FilteredWord> FilteredWords { get; set; } = new();
|
public HashSet<FilteredWord> FilteredWords { get; set; } = new();
|
||||||
public HashSet<FilterWordsChannelId> FilterWordsChannelIds { get; set; } = new();
|
public HashSet<FilterWordsChannelId> FilterWordsChannelIds { get; set; } = new();
|
||||||
|
|
||||||
// mute
|
|
||||||
public HashSet<MutedUserId> MutedUsers { get; set; } = new();
|
public HashSet<MutedUserId> MutedUsers { get; set; } = new();
|
||||||
|
|
||||||
public string MuteRoleName { get; set; }
|
public string MuteRoleName { get; set; }
|
||||||
|
|
||||||
// chatterbot
|
|
||||||
public bool CleverbotEnabled { get; set; }
|
public bool CleverbotEnabled { get; set; }
|
||||||
|
|
||||||
// protection
|
|
||||||
public AntiRaidSetting AntiRaidSetting { get; set; }
|
public AntiRaidSetting AntiRaidSetting { get; set; }
|
||||||
public AntiSpamSetting AntiSpamSetting { get; set; }
|
public AntiSpamSetting AntiSpamSetting { get; set; }
|
||||||
public AntiAltSetting AntiAltSetting { get; set; }
|
public AntiAltSetting AntiAltSetting { get; set; }
|
||||||
|
|
||||||
// time
|
|
||||||
public string Locale { get; set; }
|
public string Locale { get; set; }
|
||||||
public string TimeZoneId { get; set; }
|
public string TimeZoneId { get; set; }
|
||||||
|
|
||||||
|
|
||||||
// timers
|
|
||||||
public HashSet<UnmuteTimer> UnmuteTimers { get; set; } = new();
|
public HashSet<UnmuteTimer> UnmuteTimers { get; set; } = new();
|
||||||
public HashSet<UnbanTimer> UnbanTimer { get; set; } = new();
|
public HashSet<UnbanTimer> UnbanTimer { get; set; } = new();
|
||||||
public HashSet<UnroleTimer> UnroleTimer { get; set; } = new();
|
public HashSet<UnroleTimer> UnroleTimer { get; set; } = new();
|
||||||
|
|
||||||
// vcrole
|
|
||||||
public HashSet<VcRoleInfo> VcRoleInfos { get; set; }
|
public HashSet<VcRoleInfo> VcRoleInfos { get; set; }
|
||||||
|
|
||||||
// aliases
|
|
||||||
public HashSet<CommandAlias> CommandAliases { get; set; } = new();
|
public HashSet<CommandAlias> CommandAliases { get; set; } = new();
|
||||||
|
public List<WarningPunishment> WarnPunishments { get; set; } = new();
|
||||||
public bool WarningsInitialized { get; set; }
|
public bool WarningsInitialized { get; set; }
|
||||||
public HashSet<SlowmodeIgnoredUser> SlowmodeIgnoredUsers { get; set; }
|
public HashSet<SlowmodeIgnoredUser> SlowmodeIgnoredUsers { get; set; }
|
||||||
public HashSet<SlowmodeIgnoredRole> SlowmodeIgnoredRoles { get; set; }
|
public HashSet<SlowmodeIgnoredRole> SlowmodeIgnoredRoles { get; set; }
|
||||||
|
@ -82,10 +92,15 @@ public class GuildConfig : DbEntity
|
||||||
public List<FeedSub> FeedSubs { get; set; } = new();
|
public List<FeedSub> FeedSubs { get; set; } = new();
|
||||||
public bool NotifyStreamOffline { get; set; }
|
public bool NotifyStreamOffline { get; set; }
|
||||||
public bool DeleteStreamOnlineMessage { get; set; }
|
public bool DeleteStreamOnlineMessage { get; set; }
|
||||||
|
public List<GroupName> SelfAssignableRoleGroupNames { get; set; }
|
||||||
public int WarnExpireHours { get; set; }
|
public int WarnExpireHours { get; set; }
|
||||||
public WarnExpireAction WarnExpireAction { get; set; } = WarnExpireAction.Clear;
|
public WarnExpireAction WarnExpireAction { get; set; } = WarnExpireAction.Clear;
|
||||||
|
|
||||||
public bool DisableGlobalExpressions { get; set; } = false;
|
public bool DisableGlobalExpressions { get; set; } = false;
|
||||||
|
|
||||||
|
#region Boost Message
|
||||||
|
|
||||||
public bool StickyRoles { get; set; }
|
public bool StickyRoles { get; set; }
|
||||||
|
|
||||||
|
#endregion
|
||||||
}
|
}
|
|
@ -1,19 +0,0 @@
|
||||||
using System.ComponentModel.DataAnnotations;
|
|
||||||
|
|
||||||
namespace EllieBot.Db.Models;
|
|
||||||
|
|
||||||
public class NCPixel
|
|
||||||
{
|
|
||||||
[Key]
|
|
||||||
public int Id { get; set; }
|
|
||||||
|
|
||||||
public required int Position { get; init; }
|
|
||||||
|
|
||||||
public required long Price { get; init; }
|
|
||||||
|
|
||||||
public required ulong OwnerId { get; init; }
|
|
||||||
public required uint Color { get; init; }
|
|
||||||
|
|
||||||
[MaxLength(256)]
|
|
||||||
public required string Text { get; init; }
|
|
||||||
}
|
|
|
@ -1,24 +0,0 @@
|
||||||
using System.ComponentModel.DataAnnotations;
|
|
||||||
|
|
||||||
namespace EllieBot.Db.Models;
|
|
||||||
|
|
||||||
public class Notify
|
|
||||||
{
|
|
||||||
[Key]
|
|
||||||
public int Id { get; set; }
|
|
||||||
|
|
||||||
public ulong GuildId { get; set; }
|
|
||||||
public ulong ChannelId { get; set; }
|
|
||||||
public NotifyType Type { get; set; }
|
|
||||||
|
|
||||||
[MaxLength(10_000)]
|
|
||||||
public string Message { get; set; } = string.Empty;
|
|
||||||
}
|
|
||||||
|
|
||||||
public enum NotifyType
|
|
||||||
{
|
|
||||||
LevelUp = 0,
|
|
||||||
Protection = 1, Prot = 1,
|
|
||||||
AddRoleReward = 2,
|
|
||||||
RemoveRoleReward = 3,
|
|
||||||
}
|
|
|
@ -1,18 +0,0 @@
|
||||||
using System.ComponentModel.DataAnnotations;
|
|
||||||
|
|
||||||
namespace EllieBot.Db.Models;
|
|
||||||
|
|
||||||
public sealed class SarGroup
|
|
||||||
{
|
|
||||||
[Key]
|
|
||||||
public int Id { get; set; }
|
|
||||||
|
|
||||||
public int GroupNumber { get; set; }
|
|
||||||
public ulong GuildId { get; set; }
|
|
||||||
public ulong? RoleReq { get; set; }
|
|
||||||
public ICollection<Sar> Roles { get; set; } = [];
|
|
||||||
public bool IsExclusive { get; set; }
|
|
||||||
|
|
||||||
[MaxLength(100)]
|
|
||||||
public string? Name { get; set; }
|
|
||||||
}
|
|
|
@ -1,27 +0,0 @@
|
||||||
using System.ComponentModel.DataAnnotations;
|
|
||||||
|
|
||||||
namespace EllieBot.Db.Models;
|
|
||||||
|
|
||||||
public sealed class ButtonRole
|
|
||||||
{
|
|
||||||
[Key]
|
|
||||||
public int Id { get; set; }
|
|
||||||
|
|
||||||
[MaxLength(200)]
|
|
||||||
public string ButtonId { get; set; } = string.Empty;
|
|
||||||
|
|
||||||
public ulong GuildId { get; set; }
|
|
||||||
public ulong ChannelId { get; set; }
|
|
||||||
public ulong MessageId { get; set; }
|
|
||||||
|
|
||||||
public int Position { get; set; }
|
|
||||||
public ulong RoleId { get; set; }
|
|
||||||
|
|
||||||
[MaxLength(100)]
|
|
||||||
public string Emote { get; set; } = string.Empty;
|
|
||||||
|
|
||||||
[MaxLength(50)]
|
|
||||||
public string Label { get; set; } = string.Empty;
|
|
||||||
|
|
||||||
public bool Exclusive { get; set; }
|
|
||||||
}
|
|
|
@ -3,7 +3,6 @@ namespace EllieBot.Db.Models;
|
||||||
|
|
||||||
public class WarningPunishment : DbEntity
|
public class WarningPunishment : DbEntity
|
||||||
{
|
{
|
||||||
public ulong GuildId { get; set; }
|
|
||||||
public int Count { get; set; }
|
public int Count { get; set; }
|
||||||
public PunishmentAction Punishment { get; set; }
|
public PunishmentAction Punishment { get; set; }
|
||||||
public int Time { get; set; }
|
public int Time { get; set; }
|
||||||
|
|
|
@ -1,24 +1,11 @@
|
||||||
using System.ComponentModel.DataAnnotations;
|
#nullable disable
|
||||||
|
|
||||||
namespace EllieBot.Db.Models;
|
namespace EllieBot.Db.Models;
|
||||||
|
|
||||||
public sealed class Sar
|
public class SelfAssignedRole : DbEntity
|
||||||
{
|
{
|
||||||
[Key]
|
|
||||||
public int Id { get; set; }
|
|
||||||
|
|
||||||
public ulong GuildId { get; set; }
|
public ulong GuildId { get; set; }
|
||||||
public ulong RoleId { get; set; }
|
public ulong RoleId { get; set; }
|
||||||
|
|
||||||
public int SarGroupId { get; set; }
|
public int Group { get; set; }
|
||||||
public int LevelReq { get; set; }
|
public int LevelRequirement { get; set; }
|
||||||
}
|
|
||||||
|
|
||||||
public sealed class SarAutoDelete
|
|
||||||
{
|
|
||||||
[Key]
|
|
||||||
public int Id { get; set; }
|
|
||||||
|
|
||||||
public ulong GuildId { get; set; }
|
|
||||||
public bool IsEnabled { get; set; } = false;
|
|
||||||
}
|
}
|
|
@ -1,12 +0,0 @@
|
||||||
namespace EllieBot.Db.Models;
|
|
||||||
|
|
||||||
public class TempRole
|
|
||||||
{
|
|
||||||
public int Id { get; set; }
|
|
||||||
public ulong GuildId { get; set; }
|
|
||||||
public bool Remove { get; set; }
|
|
||||||
public ulong RoleId { get; set; }
|
|
||||||
public ulong UserId { get; set; }
|
|
||||||
|
|
||||||
public DateTime ExpiresAt { get; set; }
|
|
||||||
}
|
|
|
@ -1,12 +1,8 @@
|
||||||
#nullable disable
|
#nullable disable
|
||||||
using System.ComponentModel.DataAnnotations;
|
|
||||||
|
|
||||||
namespace EllieBot.Db.Models;
|
namespace EllieBot.Db.Models;
|
||||||
|
|
||||||
public class PatronUser
|
public class PatronUser
|
||||||
{
|
{
|
||||||
// [Key]
|
|
||||||
// public int Id { get; set; }
|
|
||||||
public string UniquePlatformUserId { get; set; }
|
public string UniquePlatformUserId { get; set; }
|
||||||
public ulong UserId { get; set; }
|
public ulong UserId { get; set; }
|
||||||
public int AmountCents { get; set; }
|
public int AmountCents { get; set; }
|
||||||
|
|
|
@ -6,4 +6,6 @@ public class UserXpStats : DbEntity
|
||||||
public ulong UserId { get; set; }
|
public ulong UserId { get; set; }
|
||||||
public ulong GuildId { get; set; }
|
public ulong GuildId { get; set; }
|
||||||
public long Xp { get; set; }
|
public long Xp { get; set; }
|
||||||
|
public long AwardedXp { get; set; }
|
||||||
|
public XpNotificationLocation NotifyOnLevelUp { get; set; }
|
||||||
}
|
}
|
|
@ -4,7 +4,7 @@
|
||||||
<Nullable>enable</Nullable>
|
<Nullable>enable</Nullable>
|
||||||
<ImplicitUsings>true</ImplicitUsings>
|
<ImplicitUsings>true</ImplicitUsings>
|
||||||
<SatelliteResourceLanguages>en</SatelliteResourceLanguages>
|
<SatelliteResourceLanguages>en</SatelliteResourceLanguages>
|
||||||
<Version>5.3.3</Version>
|
<Version>5.1.11</Version>
|
||||||
|
|
||||||
<!-- Output/build -->
|
<!-- Output/build -->
|
||||||
<RunWorkingDirectory>$(MSBuildProjectDirectory)</RunWorkingDirectory>
|
<RunWorkingDirectory>$(MSBuildProjectDirectory)</RunWorkingDirectory>
|
||||||
|
@ -29,7 +29,7 @@
|
||||||
</PackageReference>
|
</PackageReference>
|
||||||
<PackageReference Include="CodeHollow.FeedReader" Version="1.2.6" />
|
<PackageReference Include="CodeHollow.FeedReader" Version="1.2.6" />
|
||||||
<PackageReference Include="CommandLineParser" Version="2.9.1" />
|
<PackageReference Include="CommandLineParser" Version="2.9.1" />
|
||||||
<PackageReference Include="Discord.Net" Version="3.16.0" />
|
<PackageReference Include="Discord.Net" Version="3.15.3" />
|
||||||
<PackageReference Include="CoreCLR-NCalc" Version="3.1.246" />
|
<PackageReference Include="CoreCLR-NCalc" Version="3.1.246" />
|
||||||
<PackageReference Include="Google.Apis.Urlshortener.v1" Version="1.41.1.138" />
|
<PackageReference Include="Google.Apis.Urlshortener.v1" Version="1.41.1.138" />
|
||||||
<PackageReference Include="Google.Apis.YouTube.v3" Version="1.68.0.3414" />
|
<PackageReference Include="Google.Apis.YouTube.v3" Version="1.68.0.3414" />
|
||||||
|
|
|
@ -5,54 +5,6 @@ namespace EllieBot.Migrations;
|
||||||
|
|
||||||
public static class MigrationQueries
|
public static class MigrationQueries
|
||||||
{
|
{
|
||||||
public static void MergeAwardedXp(MigrationBuilder migrationBuilder)
|
|
||||||
{
|
|
||||||
migrationBuilder.Sql("""
|
|
||||||
UPDATE UserXpStats
|
|
||||||
SET Xp = AwardedXp + Xp,
|
|
||||||
AwardedXp = 0
|
|
||||||
WHERE AwardedXp > 0;
|
|
||||||
""");
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void MigrateSar(MigrationBuilder migrationBuilder)
|
|
||||||
{
|
|
||||||
if (migrationBuilder.IsNpgsql())
|
|
||||||
return;
|
|
||||||
|
|
||||||
migrationBuilder.Sql("""
|
|
||||||
INSERT INTO GroupName (Number, GuildConfigId)
|
|
||||||
SELECT DISTINCT "Group", GC.Id
|
|
||||||
FROM SelfAssignableRoles as SAR
|
|
||||||
INNER JOIN GuildConfigs as GC
|
|
||||||
ON SAR.GuildId = GC.GuildId
|
|
||||||
WHERE SAR.GuildId not in (SELECT GuildConfigs.GuildId from GroupName LEFT JOIN GuildConfigs ON GroupName.GuildConfigId = GuildConfigs.Id);
|
|
||||||
|
|
||||||
INSERT INTO SarGroup (Id, GroupNumber, Name, IsExclusive, GuildId)
|
|
||||||
SELECT GN.Id, GN.Number, GN.Name, GC.ExclusiveSelfAssignedRoles, GC.GuildId
|
|
||||||
FROM GroupName as GN
|
|
||||||
INNER JOIN GuildConfigs as GC ON GN.GuildConfigId = GC.Id;
|
|
||||||
|
|
||||||
INSERT INTO Sar (GuildId, RoleId, SarGroupId, LevelReq)
|
|
||||||
SELECT SAR.GuildId, SAR.RoleId, (SELECT Id FROM SarGroup WHERE SG.Number = SarGroup.GroupNumber AND SG.GuildId = SarGroup.GuildId), MIN(SAR.LevelRequirement)
|
|
||||||
FROM SelfAssignableRoles as SAR
|
|
||||||
INNER JOIN (SELECT GuildId, gn.Number FROM GroupName as gn
|
|
||||||
INNER JOIN GuildConfigs as gc ON gn.GuildConfigId =gc.Id
|
|
||||||
) as SG
|
|
||||||
ON SG.GuildId = SAR.GuildId
|
|
||||||
WHERE SG.Number IN (SELECT GroupNumber FROM SarGroup WHERE Sar.GuildId = SarGroup.GuildId)
|
|
||||||
GROUP BY SAR.GuildId, SAR.RoleId;
|
|
||||||
|
|
||||||
INSERT INTO SarAutoDelete (GuildId, IsEnabled)
|
|
||||||
SELECT GuildId, AutoDeleteSelfAssignedRoleMessages FROM GuildConfigs WHERE AutoDeleteSelfAssignedRoleMessages = TRUE;
|
|
||||||
""");
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void UpdateUsernames(MigrationBuilder migrationBuilder)
|
|
||||||
{
|
|
||||||
migrationBuilder.Sql("UPDATE DiscordUser SET Username = '??' || Username WHERE Discriminator = '????';");
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void MigrateRero(MigrationBuilder migrationBuilder)
|
public static void MigrateRero(MigrationBuilder migrationBuilder)
|
||||||
{
|
{
|
||||||
if (migrationBuilder.IsSqlite())
|
if (migrationBuilder.IsSqlite())
|
||||||
|
@ -86,7 +38,6 @@ left join guildconfigs on reactionrolemessage.guildconfigid = guildconfigs.id;")
|
||||||
DELETE FROM "DelMsgOnCmdChannel" WHERE "GuildConfigId" is NULL;
|
DELETE FROM "DelMsgOnCmdChannel" WHERE "GuildConfigId" is NULL;
|
||||||
DELETE FROM "WarningPunishment" WHERE "GuildConfigId" NOT IN (SELECT "Id" from "GuildConfigs");
|
DELETE FROM "WarningPunishment" WHERE "GuildConfigId" NOT IN (SELECT "Id" from "GuildConfigs");
|
||||||
DELETE FROM "StreamRoleBlacklistedUser" WHERE "StreamRoleSettingsId" is NULL;
|
DELETE FROM "StreamRoleBlacklistedUser" WHERE "StreamRoleSettingsId" is NULL;
|
||||||
DELETE FROM "Permissions" WHERE "GuildConfigId" NOT IN (SELECT "Id" from "GuildConfigs");
|
|
||||||
""");
|
""");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -114,20 +65,4 @@ left join guildconfigs on reactionrolemessage.guildconfigid = guildconfigs.id;")
|
||||||
WHERE SendBoostMessage = TRUE;
|
WHERE SendBoostMessage = TRUE;
|
||||||
""");
|
""");
|
||||||
}
|
}
|
||||||
|
|
||||||
public static void AddGuildIdsToWarningPunishment(MigrationBuilder builder)
|
|
||||||
{
|
|
||||||
builder.Sql("""
|
|
||||||
DELETE FROM WarningPunishment WHERE GuildConfigId IS NULL OR GuildConfigId NOT IN (SELECT Id FROM GuildConfigs);
|
|
||||||
UPDATE WarningPunishment
|
|
||||||
SET GuildId = (SELECT GuildId FROM GuildConfigs WHERE Id = GuildConfigId);
|
|
||||||
|
|
||||||
DELETE FROM WarningPunishment as wp
|
|
||||||
WHERE (wp.Count, wp.GuildConfigId) in (
|
|
||||||
SELECT wp2.Count, wp2.GuildConfigId FROM WarningPunishment as wp2
|
|
||||||
GROUP BY wp2.Count, wp2.GuildConfigId
|
|
||||||
HAVING COUNT(id) > 1
|
|
||||||
);
|
|
||||||
""");
|
|
||||||
}
|
|
||||||
}
|
}
|
File diff suppressed because it is too large
Load diff
|
@ -1,71 +0,0 @@
|
||||||
using Microsoft.EntityFrameworkCore.Migrations;
|
|
||||||
|
|
||||||
#nullable disable
|
|
||||||
|
|
||||||
namespace EllieBot.Migrations.PostgreSql
|
|
||||||
{
|
|
||||||
/// <inheritdoc />
|
|
||||||
public partial class warnsplit : Migration
|
|
||||||
{
|
|
||||||
/// <inheritdoc />
|
|
||||||
protected override void Up(MigrationBuilder migrationBuilder)
|
|
||||||
{
|
|
||||||
migrationBuilder.AddColumn<decimal>(
|
|
||||||
name: "guildid",
|
|
||||||
table: "warningpunishment",
|
|
||||||
type: "numeric(20,0)",
|
|
||||||
nullable: false,
|
|
||||||
defaultValue: 0m);
|
|
||||||
|
|
||||||
MigrationQueries.AddGuildIdsToWarningPunishment(migrationBuilder);
|
|
||||||
|
|
||||||
migrationBuilder.DropForeignKey(
|
|
||||||
name: "fk_warningpunishment_guildconfigs_guildconfigid",
|
|
||||||
table: "warningpunishment");
|
|
||||||
|
|
||||||
migrationBuilder.DropIndex(
|
|
||||||
name: "ix_warningpunishment_guildconfigid",
|
|
||||||
table: "warningpunishment");
|
|
||||||
|
|
||||||
migrationBuilder.DropColumn(
|
|
||||||
name: "guildconfigid",
|
|
||||||
table: "warningpunishment");
|
|
||||||
|
|
||||||
migrationBuilder.AddUniqueConstraint(
|
|
||||||
name: "ak_warningpunishment_guildid_count",
|
|
||||||
table: "warningpunishment",
|
|
||||||
columns: new[] { "guildid", "count" });
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <inheritdoc />
|
|
||||||
protected override void Down(MigrationBuilder migrationBuilder)
|
|
||||||
{
|
|
||||||
migrationBuilder.DropUniqueConstraint(
|
|
||||||
name: "ak_warningpunishment_guildid_count",
|
|
||||||
table: "warningpunishment");
|
|
||||||
|
|
||||||
migrationBuilder.DropColumn(
|
|
||||||
name: "guildid",
|
|
||||||
table: "warningpunishment");
|
|
||||||
|
|
||||||
migrationBuilder.AddColumn<int>(
|
|
||||||
name: "guildconfigid",
|
|
||||||
table: "warningpunishment",
|
|
||||||
type: "integer",
|
|
||||||
nullable: true);
|
|
||||||
|
|
||||||
migrationBuilder.CreateIndex(
|
|
||||||
name: "ix_warningpunishment_guildconfigid",
|
|
||||||
table: "warningpunishment",
|
|
||||||
column: "guildconfigid");
|
|
||||||
|
|
||||||
migrationBuilder.AddForeignKey(
|
|
||||||
name: "fk_warningpunishment_guildconfigs_guildconfigid",
|
|
||||||
table: "warningpunishment",
|
|
||||||
column: "guildconfigid",
|
|
||||||
principalTable: "guildconfigs",
|
|
||||||
principalColumn: "id",
|
|
||||||
onDelete: ReferentialAction.Cascade);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
File diff suppressed because it is too large
Load diff
|
@ -1,54 +0,0 @@
|
||||||
using Microsoft.EntityFrameworkCore.Migrations;
|
|
||||||
using Npgsql.EntityFrameworkCore.PostgreSQL.Metadata;
|
|
||||||
|
|
||||||
#nullable disable
|
|
||||||
|
|
||||||
namespace EllieBot.Migrations.PostgreSql
|
|
||||||
{
|
|
||||||
/// <inheritdoc />
|
|
||||||
public partial class ncanvas : Migration
|
|
||||||
{
|
|
||||||
/// <inheritdoc />
|
|
||||||
protected override void Up(MigrationBuilder migrationBuilder)
|
|
||||||
{
|
|
||||||
migrationBuilder.CreateTable(
|
|
||||||
name: "ncpixel",
|
|
||||||
columns: table => new
|
|
||||||
{
|
|
||||||
id = table.Column<int>(type: "integer", nullable: false)
|
|
||||||
.Annotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.IdentityByDefaultColumn),
|
|
||||||
position = table.Column<int>(type: "integer", nullable: false),
|
|
||||||
price = table.Column<long>(type: "bigint", nullable: false),
|
|
||||||
ownerid = table.Column<decimal>(type: "numeric(20,0)", nullable: false),
|
|
||||||
color = table.Column<long>(type: "bigint", nullable: false),
|
|
||||||
text = table.Column<string>(type: "character varying(256)", maxLength: 256, nullable: false)
|
|
||||||
},
|
|
||||||
constraints: table =>
|
|
||||||
{
|
|
||||||
table.PrimaryKey("pk_ncpixel", x => x.id);
|
|
||||||
table.UniqueConstraint("ak_ncpixel_position", x => x.position);
|
|
||||||
});
|
|
||||||
|
|
||||||
migrationBuilder.CreateIndex(
|
|
||||||
name: "ix_discorduser_username",
|
|
||||||
table: "discorduser",
|
|
||||||
column: "username");
|
|
||||||
|
|
||||||
migrationBuilder.CreateIndex(
|
|
||||||
name: "ix_ncpixel_ownerid",
|
|
||||||
table: "ncpixel",
|
|
||||||
column: "ownerid");
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <inheritdoc />
|
|
||||||
protected override void Down(MigrationBuilder migrationBuilder)
|
|
||||||
{
|
|
||||||
migrationBuilder.DropTable(
|
|
||||||
name: "ncpixel");
|
|
||||||
|
|
||||||
migrationBuilder.DropIndex(
|
|
||||||
name: "ix_discorduser_username",
|
|
||||||
table: "discorduser");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
File diff suppressed because it is too large
Load diff
|
@ -1,54 +0,0 @@
|
||||||
using System;
|
|
||||||
using Microsoft.EntityFrameworkCore.Migrations;
|
|
||||||
using Npgsql.EntityFrameworkCore.PostgreSQL.Metadata;
|
|
||||||
|
|
||||||
#nullable disable
|
|
||||||
|
|
||||||
namespace EllieBot.Migrations.PostgreSql
|
|
||||||
{
|
|
||||||
/// <inheritdoc />
|
|
||||||
public partial class nodiscrimandflagtranslate : Migration
|
|
||||||
{
|
|
||||||
/// <inheritdoc />
|
|
||||||
protected override void Up(MigrationBuilder migrationBuilder)
|
|
||||||
{
|
|
||||||
migrationBuilder.DropColumn(
|
|
||||||
name: "discriminator",
|
|
||||||
table: "discorduser");
|
|
||||||
|
|
||||||
migrationBuilder.CreateTable(
|
|
||||||
name: "flagtranslatechannel",
|
|
||||||
columns: table => new
|
|
||||||
{
|
|
||||||
id = table.Column<int>(type: "integer", nullable: false)
|
|
||||||
.Annotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.IdentityByDefaultColumn),
|
|
||||||
guildid = table.Column<decimal>(type: "numeric(20,0)", nullable: false),
|
|
||||||
channelid = table.Column<decimal>(type: "numeric(20,0)", nullable: false),
|
|
||||||
dateadded = table.Column<DateTime>(type: "timestamp without time zone", nullable: true)
|
|
||||||
},
|
|
||||||
constraints: table =>
|
|
||||||
{
|
|
||||||
table.PrimaryKey("pk_flagtranslatechannel", x => x.id);
|
|
||||||
});
|
|
||||||
|
|
||||||
migrationBuilder.CreateIndex(
|
|
||||||
name: "ix_flagtranslatechannel_guildid_channelid",
|
|
||||||
table: "flagtranslatechannel",
|
|
||||||
columns: new[] { "guildid", "channelid" },
|
|
||||||
unique: true);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <inheritdoc />
|
|
||||||
protected override void Down(MigrationBuilder migrationBuilder)
|
|
||||||
{
|
|
||||||
migrationBuilder.DropTable(
|
|
||||||
name: "flagtranslatechannel");
|
|
||||||
|
|
||||||
migrationBuilder.AddColumn<string>(
|
|
||||||
name: "discriminator",
|
|
||||||
table: "discorduser",
|
|
||||||
type: "text",
|
|
||||||
nullable: true);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
File diff suppressed because it is too large
Load diff
|
@ -1,48 +0,0 @@
|
||||||
using Microsoft.EntityFrameworkCore.Migrations;
|
|
||||||
using Npgsql.EntityFrameworkCore.PostgreSQL.Metadata;
|
|
||||||
|
|
||||||
#nullable disable
|
|
||||||
|
|
||||||
namespace EllieBot.Migrations.PostgreSql
|
|
||||||
{
|
|
||||||
/// <inheritdoc />
|
|
||||||
public partial class betstats : Migration
|
|
||||||
{
|
|
||||||
/// <inheritdoc />
|
|
||||||
protected override void Up(MigrationBuilder migrationBuilder)
|
|
||||||
{
|
|
||||||
migrationBuilder.CreateTable(
|
|
||||||
name: "userbetstats",
|
|
||||||
columns: table => new
|
|
||||||
{
|
|
||||||
id = table.Column<int>(type: "integer", nullable: false)
|
|
||||||
.Annotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.IdentityByDefaultColumn),
|
|
||||||
userid = table.Column<decimal>(type: "numeric(20,0)", nullable: false),
|
|
||||||
game = table.Column<int>(type: "integer", nullable: false),
|
|
||||||
wincount = table.Column<long>(type: "bigint", nullable: false),
|
|
||||||
losecount = table.Column<long>(type: "bigint", nullable: false),
|
|
||||||
totalbet = table.Column<decimal>(type: "numeric", nullable: false),
|
|
||||||
paidout = table.Column<decimal>(type: "numeric", nullable: false),
|
|
||||||
maxwin = table.Column<long>(type: "bigint", nullable: false),
|
|
||||||
maxbet = table.Column<long>(type: "bigint", nullable: false)
|
|
||||||
},
|
|
||||||
constraints: table =>
|
|
||||||
{
|
|
||||||
table.PrimaryKey("pk_userbetstats", x => x.id);
|
|
||||||
});
|
|
||||||
|
|
||||||
migrationBuilder.CreateIndex(
|
|
||||||
name: "ix_userbetstats_userid_game",
|
|
||||||
table: "userbetstats",
|
|
||||||
columns: new[] { "userid", "game" },
|
|
||||||
unique: true);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <inheritdoc />
|
|
||||||
protected override void Down(MigrationBuilder migrationBuilder)
|
|
||||||
{
|
|
||||||
migrationBuilder.DropTable(
|
|
||||||
name: "userbetstats");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
File diff suppressed because it is too large
Load diff
|
@ -1,33 +0,0 @@
|
||||||
using Microsoft.EntityFrameworkCore.Migrations;
|
|
||||||
|
|
||||||
#nullable disable
|
|
||||||
|
|
||||||
namespace EllieBot.Migrations.PostgreSql
|
|
||||||
{
|
|
||||||
/// <inheritdoc />
|
|
||||||
public partial class rakeback : Migration
|
|
||||||
{
|
|
||||||
/// <inheritdoc />
|
|
||||||
protected override void Up(MigrationBuilder migrationBuilder)
|
|
||||||
{
|
|
||||||
migrationBuilder.CreateTable(
|
|
||||||
name: "rakeback",
|
|
||||||
columns: table => new
|
|
||||||
{
|
|
||||||
userid = table.Column<decimal>(type: "numeric(20,0)", nullable: false),
|
|
||||||
amount = table.Column<decimal>(type: "numeric", nullable: false)
|
|
||||||
},
|
|
||||||
constraints: table =>
|
|
||||||
{
|
|
||||||
table.PrimaryKey("pk_rakeback", x => x.userid);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <inheritdoc />
|
|
||||||
protected override void Down(MigrationBuilder migrationBuilder)
|
|
||||||
{
|
|
||||||
migrationBuilder.DropTable(
|
|
||||||
name: "rakeback");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
File diff suppressed because it is too large
Load diff
|
@ -1,74 +0,0 @@
|
||||||
using Microsoft.EntityFrameworkCore.Migrations;
|
|
||||||
using Npgsql.EntityFrameworkCore.PostgreSQL.Metadata;
|
|
||||||
|
|
||||||
#nullable disable
|
|
||||||
|
|
||||||
namespace EllieBot.Migrations.PostgreSql
|
|
||||||
{
|
|
||||||
/// <inheritdoc />
|
|
||||||
public partial class btnroles_guildcolors : Migration
|
|
||||||
{
|
|
||||||
/// <inheritdoc />
|
|
||||||
protected override void Up(MigrationBuilder migrationBuilder)
|
|
||||||
{
|
|
||||||
migrationBuilder.CreateTable(
|
|
||||||
name: "buttonrole",
|
|
||||||
columns: table => new
|
|
||||||
{
|
|
||||||
id = table.Column<int>(type: "integer", nullable: false)
|
|
||||||
.Annotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.IdentityByDefaultColumn),
|
|
||||||
buttonid = table.Column<string>(type: "character varying(200)", maxLength: 200, nullable: false),
|
|
||||||
guildid = table.Column<decimal>(type: "numeric(20,0)", nullable: false),
|
|
||||||
channelid = table.Column<decimal>(type: "numeric(20,0)", nullable: false),
|
|
||||||
messageid = table.Column<decimal>(type: "numeric(20,0)", nullable: false),
|
|
||||||
position = table.Column<int>(type: "integer", nullable: false),
|
|
||||||
roleid = table.Column<decimal>(type: "numeric(20,0)", nullable: false),
|
|
||||||
emote = table.Column<string>(type: "character varying(100)", maxLength: 100, nullable: false),
|
|
||||||
label = table.Column<string>(type: "character varying(50)", maxLength: 50, nullable: false),
|
|
||||||
exclusive = table.Column<bool>(type: "boolean", nullable: false)
|
|
||||||
},
|
|
||||||
constraints: table =>
|
|
||||||
{
|
|
||||||
table.PrimaryKey("pk_buttonrole", x => x.id);
|
|
||||||
table.UniqueConstraint("ak_buttonrole_roleid_messageid", x => new { x.roleid, x.messageid });
|
|
||||||
});
|
|
||||||
|
|
||||||
migrationBuilder.CreateTable(
|
|
||||||
name: "guildcolors",
|
|
||||||
columns: table => new
|
|
||||||
{
|
|
||||||
id = table.Column<int>(type: "integer", nullable: false)
|
|
||||||
.Annotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.IdentityByDefaultColumn),
|
|
||||||
guildid = table.Column<decimal>(type: "numeric(20,0)", nullable: false),
|
|
||||||
okcolor = table.Column<string>(type: "character varying(9)", maxLength: 9, nullable: true),
|
|
||||||
errorcolor = table.Column<string>(type: "character varying(9)", maxLength: 9, nullable: true),
|
|
||||||
pendingcolor = table.Column<string>(type: "character varying(9)", maxLength: 9, nullable: true)
|
|
||||||
},
|
|
||||||
constraints: table =>
|
|
||||||
{
|
|
||||||
table.PrimaryKey("pk_guildcolors", x => x.id);
|
|
||||||
});
|
|
||||||
|
|
||||||
migrationBuilder.CreateIndex(
|
|
||||||
name: "ix_buttonrole_guildid",
|
|
||||||
table: "buttonrole",
|
|
||||||
column: "guildid");
|
|
||||||
|
|
||||||
migrationBuilder.CreateIndex(
|
|
||||||
name: "ix_guildcolors_guildid",
|
|
||||||
table: "guildcolors",
|
|
||||||
column: "guildid",
|
|
||||||
unique: true);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <inheritdoc />
|
|
||||||
protected override void Down(MigrationBuilder migrationBuilder)
|
|
||||||
{
|
|
||||||
migrationBuilder.DropTable(
|
|
||||||
name: "buttonrole");
|
|
||||||
|
|
||||||
migrationBuilder.DropTable(
|
|
||||||
name: "guildcolors");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
File diff suppressed because it is too large
Load diff
|
@ -1,153 +0,0 @@
|
||||||
using System;
|
|
||||||
using Microsoft.EntityFrameworkCore.Migrations;
|
|
||||||
using Npgsql.EntityFrameworkCore.PostgreSQL.Metadata;
|
|
||||||
|
|
||||||
#nullable disable
|
|
||||||
|
|
||||||
namespace EllieBot.Migrations.PostgreSql
|
|
||||||
{
|
|
||||||
/// <inheritdoc />
|
|
||||||
public partial class sarrework : Migration
|
|
||||||
{
|
|
||||||
/// <inheritdoc />
|
|
||||||
protected override void Up(MigrationBuilder migrationBuilder)
|
|
||||||
{
|
|
||||||
migrationBuilder.DropTable(
|
|
||||||
name: "groupname");
|
|
||||||
|
|
||||||
migrationBuilder.DropTable(
|
|
||||||
name: "selfassignableroles");
|
|
||||||
|
|
||||||
migrationBuilder.CreateTable(
|
|
||||||
name: "sarautodelete",
|
|
||||||
columns: table => new
|
|
||||||
{
|
|
||||||
id = table.Column<int>(type: "integer", nullable: false)
|
|
||||||
.Annotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.IdentityByDefaultColumn),
|
|
||||||
guildid = table.Column<decimal>(type: "numeric(20,0)", nullable: false),
|
|
||||||
isenabled = table.Column<bool>(type: "boolean", nullable: false)
|
|
||||||
},
|
|
||||||
constraints: table =>
|
|
||||||
{
|
|
||||||
table.PrimaryKey("pk_sarautodelete", x => x.id);
|
|
||||||
});
|
|
||||||
|
|
||||||
migrationBuilder.CreateTable(
|
|
||||||
name: "sargroup",
|
|
||||||
columns: table => new
|
|
||||||
{
|
|
||||||
id = table.Column<int>(type: "integer", nullable: false)
|
|
||||||
.Annotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.IdentityByDefaultColumn),
|
|
||||||
groupnumber = table.Column<int>(type: "integer", nullable: false),
|
|
||||||
guildid = table.Column<decimal>(type: "numeric(20,0)", nullable: false),
|
|
||||||
rolereq = table.Column<decimal>(type: "numeric(20,0)", nullable: true),
|
|
||||||
isexclusive = table.Column<bool>(type: "boolean", nullable: false),
|
|
||||||
name = table.Column<string>(type: "character varying(100)", maxLength: 100, nullable: true),
|
|
||||||
dateadded = table.Column<DateTime>(type: "timestamp without time zone", nullable: true)
|
|
||||||
},
|
|
||||||
constraints: table =>
|
|
||||||
{
|
|
||||||
table.PrimaryKey("pk_sargroup", x => x.id);
|
|
||||||
table.UniqueConstraint("ak_sargroup_guildid_groupnumber", x => new { x.guildid, x.groupnumber });
|
|
||||||
});
|
|
||||||
|
|
||||||
migrationBuilder.CreateTable(
|
|
||||||
name: "sar",
|
|
||||||
columns: table => new
|
|
||||||
{
|
|
||||||
id = table.Column<int>(type: "integer", nullable: false)
|
|
||||||
.Annotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.IdentityByDefaultColumn),
|
|
||||||
guildid = table.Column<decimal>(type: "numeric(20,0)", nullable: false),
|
|
||||||
roleid = table.Column<decimal>(type: "numeric(20,0)", nullable: false),
|
|
||||||
sargroupid = table.Column<int>(type: "integer", nullable: false),
|
|
||||||
levelreq = table.Column<int>(type: "integer", nullable: false)
|
|
||||||
},
|
|
||||||
constraints: table =>
|
|
||||||
{
|
|
||||||
table.PrimaryKey("pk_sar", x => x.id);
|
|
||||||
table.UniqueConstraint("ak_sar_guildid_roleid", x => new { x.guildid, x.roleid });
|
|
||||||
table.ForeignKey(
|
|
||||||
name: "fk_sar_sargroup_sargroupid",
|
|
||||||
column: x => x.sargroupid,
|
|
||||||
principalTable: "sargroup",
|
|
||||||
principalColumn: "id",
|
|
||||||
onDelete: ReferentialAction.Cascade);
|
|
||||||
});
|
|
||||||
|
|
||||||
migrationBuilder.CreateIndex(
|
|
||||||
name: "ix_sar_sargroupid",
|
|
||||||
table: "sar",
|
|
||||||
column: "sargroupid");
|
|
||||||
|
|
||||||
migrationBuilder.CreateIndex(
|
|
||||||
name: "ix_sarautodelete_guildid",
|
|
||||||
table: "sarautodelete",
|
|
||||||
column: "guildid",
|
|
||||||
unique: true);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <inheritdoc />
|
|
||||||
protected override void Down(MigrationBuilder migrationBuilder)
|
|
||||||
{
|
|
||||||
migrationBuilder.DropTable(
|
|
||||||
name: "sar");
|
|
||||||
|
|
||||||
migrationBuilder.DropTable(
|
|
||||||
name: "sarautodelete");
|
|
||||||
|
|
||||||
migrationBuilder.DropTable(
|
|
||||||
name: "sargroup");
|
|
||||||
|
|
||||||
migrationBuilder.CreateTable(
|
|
||||||
name: "groupname",
|
|
||||||
columns: table => new
|
|
||||||
{
|
|
||||||
id = table.Column<int>(type: "integer", nullable: false)
|
|
||||||
.Annotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.IdentityByDefaultColumn),
|
|
||||||
guildconfigid = table.Column<int>(type: "integer", nullable: false),
|
|
||||||
dateadded = table.Column<DateTime>(type: "timestamp without time zone", nullable: true),
|
|
||||||
name = table.Column<string>(type: "text", nullable: true),
|
|
||||||
number = table.Column<int>(type: "integer", nullable: false)
|
|
||||||
},
|
|
||||||
constraints: table =>
|
|
||||||
{
|
|
||||||
table.PrimaryKey("pk_groupname", x => x.id);
|
|
||||||
table.ForeignKey(
|
|
||||||
name: "fk_groupname_guildconfigs_guildconfigid",
|
|
||||||
column: x => x.guildconfigid,
|
|
||||||
principalTable: "guildconfigs",
|
|
||||||
principalColumn: "id",
|
|
||||||
onDelete: ReferentialAction.Cascade);
|
|
||||||
});
|
|
||||||
|
|
||||||
migrationBuilder.CreateTable(
|
|
||||||
name: "selfassignableroles",
|
|
||||||
columns: table => new
|
|
||||||
{
|
|
||||||
id = table.Column<int>(type: "integer", nullable: false)
|
|
||||||
.Annotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.IdentityByDefaultColumn),
|
|
||||||
dateadded = table.Column<DateTime>(type: "timestamp without time zone", nullable: true),
|
|
||||||
group = table.Column<int>(type: "integer", nullable: false, defaultValue: 0),
|
|
||||||
guildid = table.Column<decimal>(type: "numeric(20,0)", nullable: false),
|
|
||||||
levelrequirement = table.Column<int>(type: "integer", nullable: false),
|
|
||||||
roleid = table.Column<decimal>(type: "numeric(20,0)", nullable: false)
|
|
||||||
},
|
|
||||||
constraints: table =>
|
|
||||||
{
|
|
||||||
table.PrimaryKey("pk_selfassignableroles", x => x.id);
|
|
||||||
});
|
|
||||||
|
|
||||||
migrationBuilder.CreateIndex(
|
|
||||||
name: "ix_groupname_guildconfigid_number",
|
|
||||||
table: "groupname",
|
|
||||||
columns: new[] { "guildconfigid", "number" },
|
|
||||||
unique: true);
|
|
||||||
|
|
||||||
migrationBuilder.CreateIndex(
|
|
||||||
name: "ix_selfassignableroles_guildid_roleid",
|
|
||||||
table: "selfassignableroles",
|
|
||||||
columns: new[] { "guildid", "roleid" },
|
|
||||||
unique: true);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
File diff suppressed because it is too large
Load diff
|
@ -1,109 +0,0 @@
|
||||||
using System;
|
|
||||||
using EllieBot.Migrations;
|
|
||||||
using Microsoft.EntityFrameworkCore.Migrations;
|
|
||||||
using Npgsql.EntityFrameworkCore.PostgreSQL.Metadata;
|
|
||||||
|
|
||||||
#nullable disable
|
|
||||||
|
|
||||||
namespace EllieBot.Migrations.PostgreSql
|
|
||||||
{
|
|
||||||
/// <inheritdoc />
|
|
||||||
public partial class awardedxptemprolenotify : Migration
|
|
||||||
{
|
|
||||||
/// <inheritdoc />
|
|
||||||
protected override void Up(MigrationBuilder migrationBuilder)
|
|
||||||
{
|
|
||||||
migrationBuilder.DropIndex(
|
|
||||||
name: "ix_userxpstats_awardedxp",
|
|
||||||
table: "userxpstats");
|
|
||||||
|
|
||||||
MigrationQueries.MergeAwardedXp(migrationBuilder);
|
|
||||||
|
|
||||||
migrationBuilder.DropColumn(
|
|
||||||
name: "awardedxp",
|
|
||||||
table: "userxpstats");
|
|
||||||
|
|
||||||
migrationBuilder.DropColumn(
|
|
||||||
name: "notifyonlevelup",
|
|
||||||
table: "userxpstats");
|
|
||||||
|
|
||||||
migrationBuilder.CreateTable(
|
|
||||||
name: "notify",
|
|
||||||
columns: table => new
|
|
||||||
{
|
|
||||||
id = table.Column<int>(type: "integer", nullable: false)
|
|
||||||
.Annotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.IdentityByDefaultColumn),
|
|
||||||
guildid = table.Column<decimal>(type: "numeric(20,0)", nullable: false),
|
|
||||||
channelid = table.Column<decimal>(type: "numeric(20,0)", nullable: false),
|
|
||||||
type = table.Column<int>(type: "integer", nullable: false),
|
|
||||||
message = table.Column<string>(type: "character varying(10000)", maxLength: 10000, nullable: false)
|
|
||||||
},
|
|
||||||
constraints: table =>
|
|
||||||
{
|
|
||||||
table.PrimaryKey("pk_notify", x => x.id);
|
|
||||||
table.UniqueConstraint("ak_notify_guildid_type", x => new { x.guildid, x.type });
|
|
||||||
});
|
|
||||||
|
|
||||||
migrationBuilder.CreateTable(
|
|
||||||
name: "temprole",
|
|
||||||
columns: table => new
|
|
||||||
{
|
|
||||||
id = table.Column<int>(type: "integer", nullable: false)
|
|
||||||
.Annotation("Npgsql:ValueGenerationStrategy", NpgsqlValueGenerationStrategy.IdentityByDefaultColumn),
|
|
||||||
guildid = table.Column<decimal>(type: "numeric(20,0)", nullable: false),
|
|
||||||
remove = table.Column<bool>(type: "boolean", nullable: false),
|
|
||||||
roleid = table.Column<decimal>(type: "numeric(20,0)", nullable: false),
|
|
||||||
userid = table.Column<decimal>(type: "numeric(20,0)", nullable: false),
|
|
||||||
expiresat = table.Column<DateTime>(type: "timestamp without time zone", nullable: false)
|
|
||||||
},
|
|
||||||
constraints: table =>
|
|
||||||
{
|
|
||||||
table.PrimaryKey("pk_temprole", x => x.id);
|
|
||||||
table.UniqueConstraint("ak_temprole_guildid_userid_roleid", x => new { x.guildid, x.userid, x.roleid });
|
|
||||||
});
|
|
||||||
|
|
||||||
migrationBuilder.CreateIndex(
|
|
||||||
name: "ix_userbetstats_maxwin",
|
|
||||||
table: "userbetstats",
|
|
||||||
column: "maxwin");
|
|
||||||
|
|
||||||
migrationBuilder.CreateIndex(
|
|
||||||
name: "ix_temprole_expiresat",
|
|
||||||
table: "temprole",
|
|
||||||
column: "expiresat");
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <inheritdoc />
|
|
||||||
protected override void Down(MigrationBuilder migrationBuilder)
|
|
||||||
{
|
|
||||||
migrationBuilder.DropTable(
|
|
||||||
name: "notify");
|
|
||||||
|
|
||||||
migrationBuilder.DropTable(
|
|
||||||
name: "temprole");
|
|
||||||
|
|
||||||
migrationBuilder.DropIndex(
|
|
||||||
name: "ix_userbetstats_maxwin",
|
|
||||||
table: "userbetstats");
|
|
||||||
|
|
||||||
migrationBuilder.AddColumn<long>(
|
|
||||||
name: "awardedxp",
|
|
||||||
table: "userxpstats",
|
|
||||||
type: "bigint",
|
|
||||||
nullable: false,
|
|
||||||
defaultValue: 0L);
|
|
||||||
|
|
||||||
migrationBuilder.AddColumn<int>(
|
|
||||||
name: "notifyonlevelup",
|
|
||||||
table: "userxpstats",
|
|
||||||
type: "integer",
|
|
||||||
nullable: false,
|
|
||||||
defaultValue: 0);
|
|
||||||
|
|
||||||
migrationBuilder.CreateIndex(
|
|
||||||
name: "ix_userxpstats_awardedxp",
|
|
||||||
table: "userxpstats",
|
|
||||||
column: "awardedxp");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
File diff suppressed because it is too large
Load diff
File diff suppressed because it is too large
Load diff
|
@ -1,72 +0,0 @@
|
||||||
using Microsoft.EntityFrameworkCore.Metadata.Internal;
|
|
||||||
using Microsoft.EntityFrameworkCore.Migrations;
|
|
||||||
|
|
||||||
#nullable disable
|
|
||||||
|
|
||||||
namespace EllieBot.Migrations
|
|
||||||
{
|
|
||||||
/// <inheritdoc />
|
|
||||||
public partial class warnsplit : Migration
|
|
||||||
{
|
|
||||||
/// <inheritdoc />
|
|
||||||
protected override void Up(MigrationBuilder migrationBuilder)
|
|
||||||
{
|
|
||||||
migrationBuilder.AddColumn<ulong>(
|
|
||||||
name: "GuildId",
|
|
||||||
table: "WarningPunishment",
|
|
||||||
type: "INTEGER",
|
|
||||||
nullable: false,
|
|
||||||
defaultValue: 0ul);
|
|
||||||
|
|
||||||
MigrationQueries.AddGuildIdsToWarningPunishment(migrationBuilder);
|
|
||||||
|
|
||||||
migrationBuilder.DropForeignKey(
|
|
||||||
name: "FK_WarningPunishment_GuildConfigs_GuildConfigId",
|
|
||||||
table: "WarningPunishment");
|
|
||||||
|
|
||||||
migrationBuilder.DropIndex(
|
|
||||||
name: "IX_WarningPunishment_GuildConfigId",
|
|
||||||
table: "WarningPunishment");
|
|
||||||
|
|
||||||
migrationBuilder.DropColumn(
|
|
||||||
name: "GuildConfigId",
|
|
||||||
table: "WarningPunishment");
|
|
||||||
|
|
||||||
migrationBuilder.AddUniqueConstraint(
|
|
||||||
name: "AK_WarningPunishment_GuildId_Count",
|
|
||||||
table: "WarningPunishment",
|
|
||||||
columns: new[] { "GuildId", "Count" });
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <inheritdoc />
|
|
||||||
protected override void Down(MigrationBuilder migrationBuilder)
|
|
||||||
{
|
|
||||||
migrationBuilder.DropUniqueConstraint(
|
|
||||||
name: "AK_WarningPunishment_GuildId_Count",
|
|
||||||
table: "WarningPunishment");
|
|
||||||
|
|
||||||
migrationBuilder.DropColumn(
|
|
||||||
name: "GuildId",
|
|
||||||
table: "WarningPunishment");
|
|
||||||
|
|
||||||
migrationBuilder.AddColumn<int>(
|
|
||||||
name: "GuildConfigId",
|
|
||||||
table: "WarningPunishment",
|
|
||||||
type: "INTEGER",
|
|
||||||
nullable: true);
|
|
||||||
|
|
||||||
migrationBuilder.CreateIndex(
|
|
||||||
name: "IX_WarningPunishment_GuildConfigId",
|
|
||||||
table: "WarningPunishment",
|
|
||||||
column: "GuildConfigId");
|
|
||||||
|
|
||||||
migrationBuilder.AddForeignKey(
|
|
||||||
name: "FK_WarningPunishment_GuildConfigs_GuildConfigId",
|
|
||||||
table: "WarningPunishment",
|
|
||||||
column: "GuildConfigId",
|
|
||||||
principalTable: "GuildConfigs",
|
|
||||||
principalColumn: "Id",
|
|
||||||
onDelete: ReferentialAction.Cascade);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
File diff suppressed because it is too large
Load diff
|
@ -1,53 +0,0 @@
|
||||||
using Microsoft.EntityFrameworkCore.Migrations;
|
|
||||||
|
|
||||||
#nullable disable
|
|
||||||
|
|
||||||
namespace EllieBot.Migrations
|
|
||||||
{
|
|
||||||
/// <inheritdoc />
|
|
||||||
public partial class ncanvas : Migration
|
|
||||||
{
|
|
||||||
/// <inheritdoc />
|
|
||||||
protected override void Up(MigrationBuilder migrationBuilder)
|
|
||||||
{
|
|
||||||
migrationBuilder.CreateTable(
|
|
||||||
name: "NCPixel",
|
|
||||||
columns: table => new
|
|
||||||
{
|
|
||||||
Id = table.Column<int>(type: "INTEGER", nullable: false)
|
|
||||||
.Annotation("Sqlite:Autoincrement", true),
|
|
||||||
Position = table.Column<int>(type: "INTEGER", nullable: false),
|
|
||||||
Price = table.Column<long>(type: "INTEGER", nullable: false),
|
|
||||||
OwnerId = table.Column<ulong>(type: "INTEGER", nullable: false),
|
|
||||||
Color = table.Column<uint>(type: "INTEGER", nullable: false),
|
|
||||||
Text = table.Column<string>(type: "TEXT", maxLength: 256, nullable: false)
|
|
||||||
},
|
|
||||||
constraints: table =>
|
|
||||||
{
|
|
||||||
table.PrimaryKey("PK_NCPixel", x => x.Id);
|
|
||||||
table.UniqueConstraint("AK_NCPixel_Position", x => x.Position);
|
|
||||||
});
|
|
||||||
|
|
||||||
migrationBuilder.CreateIndex(
|
|
||||||
name: "IX_DiscordUser_Username",
|
|
||||||
table: "DiscordUser",
|
|
||||||
column: "Username");
|
|
||||||
|
|
||||||
migrationBuilder.CreateIndex(
|
|
||||||
name: "IX_NCPixel_OwnerId",
|
|
||||||
table: "NCPixel",
|
|
||||||
column: "OwnerId");
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <inheritdoc />
|
|
||||||
protected override void Down(MigrationBuilder migrationBuilder)
|
|
||||||
{
|
|
||||||
migrationBuilder.DropTable(
|
|
||||||
name: "NCPixel");
|
|
||||||
|
|
||||||
migrationBuilder.DropIndex(
|
|
||||||
name: "IX_DiscordUser_Username",
|
|
||||||
table: "DiscordUser");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
File diff suppressed because it is too large
Load diff
|
@ -1,53 +0,0 @@
|
||||||
using System;
|
|
||||||
using Microsoft.EntityFrameworkCore.Migrations;
|
|
||||||
|
|
||||||
#nullable disable
|
|
||||||
|
|
||||||
namespace EllieBot.Migrations
|
|
||||||
{
|
|
||||||
/// <inheritdoc />
|
|
||||||
public partial class nodiscrimandflagtranslate : Migration
|
|
||||||
{
|
|
||||||
/// <inheritdoc />
|
|
||||||
protected override void Up(MigrationBuilder migrationBuilder)
|
|
||||||
{
|
|
||||||
migrationBuilder.DropColumn(
|
|
||||||
name: "Discriminator",
|
|
||||||
table: "DiscordUser");
|
|
||||||
|
|
||||||
migrationBuilder.CreateTable(
|
|
||||||
name: "FlagTranslateChannel",
|
|
||||||
columns: table => new
|
|
||||||
{
|
|
||||||
Id = table.Column<int>(type: "INTEGER", nullable: false)
|
|
||||||
.Annotation("Sqlite:Autoincrement", true),
|
|
||||||
GuildId = table.Column<ulong>(type: "INTEGER", nullable: false),
|
|
||||||
ChannelId = table.Column<ulong>(type: "INTEGER", nullable: false),
|
|
||||||
DateAdded = table.Column<DateTime>(type: "TEXT", nullable: true)
|
|
||||||
},
|
|
||||||
constraints: table =>
|
|
||||||
{
|
|
||||||
table.PrimaryKey("PK_FlagTranslateChannel", x => x.Id);
|
|
||||||
});
|
|
||||||
|
|
||||||
migrationBuilder.CreateIndex(
|
|
||||||
name: "IX_FlagTranslateChannel_GuildId_ChannelId",
|
|
||||||
table: "FlagTranslateChannel",
|
|
||||||
columns: new[] { "GuildId", "ChannelId" },
|
|
||||||
unique: true);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <inheritdoc />
|
|
||||||
protected override void Down(MigrationBuilder migrationBuilder)
|
|
||||||
{
|
|
||||||
migrationBuilder.DropTable(
|
|
||||||
name: "FlagTranslateChannel");
|
|
||||||
|
|
||||||
migrationBuilder.AddColumn<string>(
|
|
||||||
name: "Discriminator",
|
|
||||||
table: "DiscordUser",
|
|
||||||
type: "TEXT",
|
|
||||||
nullable: true);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
File diff suppressed because it is too large
Load diff
|
@ -1,47 +0,0 @@
|
||||||
using Microsoft.EntityFrameworkCore.Migrations;
|
|
||||||
|
|
||||||
#nullable disable
|
|
||||||
|
|
||||||
namespace EllieBot.Migrations
|
|
||||||
{
|
|
||||||
/// <inheritdoc />
|
|
||||||
public partial class betstats : Migration
|
|
||||||
{
|
|
||||||
/// <inheritdoc />
|
|
||||||
protected override void Up(MigrationBuilder migrationBuilder)
|
|
||||||
{
|
|
||||||
migrationBuilder.CreateTable(
|
|
||||||
name: "UserBetStats",
|
|
||||||
columns: table => new
|
|
||||||
{
|
|
||||||
Id = table.Column<int>(type: "INTEGER", nullable: false)
|
|
||||||
.Annotation("Sqlite:Autoincrement", true),
|
|
||||||
UserId = table.Column<ulong>(type: "INTEGER", nullable: false),
|
|
||||||
Game = table.Column<int>(type: "INTEGER", nullable: false),
|
|
||||||
WinCount = table.Column<long>(type: "INTEGER", nullable: false),
|
|
||||||
LoseCount = table.Column<long>(type: "INTEGER", nullable: false),
|
|
||||||
TotalBet = table.Column<decimal>(type: "TEXT", nullable: false),
|
|
||||||
PaidOut = table.Column<decimal>(type: "TEXT", nullable: false),
|
|
||||||
MaxWin = table.Column<long>(type: "INTEGER", nullable: false),
|
|
||||||
MaxBet = table.Column<long>(type: "INTEGER", nullable: false)
|
|
||||||
},
|
|
||||||
constraints: table =>
|
|
||||||
{
|
|
||||||
table.PrimaryKey("PK_UserBetStats", x => x.Id);
|
|
||||||
});
|
|
||||||
|
|
||||||
migrationBuilder.CreateIndex(
|
|
||||||
name: "IX_UserBetStats_UserId_Game",
|
|
||||||
table: "UserBetStats",
|
|
||||||
columns: new[] { "UserId", "Game" },
|
|
||||||
unique: true);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <inheritdoc />
|
|
||||||
protected override void Down(MigrationBuilder migrationBuilder)
|
|
||||||
{
|
|
||||||
migrationBuilder.DropTable(
|
|
||||||
name: "UserBetStats");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
File diff suppressed because it is too large
Load diff
|
@ -1,34 +0,0 @@
|
||||||
using Microsoft.EntityFrameworkCore.Migrations;
|
|
||||||
|
|
||||||
#nullable disable
|
|
||||||
|
|
||||||
namespace EllieBot.Migrations
|
|
||||||
{
|
|
||||||
/// <inheritdoc />
|
|
||||||
public partial class rakeback : Migration
|
|
||||||
{
|
|
||||||
/// <inheritdoc />
|
|
||||||
protected override void Up(MigrationBuilder migrationBuilder)
|
|
||||||
{
|
|
||||||
migrationBuilder.CreateTable(
|
|
||||||
name: "Rakeback",
|
|
||||||
columns: table => new
|
|
||||||
{
|
|
||||||
UserId = table.Column<ulong>(type: "INTEGER", nullable: false)
|
|
||||||
.Annotation("Sqlite:Autoincrement", true),
|
|
||||||
Amount = table.Column<decimal>(type: "TEXT", nullable: false)
|
|
||||||
},
|
|
||||||
constraints: table =>
|
|
||||||
{
|
|
||||||
table.PrimaryKey("PK_Rakeback", x => x.UserId);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <inheritdoc />
|
|
||||||
protected override void Down(MigrationBuilder migrationBuilder)
|
|
||||||
{
|
|
||||||
migrationBuilder.DropTable(
|
|
||||||
name: "Rakeback");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
File diff suppressed because it is too large
Load diff
|
@ -1,73 +0,0 @@
|
||||||
using Microsoft.EntityFrameworkCore.Migrations;
|
|
||||||
|
|
||||||
#nullable disable
|
|
||||||
|
|
||||||
namespace EllieBot.Migrations
|
|
||||||
{
|
|
||||||
/// <inheritdoc />
|
|
||||||
public partial class btnroles_guildcolors : Migration
|
|
||||||
{
|
|
||||||
/// <inheritdoc />
|
|
||||||
protected override void Up(MigrationBuilder migrationBuilder)
|
|
||||||
{
|
|
||||||
migrationBuilder.CreateTable(
|
|
||||||
name: "ButtonRole",
|
|
||||||
columns: table => new
|
|
||||||
{
|
|
||||||
Id = table.Column<int>(type: "INTEGER", nullable: false)
|
|
||||||
.Annotation("Sqlite:Autoincrement", true),
|
|
||||||
ButtonId = table.Column<string>(type: "TEXT", maxLength: 200, nullable: false),
|
|
||||||
GuildId = table.Column<ulong>(type: "INTEGER", nullable: false),
|
|
||||||
ChannelId = table.Column<ulong>(type: "INTEGER", nullable: false),
|
|
||||||
MessageId = table.Column<ulong>(type: "INTEGER", nullable: false),
|
|
||||||
Position = table.Column<int>(type: "INTEGER", nullable: false),
|
|
||||||
RoleId = table.Column<ulong>(type: "INTEGER", nullable: false),
|
|
||||||
Emote = table.Column<string>(type: "TEXT", maxLength: 100, nullable: false),
|
|
||||||
Label = table.Column<string>(type: "TEXT", maxLength: 50, nullable: false),
|
|
||||||
Exclusive = table.Column<bool>(type: "INTEGER", nullable: false)
|
|
||||||
},
|
|
||||||
constraints: table =>
|
|
||||||
{
|
|
||||||
table.PrimaryKey("PK_ButtonRole", x => x.Id);
|
|
||||||
table.UniqueConstraint("AK_ButtonRole_RoleId_MessageId", x => new { x.RoleId, x.MessageId });
|
|
||||||
});
|
|
||||||
|
|
||||||
migrationBuilder.CreateTable(
|
|
||||||
name: "GuildColors",
|
|
||||||
columns: table => new
|
|
||||||
{
|
|
||||||
Id = table.Column<int>(type: "INTEGER", nullable: false)
|
|
||||||
.Annotation("Sqlite:Autoincrement", true),
|
|
||||||
GuildId = table.Column<ulong>(type: "INTEGER", nullable: false),
|
|
||||||
OkColor = table.Column<string>(type: "TEXT", maxLength: 9, nullable: true),
|
|
||||||
ErrorColor = table.Column<string>(type: "TEXT", maxLength: 9, nullable: true),
|
|
||||||
PendingColor = table.Column<string>(type: "TEXT", maxLength: 9, nullable: true)
|
|
||||||
},
|
|
||||||
constraints: table =>
|
|
||||||
{
|
|
||||||
table.PrimaryKey("PK_GuildColors", x => x.Id);
|
|
||||||
});
|
|
||||||
|
|
||||||
migrationBuilder.CreateIndex(
|
|
||||||
name: "IX_ButtonRole_GuildId",
|
|
||||||
table: "ButtonRole",
|
|
||||||
column: "GuildId");
|
|
||||||
|
|
||||||
migrationBuilder.CreateIndex(
|
|
||||||
name: "IX_GuildColors_GuildId",
|
|
||||||
table: "GuildColors",
|
|
||||||
column: "GuildId",
|
|
||||||
unique: true);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <inheritdoc />
|
|
||||||
protected override void Down(MigrationBuilder migrationBuilder)
|
|
||||||
{
|
|
||||||
migrationBuilder.DropTable(
|
|
||||||
name: "ButtonRole");
|
|
||||||
|
|
||||||
migrationBuilder.DropTable(
|
|
||||||
name: "GuildColors");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
File diff suppressed because it is too large
Load diff
|
@ -1,152 +0,0 @@
|
||||||
using System;
|
|
||||||
using Microsoft.EntityFrameworkCore.Migrations;
|
|
||||||
|
|
||||||
#nullable disable
|
|
||||||
|
|
||||||
namespace EllieBot.Migrations
|
|
||||||
{
|
|
||||||
/// <inheritdoc />
|
|
||||||
public partial class sarrework : Migration
|
|
||||||
{
|
|
||||||
/// <inheritdoc />
|
|
||||||
protected override void Up(MigrationBuilder migrationBuilder)
|
|
||||||
{
|
|
||||||
migrationBuilder.DropTable(
|
|
||||||
name: "GroupName");
|
|
||||||
|
|
||||||
migrationBuilder.DropTable(
|
|
||||||
name: "SelfAssignableRoles");
|
|
||||||
|
|
||||||
migrationBuilder.CreateTable(
|
|
||||||
name: "SarAutoDelete",
|
|
||||||
columns: table => new
|
|
||||||
{
|
|
||||||
Id = table.Column<int>(type: "INTEGER", nullable: false)
|
|
||||||
.Annotation("Sqlite:Autoincrement", true),
|
|
||||||
GuildId = table.Column<ulong>(type: "INTEGER", nullable: false),
|
|
||||||
IsEnabled = table.Column<bool>(type: "INTEGER", nullable: false)
|
|
||||||
},
|
|
||||||
constraints: table =>
|
|
||||||
{
|
|
||||||
table.PrimaryKey("PK_SarAutoDelete", x => x.Id);
|
|
||||||
});
|
|
||||||
|
|
||||||
migrationBuilder.CreateTable(
|
|
||||||
name: "SarGroup",
|
|
||||||
columns: table => new
|
|
||||||
{
|
|
||||||
Id = table.Column<int>(type: "INTEGER", nullable: false)
|
|
||||||
.Annotation("Sqlite:Autoincrement", true),
|
|
||||||
GroupNumber = table.Column<int>(type: "INTEGER", nullable: false),
|
|
||||||
GuildId = table.Column<ulong>(type: "INTEGER", nullable: false),
|
|
||||||
RoleReq = table.Column<ulong>(type: "INTEGER", nullable: true),
|
|
||||||
IsExclusive = table.Column<bool>(type: "INTEGER", nullable: false),
|
|
||||||
Name = table.Column<string>(type: "TEXT", maxLength: 100, nullable: true),
|
|
||||||
DateAdded = table.Column<DateTime>(type: "TEXT", nullable: true)
|
|
||||||
},
|
|
||||||
constraints: table =>
|
|
||||||
{
|
|
||||||
table.PrimaryKey("PK_SarGroup", x => x.Id);
|
|
||||||
table.UniqueConstraint("AK_SarGroup_GuildId_GroupNumber", x => new { x.GuildId, x.GroupNumber });
|
|
||||||
});
|
|
||||||
|
|
||||||
migrationBuilder.CreateTable(
|
|
||||||
name: "Sar",
|
|
||||||
columns: table => new
|
|
||||||
{
|
|
||||||
Id = table.Column<int>(type: "INTEGER", nullable: false)
|
|
||||||
.Annotation("Sqlite:Autoincrement", true),
|
|
||||||
GuildId = table.Column<ulong>(type: "INTEGER", nullable: false),
|
|
||||||
RoleId = table.Column<ulong>(type: "INTEGER", nullable: false),
|
|
||||||
SarGroupId = table.Column<int>(type: "INTEGER", nullable: false),
|
|
||||||
LevelReq = table.Column<int>(type: "INTEGER", nullable: false)
|
|
||||||
},
|
|
||||||
constraints: table =>
|
|
||||||
{
|
|
||||||
table.PrimaryKey("PK_Sar", x => x.Id);
|
|
||||||
table.UniqueConstraint("AK_Sar_GuildId_RoleId", x => new { x.GuildId, x.RoleId });
|
|
||||||
table.ForeignKey(
|
|
||||||
name: "FK_Sar_SarGroup_SarGroupId",
|
|
||||||
column: x => x.SarGroupId,
|
|
||||||
principalTable: "SarGroup",
|
|
||||||
principalColumn: "Id",
|
|
||||||
onDelete: ReferentialAction.Cascade);
|
|
||||||
});
|
|
||||||
|
|
||||||
migrationBuilder.CreateIndex(
|
|
||||||
name: "IX_Sar_SarGroupId",
|
|
||||||
table: "Sar",
|
|
||||||
column: "SarGroupId");
|
|
||||||
|
|
||||||
migrationBuilder.CreateIndex(
|
|
||||||
name: "IX_SarAutoDelete_GuildId",
|
|
||||||
table: "SarAutoDelete",
|
|
||||||
column: "GuildId",
|
|
||||||
unique: true);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <inheritdoc />
|
|
||||||
protected override void Down(MigrationBuilder migrationBuilder)
|
|
||||||
{
|
|
||||||
migrationBuilder.DropTable(
|
|
||||||
name: "Sar");
|
|
||||||
|
|
||||||
migrationBuilder.DropTable(
|
|
||||||
name: "SarAutoDelete");
|
|
||||||
|
|
||||||
migrationBuilder.DropTable(
|
|
||||||
name: "SarGroup");
|
|
||||||
|
|
||||||
migrationBuilder.CreateTable(
|
|
||||||
name: "GroupName",
|
|
||||||
columns: table => new
|
|
||||||
{
|
|
||||||
Id = table.Column<int>(type: "INTEGER", nullable: false)
|
|
||||||
.Annotation("Sqlite:Autoincrement", true),
|
|
||||||
GuildConfigId = table.Column<int>(type: "INTEGER", nullable: false),
|
|
||||||
DateAdded = table.Column<DateTime>(type: "TEXT", nullable: true),
|
|
||||||
Name = table.Column<string>(type: "TEXT", nullable: true),
|
|
||||||
Number = table.Column<int>(type: "INTEGER", nullable: false)
|
|
||||||
},
|
|
||||||
constraints: table =>
|
|
||||||
{
|
|
||||||
table.PrimaryKey("PK_GroupName", x => x.Id);
|
|
||||||
table.ForeignKey(
|
|
||||||
name: "FK_GroupName_GuildConfigs_GuildConfigId",
|
|
||||||
column: x => x.GuildConfigId,
|
|
||||||
principalTable: "GuildConfigs",
|
|
||||||
principalColumn: "Id",
|
|
||||||
onDelete: ReferentialAction.Cascade);
|
|
||||||
});
|
|
||||||
|
|
||||||
migrationBuilder.CreateTable(
|
|
||||||
name: "SelfAssignableRoles",
|
|
||||||
columns: table => new
|
|
||||||
{
|
|
||||||
Id = table.Column<int>(type: "INTEGER", nullable: false)
|
|
||||||
.Annotation("Sqlite:Autoincrement", true),
|
|
||||||
DateAdded = table.Column<DateTime>(type: "TEXT", nullable: true),
|
|
||||||
Group = table.Column<int>(type: "INTEGER", nullable: false, defaultValue: 0),
|
|
||||||
GuildId = table.Column<ulong>(type: "INTEGER", nullable: false),
|
|
||||||
LevelRequirement = table.Column<int>(type: "INTEGER", nullable: false),
|
|
||||||
RoleId = table.Column<ulong>(type: "INTEGER", nullable: false)
|
|
||||||
},
|
|
||||||
constraints: table =>
|
|
||||||
{
|
|
||||||
table.PrimaryKey("PK_SelfAssignableRoles", x => x.Id);
|
|
||||||
});
|
|
||||||
|
|
||||||
migrationBuilder.CreateIndex(
|
|
||||||
name: "IX_GroupName_GuildConfigId_Number",
|
|
||||||
table: "GroupName",
|
|
||||||
columns: new[] { "GuildConfigId", "Number" },
|
|
||||||
unique: true);
|
|
||||||
|
|
||||||
migrationBuilder.CreateIndex(
|
|
||||||
name: "IX_SelfAssignableRoles_GuildId_RoleId",
|
|
||||||
table: "SelfAssignableRoles",
|
|
||||||
columns: new[] { "GuildId", "RoleId" },
|
|
||||||
unique: true);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
File diff suppressed because it is too large
Load diff
|
@ -1,107 +0,0 @@
|
||||||
using System;
|
|
||||||
using EllieBot.Migrations;
|
|
||||||
using Microsoft.EntityFrameworkCore.Migrations;
|
|
||||||
|
|
||||||
#nullable disable
|
|
||||||
|
|
||||||
namespace EllieBot.Migrations
|
|
||||||
{
|
|
||||||
/// <inheritdoc />
|
|
||||||
public partial class awardedxptemprolenotify : Migration
|
|
||||||
{
|
|
||||||
/// <inheritdoc />
|
|
||||||
protected override void Up(MigrationBuilder migrationBuilder)
|
|
||||||
{
|
|
||||||
migrationBuilder.DropIndex(
|
|
||||||
name: "IX_UserXpStats_AwardedXp",
|
|
||||||
table: "UserXpStats");
|
|
||||||
|
|
||||||
MigrationQueries.MergeAwardedXp(migrationBuilder);
|
|
||||||
migrationBuilder.DropColumn(
|
|
||||||
name: "AwardedXp",
|
|
||||||
table: "UserXpStats");
|
|
||||||
|
|
||||||
migrationBuilder.DropColumn(
|
|
||||||
name: "NotifyOnLevelUp",
|
|
||||||
table: "UserXpStats");
|
|
||||||
|
|
||||||
migrationBuilder.CreateTable(
|
|
||||||
name: "Notify",
|
|
||||||
columns: table => new
|
|
||||||
{
|
|
||||||
Id = table.Column<int>(type: "INTEGER", nullable: false)
|
|
||||||
.Annotation("Sqlite:Autoincrement", true),
|
|
||||||
GuildId = table.Column<ulong>(type: "INTEGER", nullable: false),
|
|
||||||
ChannelId = table.Column<ulong>(type: "INTEGER", nullable: false),
|
|
||||||
Type = table.Column<int>(type: "INTEGER", nullable: false),
|
|
||||||
Message = table.Column<string>(type: "TEXT", maxLength: 10000, nullable: false)
|
|
||||||
},
|
|
||||||
constraints: table =>
|
|
||||||
{
|
|
||||||
table.PrimaryKey("PK_Notify", x => x.Id);
|
|
||||||
table.UniqueConstraint("AK_Notify_GuildId_Type", x => new { x.GuildId, x.Type });
|
|
||||||
});
|
|
||||||
|
|
||||||
migrationBuilder.CreateTable(
|
|
||||||
name: "TempRole",
|
|
||||||
columns: table => new
|
|
||||||
{
|
|
||||||
Id = table.Column<int>(type: "INTEGER", nullable: false)
|
|
||||||
.Annotation("Sqlite:Autoincrement", true),
|
|
||||||
GuildId = table.Column<ulong>(type: "INTEGER", nullable: false),
|
|
||||||
Remove = table.Column<bool>(type: "INTEGER", nullable: false),
|
|
||||||
RoleId = table.Column<ulong>(type: "INTEGER", nullable: false),
|
|
||||||
UserId = table.Column<ulong>(type: "INTEGER", nullable: false),
|
|
||||||
ExpiresAt = table.Column<DateTime>(type: "TEXT", nullable: false)
|
|
||||||
},
|
|
||||||
constraints: table =>
|
|
||||||
{
|
|
||||||
table.PrimaryKey("PK_TempRole", x => x.Id);
|
|
||||||
table.UniqueConstraint("AK_TempRole_GuildId_UserId_RoleId", x => new { x.GuildId, x.UserId, x.RoleId });
|
|
||||||
});
|
|
||||||
|
|
||||||
migrationBuilder.CreateIndex(
|
|
||||||
name: "IX_UserBetStats_MaxWin",
|
|
||||||
table: "UserBetStats",
|
|
||||||
column: "MaxWin");
|
|
||||||
|
|
||||||
migrationBuilder.CreateIndex(
|
|
||||||
name: "IX_TempRole_ExpiresAt",
|
|
||||||
table: "TempRole",
|
|
||||||
column: "ExpiresAt");
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <inheritdoc />
|
|
||||||
protected override void Down(MigrationBuilder migrationBuilder)
|
|
||||||
{
|
|
||||||
migrationBuilder.DropTable(
|
|
||||||
name: "Notify");
|
|
||||||
|
|
||||||
migrationBuilder.DropTable(
|
|
||||||
name: "TempRole");
|
|
||||||
|
|
||||||
migrationBuilder.DropIndex(
|
|
||||||
name: "IX_UserBetStats_MaxWin",
|
|
||||||
table: "UserBetStats");
|
|
||||||
|
|
||||||
migrationBuilder.AddColumn<long>(
|
|
||||||
name: "AwardedXp",
|
|
||||||
table: "UserXpStats",
|
|
||||||
type: "INTEGER",
|
|
||||||
nullable: false,
|
|
||||||
defaultValue: 0L);
|
|
||||||
|
|
||||||
migrationBuilder.AddColumn<int>(
|
|
||||||
name: "NotifyOnLevelUp",
|
|
||||||
table: "UserXpStats",
|
|
||||||
type: "INTEGER",
|
|
||||||
nullable: false,
|
|
||||||
defaultValue: 0);
|
|
||||||
|
|
||||||
migrationBuilder.CreateIndex(
|
|
||||||
name: "IX_UserXpStats_AwardedXp",
|
|
||||||
table: "UserXpStats",
|
|
||||||
column: "AwardedXp");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
File diff suppressed because it is too large
Load diff
|
@ -46,7 +46,7 @@ public partial class Administration : EllieModule<AdministrationService>
|
||||||
[RequireContext(ContextType.Guild)]
|
[RequireContext(ContextType.Guild)]
|
||||||
[UserPerm(GuildPerm.Administrator)]
|
[UserPerm(GuildPerm.Administrator)]
|
||||||
[BotPerm(GuildPerm.ManageGuild)]
|
[BotPerm(GuildPerm.ManageGuild)]
|
||||||
public async Task ImageOnlyChannel(ParsedTimespan timespan = null)
|
public async Task ImageOnlyChannel(StoopidTime time = null)
|
||||||
{
|
{
|
||||||
var newValue = await _somethingOnly.ToggleImageOnlyChannelAsync(ctx.Guild.Id, ctx.Channel.Id);
|
var newValue = await _somethingOnly.ToggleImageOnlyChannelAsync(ctx.Guild.Id, ctx.Channel.Id);
|
||||||
if (newValue)
|
if (newValue)
|
||||||
|
@ -54,12 +54,12 @@ public partial class Administration : EllieModule<AdministrationService>
|
||||||
else
|
else
|
||||||
await Response().Pending(strs.imageonly_disable).SendAsync();
|
await Response().Pending(strs.imageonly_disable).SendAsync();
|
||||||
}
|
}
|
||||||
|
|
||||||
[Cmd]
|
[Cmd]
|
||||||
[RequireContext(ContextType.Guild)]
|
[RequireContext(ContextType.Guild)]
|
||||||
[UserPerm(GuildPerm.Administrator)]
|
[UserPerm(GuildPerm.Administrator)]
|
||||||
[BotPerm(GuildPerm.ManageGuild)]
|
[BotPerm(GuildPerm.ManageGuild)]
|
||||||
public async Task LinkOnlyChannel(ParsedTimespan timespan = null)
|
public async Task LinkOnlyChannel(StoopidTime time = null)
|
||||||
{
|
{
|
||||||
var newValue = await _somethingOnly.ToggleLinkOnlyChannelAsync(ctx.Guild.Id, ctx.Channel.Id);
|
var newValue = await _somethingOnly.ToggleLinkOnlyChannelAsync(ctx.Guild.Id, ctx.Channel.Id);
|
||||||
if (newValue)
|
if (newValue)
|
||||||
|
@ -72,10 +72,10 @@ public partial class Administration : EllieModule<AdministrationService>
|
||||||
[RequireContext(ContextType.Guild)]
|
[RequireContext(ContextType.Guild)]
|
||||||
[UserPerm(ChannelPerm.ManageChannels)]
|
[UserPerm(ChannelPerm.ManageChannels)]
|
||||||
[BotPerm(ChannelPerm.ManageChannels)]
|
[BotPerm(ChannelPerm.ManageChannels)]
|
||||||
public async Task Slowmode(ParsedTimespan timespan = null)
|
public async Task Slowmode(StoopidTime time = null)
|
||||||
{
|
{
|
||||||
var seconds = (int?)timespan?.Time.TotalSeconds ?? 0;
|
var seconds = (int?)time?.Time.TotalSeconds ?? 0;
|
||||||
if (timespan is not null && (timespan.Time < TimeSpan.FromSeconds(0) || timespan.Time > TimeSpan.FromHours(6)))
|
if (time is not null && (time.Time < TimeSpan.FromSeconds(0) || time.Time > TimeSpan.FromHours(6)))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
await ((ITextChannel)ctx.Channel).ModifyAsync(tcp =>
|
await ((ITextChannel)ctx.Channel).ModifyAsync(tcp =>
|
||||||
|
@ -96,7 +96,7 @@ public partial class Administration : EllieModule<AdministrationService>
|
||||||
var guild = (SocketGuild)ctx.Guild;
|
var guild = (SocketGuild)ctx.Guild;
|
||||||
var (enabled, channels) = _service.GetDelMsgOnCmdData(ctx.Guild.Id);
|
var (enabled, channels) = _service.GetDelMsgOnCmdData(ctx.Guild.Id);
|
||||||
|
|
||||||
var embed = CreateEmbed()
|
var embed = _sender.CreateEmbed()
|
||||||
.WithOkColor()
|
.WithOkColor()
|
||||||
.WithTitle(GetText(strs.server_delmsgoncmd))
|
.WithTitle(GetText(strs.server_delmsgoncmd))
|
||||||
.WithDescription(enabled ? "✅" : "❌");
|
.WithDescription(enabled ? "✅" : "❌");
|
||||||
|
@ -221,7 +221,7 @@ public partial class Administration : EllieModule<AdministrationService>
|
||||||
[BotPerm(GuildPerm.ManageChannels)]
|
[BotPerm(GuildPerm.ManageChannels)]
|
||||||
public async Task CreaTxtChanl([Leftover] string channelName)
|
public async Task CreaTxtChanl([Leftover] string channelName)
|
||||||
{
|
{
|
||||||
var txtCh = await ctx.Guild.CreateTextChannelAsync(channelName);
|
var txtCh = await ctx.Guild.CreateTextChannelAsync(channelName);
|
||||||
await Response().Confirm(strs.createtextchan(Format.Bold(txtCh.Name))).SendAsync();
|
await Response().Confirm(strs.createtextchan(Format.Bold(txtCh.Name))).SendAsync();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -298,18 +298,18 @@ public partial class Administration : EllieModule<AdministrationService>
|
||||||
[RequireContext(ContextType.Guild)]
|
[RequireContext(ContextType.Guild)]
|
||||||
[UserPerm(ChannelPerm.ManageMessages)]
|
[UserPerm(ChannelPerm.ManageMessages)]
|
||||||
[BotPerm(ChannelPerm.ManageMessages)]
|
[BotPerm(ChannelPerm.ManageMessages)]
|
||||||
public Task Delete(ulong messageId, ParsedTimespan timespan = null)
|
public Task Delete(ulong messageId, StoopidTime time = null)
|
||||||
=> Delete((ITextChannel)ctx.Channel, messageId, timespan);
|
=> Delete((ITextChannel)ctx.Channel, messageId, time);
|
||||||
|
|
||||||
[Cmd]
|
[Cmd]
|
||||||
[RequireContext(ContextType.Guild)]
|
[RequireContext(ContextType.Guild)]
|
||||||
public async Task Delete(ITextChannel channel, ulong messageId, ParsedTimespan timespan = null)
|
public async Task Delete(ITextChannel channel, ulong messageId, StoopidTime time = null)
|
||||||
=> await InternalMessageAction(channel, messageId, timespan, msg => msg.DeleteAsync());
|
=> await InternalMessageAction(channel, messageId, time, msg => msg.DeleteAsync());
|
||||||
|
|
||||||
private async Task InternalMessageAction(
|
private async Task InternalMessageAction(
|
||||||
ITextChannel channel,
|
ITextChannel channel,
|
||||||
ulong messageId,
|
ulong messageId,
|
||||||
ParsedTimespan timespan,
|
StoopidTime time,
|
||||||
Func<IMessage, Task> func)
|
Func<IMessage, Task> func)
|
||||||
{
|
{
|
||||||
var userPerms = ((SocketGuildUser)ctx.User).GetPermissions(channel);
|
var userPerms = ((SocketGuildUser)ctx.User).GetPermissions(channel);
|
||||||
|
@ -334,13 +334,13 @@ public partial class Administration : EllieModule<AdministrationService>
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (timespan is null)
|
if (time is null)
|
||||||
await msg.DeleteAsync();
|
await msg.DeleteAsync();
|
||||||
else if (timespan.Time <= TimeSpan.FromDays(7))
|
else if (time.Time <= TimeSpan.FromDays(7))
|
||||||
{
|
{
|
||||||
_ = Task.Run(async () =>
|
_ = Task.Run(async () =>
|
||||||
{
|
{
|
||||||
await Task.Delay(timespan.Time);
|
await Task.Delay(time.Time);
|
||||||
await msg.DeleteAsync();
|
await msg.DeleteAsync();
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@ -360,11 +360,11 @@ public partial class Administration : EllieModule<AdministrationService>
|
||||||
{
|
{
|
||||||
if (ctx.Channel is not SocketTextChannel stc)
|
if (ctx.Channel is not SocketTextChannel stc)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
await stc.CreateThreadAsync(name, message: ctx.Message.ReferencedMessage);
|
await stc.CreateThreadAsync(name, message: ctx.Message.ReferencedMessage);
|
||||||
await ctx.OkAsync();
|
await ctx.OkAsync();
|
||||||
}
|
}
|
||||||
|
|
||||||
[Cmd]
|
[Cmd]
|
||||||
[BotPerm(ChannelPermission.ManageThreads)]
|
[BotPerm(ChannelPermission.ManageThreads)]
|
||||||
[UserPerm(ChannelPermission.ManageThreads)]
|
[UserPerm(ChannelPermission.ManageThreads)]
|
||||||
|
@ -380,7 +380,7 @@ public partial class Administration : EllieModule<AdministrationService>
|
||||||
await Response().Error(strs.not_found).SendAsync();
|
await Response().Error(strs.not_found).SendAsync();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
await t.DeleteAsync();
|
await t.DeleteAsync();
|
||||||
await ctx.OkAsync();
|
await ctx.OkAsync();
|
||||||
}
|
}
|
||||||
|
@ -406,7 +406,7 @@ public partial class Administration : EllieModule<AdministrationService>
|
||||||
await Response().Confirm(strs.autopublish_disable).SendAsync();
|
await Response().Confirm(strs.autopublish_disable).SendAsync();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
[Cmd]
|
[Cmd]
|
||||||
[UserPerm(GuildPerm.ManageNicknames)]
|
[UserPerm(GuildPerm.ManageNicknames)]
|
||||||
[BotPerm(GuildPerm.ChangeNickname)]
|
[BotPerm(GuildPerm.ChangeNickname)]
|
||||||
|
@ -450,9 +450,8 @@ public partial class Administration : EllieModule<AdministrationService>
|
||||||
public async Task SetServerBanner([Leftover] string img = null)
|
public async Task SetServerBanner([Leftover] string img = null)
|
||||||
{
|
{
|
||||||
// Tier2 or higher is required to set a banner.
|
// Tier2 or higher is required to set a banner.
|
||||||
if (ctx.Guild.PremiumTier is PremiumTier.Tier1 or PremiumTier.None)
|
if (ctx.Guild.PremiumTier is PremiumTier.Tier1 or PremiumTier.None) return;
|
||||||
return;
|
|
||||||
|
|
||||||
var result = await _service.SetServerBannerAsync(ctx.Guild, img);
|
var result = await _service.SetServerBannerAsync(ctx.Guild, img);
|
||||||
|
|
||||||
switch (result)
|
switch (result)
|
||||||
|
@ -473,7 +472,7 @@ public partial class Administration : EllieModule<AdministrationService>
|
||||||
throw new ArgumentOutOfRangeException();
|
throw new ArgumentOutOfRangeException();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
[Cmd]
|
[Cmd]
|
||||||
[RequireContext(ContextType.Guild)]
|
[RequireContext(ContextType.Guild)]
|
||||||
[UserPerm(GuildPermission.ManageGuild)]
|
[UserPerm(GuildPermission.ManageGuild)]
|
||||||
|
|
|
@ -3,7 +3,7 @@ using Microsoft.EntityFrameworkCore;
|
||||||
using EllieBot.Db.Models;
|
using EllieBot.Db.Models;
|
||||||
using EllieBot.Modules.Administration._common.results;
|
using EllieBot.Modules.Administration._common.results;
|
||||||
|
|
||||||
namespace EllieBot.Modules.Administration;
|
namespace EllieBot.Modules.Administration.Services;
|
||||||
|
|
||||||
public class AdministrationService : IEService
|
public class AdministrationService : IEService
|
||||||
{
|
{
|
||||||
|
|
|
@ -25,13 +25,6 @@ public partial class Administration
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// the user can't aar the role which is greater or equal to the bot's highest role
|
|
||||||
if (role.Position >= ((SocketGuild)ctx.Guild).CurrentUser.GetRoles().Max(x => x.Position))
|
|
||||||
{
|
|
||||||
await Response().Error(strs.hierarchy).SendAsync();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
var roles = await _service.ToggleAarAsync(ctx.Guild.Id, role.Id);
|
var roles = await _service.ToggleAarAsync(ctx.Guild.Id, role.Id);
|
||||||
if (roles.Count == 0)
|
if (roles.Count == 0)
|
||||||
await Response().Confirm(strs.aar_disabled).SendAsync();
|
await Response().Confirm(strs.aar_disabled).SendAsync();
|
||||||
|
|
|
@ -1,3 +1,4 @@
|
||||||
|
#nullable disable
|
||||||
using LinqToDB;
|
using LinqToDB;
|
||||||
using LinqToDB.EntityFrameworkCore;
|
using LinqToDB.EntityFrameworkCore;
|
||||||
using EllieBot.Common.ModuleBehaviors;
|
using EllieBot.Common.ModuleBehaviors;
|
||||||
|
@ -10,7 +11,7 @@ public class AutoPublishService : IExecNoCommand, IReadyExecutor, IEService
|
||||||
private readonly DbService _db;
|
private readonly DbService _db;
|
||||||
private readonly DiscordSocketClient _client;
|
private readonly DiscordSocketClient _client;
|
||||||
private readonly IBotCredsProvider _creds;
|
private readonly IBotCredsProvider _creds;
|
||||||
private ConcurrentDictionary<ulong, ulong> _enabled = new();
|
private ConcurrentDictionary<ulong, ulong> _enabled;
|
||||||
|
|
||||||
public AutoPublishService(DbService db, DiscordSocketClient client, IBotCredsProvider creds)
|
public AutoPublishService(DbService db, DiscordSocketClient client, IBotCredsProvider creds)
|
||||||
{
|
{
|
||||||
|
@ -19,7 +20,7 @@ public class AutoPublishService : IExecNoCommand, IReadyExecutor, IEService
|
||||||
_creds = creds;
|
_creds = creds;
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task ExecOnNoCommandAsync(IGuild? guild, IUserMessage msg)
|
public async Task ExecOnNoCommandAsync(IGuild guild, IUserMessage msg)
|
||||||
{
|
{
|
||||||
if (guild is null)
|
if (guild is null)
|
||||||
return;
|
return;
|
||||||
|
@ -35,6 +36,8 @@ public class AutoPublishService : IExecNoCommand, IReadyExecutor, IEService
|
||||||
RetryMode = RetryMode.AlwaysFail
|
RetryMode = RetryMode.AlwaysFail
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// todo GUILDS
|
||||||
|
|
||||||
public async Task OnReadyAsync()
|
public async Task OnReadyAsync()
|
||||||
{
|
{
|
||||||
|
@ -42,31 +45,19 @@ public class AutoPublishService : IExecNoCommand, IReadyExecutor, IEService
|
||||||
|
|
||||||
await using var ctx = _db.GetDbContext();
|
await using var ctx = _db.GetDbContext();
|
||||||
var items = await ctx.GetTable<AutoPublishChannel>()
|
var items = await ctx.GetTable<AutoPublishChannel>()
|
||||||
.Where(x => Linq2DbExpressions.GuildOnShard(x.GuildId, creds.TotalShards, _client.ShardId))
|
.Where(x => Linq2DbExpressions.GuildOnShard(x.GuildId, creds.TotalShards, _client.ShardId))
|
||||||
.ToListAsyncLinqToDB();
|
.ToListAsyncLinqToDB();
|
||||||
|
|
||||||
_enabled = items
|
_enabled = items
|
||||||
.ToDictionary(x => x.GuildId, x => x.ChannelId)
|
.ToDictionary(x => x.GuildId, x => x.ChannelId)
|
||||||
.ToConcurrent();
|
.ToConcurrent();
|
||||||
|
|
||||||
_client.LeftGuild += ClientOnLeftGuild;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task ClientOnLeftGuild(SocketGuild guild)
|
|
||||||
{
|
|
||||||
await using var ctx = _db.GetDbContext();
|
|
||||||
_enabled.TryRemove(guild.Id, out _);
|
|
||||||
|
|
||||||
await ctx.GetTable<AutoPublishChannel>()
|
|
||||||
.Where(x => x.GuildId == guild.Id)
|
|
||||||
.DeleteAsync();
|
|
||||||
}
|
|
||||||
|
|
||||||
public async Task<bool> ToggleAutoPublish(ulong guildId, ulong channelId)
|
public async Task<bool> ToggleAutoPublish(ulong guildId, ulong channelId)
|
||||||
{
|
{
|
||||||
await using var ctx = _db.GetDbContext();
|
await using var ctx = _db.GetDbContext();
|
||||||
var deleted = await ctx.GetTable<AutoPublishChannel>()
|
var deleted = await ctx.GetTable<AutoPublishChannel>()
|
||||||
.DeleteAsync(x => x.GuildId == guildId && x.ChannelId == channelId);
|
.DeleteAsync(x => x.GuildId == guildId && x.ChannelId == channelId);
|
||||||
|
|
||||||
if (deleted != 0)
|
if (deleted != 0)
|
||||||
{
|
{
|
||||||
|
@ -75,22 +66,22 @@ public class AutoPublishService : IExecNoCommand, IReadyExecutor, IEService
|
||||||
}
|
}
|
||||||
|
|
||||||
await ctx.GetTable<AutoPublishChannel>()
|
await ctx.GetTable<AutoPublishChannel>()
|
||||||
.InsertOrUpdateAsync(() => new()
|
.InsertOrUpdateAsync(() => new()
|
||||||
{
|
{
|
||||||
GuildId = guildId,
|
GuildId = guildId,
|
||||||
ChannelId = channelId,
|
ChannelId = channelId,
|
||||||
DateAdded = DateTime.UtcNow,
|
DateAdded = DateTime.UtcNow,
|
||||||
},
|
},
|
||||||
old => new()
|
old => new()
|
||||||
{
|
{
|
||||||
ChannelId = channelId,
|
ChannelId = channelId,
|
||||||
DateAdded = DateTime.UtcNow,
|
DateAdded = DateTime.UtcNow,
|
||||||
},
|
},
|
||||||
() => new()
|
() => new()
|
||||||
{
|
{
|
||||||
GuildId = guildId
|
GuildId = guildId
|
||||||
});
|
});
|
||||||
|
|
||||||
_enabled[guildId] = channelId;
|
_enabled[guildId] = channelId;
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
|
|
|
@ -75,7 +75,7 @@ public sealed class CleanupService : ICleanupService, IReadyExecutor, IEService
|
||||||
Log.Information("Leaving {RemainingCount} guilds, 1 every second. {DontDeleteCount} will remain",
|
Log.Information("Leaving {RemainingCount} guilds, 1 every second. {DontDeleteCount} will remain",
|
||||||
allGuildIds.Length - dontDelete.Count,
|
allGuildIds.Length - dontDelete.Count,
|
||||||
dontDelete.Count);
|
dontDelete.Count);
|
||||||
|
|
||||||
foreach (var guildId in allGuildIds)
|
foreach (var guildId in allGuildIds)
|
||||||
{
|
{
|
||||||
if (dontDelete.Contains(guildId))
|
if (dontDelete.Contains(guildId))
|
||||||
|
@ -136,7 +136,7 @@ public sealed class CleanupService : ICleanupService, IReadyExecutor, IEService
|
||||||
await using var linqCtx = ctx.CreateLinqToDBContext();
|
await using var linqCtx = ctx.CreateLinqToDBContext();
|
||||||
await using var tempTable = linqCtx.CreateTempTable<CleanupId>();
|
await using var tempTable = linqCtx.CreateTempTable<CleanupId>();
|
||||||
|
|
||||||
foreach (var chunk in allIds.Chunk(10000))
|
foreach (var chunk in allIds.Chunk(20000))
|
||||||
{
|
{
|
||||||
await tempTable.BulkCopyAsync(chunk.Select(x => new CleanupId()
|
await tempTable.BulkCopyAsync(chunk.Select(x => new CleanupId()
|
||||||
{
|
{
|
||||||
|
@ -187,6 +187,13 @@ public sealed class CleanupService : ICleanupService, IReadyExecutor, IEService
|
||||||
.Contains(x.GuildId))
|
.Contains(x.GuildId))
|
||||||
.DeleteAsync();
|
.DeleteAsync();
|
||||||
|
|
||||||
|
// delete ignored users
|
||||||
|
await ctx.GetTable<DiscordPermOverride>()
|
||||||
|
.Where(x => x.GuildId != null
|
||||||
|
&& !tempTable.Select(x => x.GuildId)
|
||||||
|
.Contains(x.GuildId.Value))
|
||||||
|
.DeleteAsync();
|
||||||
|
|
||||||
// delete perm overrides
|
// delete perm overrides
|
||||||
await ctx.GetTable<DiscordPermOverride>()
|
await ctx.GetTable<DiscordPermOverride>()
|
||||||
.Where(x => x.GuildId != null
|
.Where(x => x.GuildId != null
|
||||||
|
@ -200,66 +207,6 @@ public sealed class CleanupService : ICleanupService, IReadyExecutor, IEService
|
||||||
.Contains(x.GuildId))
|
.Contains(x.GuildId))
|
||||||
.DeleteAsync();
|
.DeleteAsync();
|
||||||
|
|
||||||
// delete autopublish channels
|
|
||||||
await ctx.GetTable<AutoPublishChannel>()
|
|
||||||
.Where(x => !tempTable.Select(x => x.GuildId)
|
|
||||||
.Contains(x.GuildId))
|
|
||||||
.DeleteAsync();
|
|
||||||
|
|
||||||
// delete greet settings
|
|
||||||
await ctx.GetTable<GreetSettings>()
|
|
||||||
.Where(x => !tempTable.Select(x => x.GuildId)
|
|
||||||
.Contains(x.GuildId))
|
|
||||||
.DeleteAsync();
|
|
||||||
|
|
||||||
// delete sar
|
|
||||||
await ctx.GetTable<SarGroup>()
|
|
||||||
.Where(x => !tempTable.Select(x => x.GuildId)
|
|
||||||
.Contains(x.GuildId))
|
|
||||||
.DeleteAsync();
|
|
||||||
|
|
||||||
// delete warnings
|
|
||||||
await ctx.GetTable<Warning>()
|
|
||||||
.Where(x => !tempTable.Select(x => x.GuildId)
|
|
||||||
.Contains(x.GuildId))
|
|
||||||
.DeleteAsync();
|
|
||||||
|
|
||||||
// delete warn punishments
|
|
||||||
await ctx.GetTable<WarningPunishment>()
|
|
||||||
.Where(x => !tempTable.Select(x => x.GuildId)
|
|
||||||
.Contains(x.GuildId))
|
|
||||||
.DeleteAsync();
|
|
||||||
|
|
||||||
// delete sticky roles
|
|
||||||
await ctx.GetTable<StickyRole>()
|
|
||||||
.Where(x => !tempTable.Select(x => x.GuildId)
|
|
||||||
.Contains(x.GuildId))
|
|
||||||
.DeleteAsync();
|
|
||||||
|
|
||||||
// delete at channels
|
|
||||||
await ctx.GetTable<AutoTranslateChannel>()
|
|
||||||
.Where(x => !tempTable.Select(x => x.GuildId)
|
|
||||||
.Contains(x.GuildId))
|
|
||||||
.DeleteAsync();
|
|
||||||
|
|
||||||
// delete ban templates
|
|
||||||
await ctx.GetTable<BanTemplate>()
|
|
||||||
.Where(x => !tempTable.Select(x => x.GuildId)
|
|
||||||
.Contains(x.GuildId))
|
|
||||||
.DeleteAsync();
|
|
||||||
|
|
||||||
// delete reminders
|
|
||||||
await ctx.GetTable<Reminder>()
|
|
||||||
.Where(x => !tempTable.Select(x => x.GuildId)
|
|
||||||
.Contains(x.ServerId))
|
|
||||||
.DeleteAsync();
|
|
||||||
|
|
||||||
// delete button roles
|
|
||||||
await ctx.GetTable<ButtonRole>()
|
|
||||||
.Where(x => !tempTable.Select(x => x.GuildId)
|
|
||||||
.Contains(x.GuildId))
|
|
||||||
.DeleteAsync();
|
|
||||||
|
|
||||||
return new()
|
return new()
|
||||||
{
|
{
|
||||||
GuildCount = guildIds.Keys.Count,
|
GuildCount = guildIds.Keys.Count,
|
||||||
|
|
|
@ -42,9 +42,9 @@ public partial class Administration
|
||||||
.Page((items, _) =>
|
.Page((items, _) =>
|
||||||
{
|
{
|
||||||
if (!items.Any())
|
if (!items.Any())
|
||||||
return CreateEmbed().WithErrorColor().WithFooter(sql).WithDescription("-");
|
return _sender.CreateEmbed().WithErrorColor().WithFooter(sql).WithDescription("-");
|
||||||
|
|
||||||
return CreateEmbed()
|
return _sender.CreateEmbed()
|
||||||
.WithOkColor()
|
.WithOkColor()
|
||||||
.WithFooter(sql)
|
.WithFooter(sql)
|
||||||
.WithTitle(string.Join(" ║ ", result.ColumnNames))
|
.WithTitle(string.Join(" ║ ", result.ColumnNames))
|
||||||
|
@ -99,7 +99,7 @@ public partial class Administration
|
||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
var embed = CreateEmbed()
|
var embed = _sender.CreateEmbed()
|
||||||
.WithTitle(GetText(strs.sql_confirm_exec))
|
.WithTitle(GetText(strs.sql_confirm_exec))
|
||||||
.WithDescription(Format.Code(sql));
|
.WithDescription(Format.Code(sql));
|
||||||
|
|
||||||
|
@ -119,7 +119,7 @@ public partial class Administration
|
||||||
[OwnerOnly]
|
[OwnerOnly]
|
||||||
public async Task PurgeUser(ulong userId)
|
public async Task PurgeUser(ulong userId)
|
||||||
{
|
{
|
||||||
var embed = CreateEmbed()
|
var embed = _sender.CreateEmbed()
|
||||||
.WithDescription(GetText(strs.purge_user_confirm(Format.Bold(userId.ToString()))));
|
.WithDescription(GetText(strs.purge_user_confirm(Format.Bold(userId.ToString()))));
|
||||||
|
|
||||||
if (!await PromptUserConfirmAsync(embed))
|
if (!await PromptUserConfirmAsync(embed))
|
||||||
|
@ -139,6 +139,7 @@ public partial class Administration
|
||||||
public Task DeleteXp()
|
public Task DeleteXp()
|
||||||
=> ConfirmActionInternalAsync("Delete Xp", () => _xcs.DeleteXp());
|
=> ConfirmActionInternalAsync("Delete Xp", () => _xcs.DeleteXp());
|
||||||
|
|
||||||
|
|
||||||
[Cmd]
|
[Cmd]
|
||||||
[OwnerOnly]
|
[OwnerOnly]
|
||||||
public Task DeleteWaifus()
|
public Task DeleteWaifus()
|
||||||
|
|
|
@ -200,7 +200,9 @@ public partial class Administration
|
||||||
|
|
||||||
if (!isEnabled)
|
if (!isEnabled)
|
||||||
{
|
{
|
||||||
await SendGreetEnableHint(type);
|
var cmdName = GetCmdName(type);
|
||||||
|
|
||||||
|
await Response().Pending(strs.boostmsg_enable($"`{prefix}{cmdName}`")).SendAsync();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -224,24 +226,18 @@ public partial class Administration
|
||||||
await _service.Test(ctx.Guild.Id, type, (ITextChannel)ctx.Channel, user);
|
await _service.Test(ctx.Guild.Id, type, (ITextChannel)ctx.Channel, user);
|
||||||
var conf = await _service.GetGreetSettingsAsync(ctx.Guild.Id, type);
|
var conf = await _service.GetGreetSettingsAsync(ctx.Guild.Id, type);
|
||||||
|
|
||||||
if (conf?.IsEnabled is not true)
|
|
||||||
await SendGreetEnableHint(type);
|
|
||||||
}
|
|
||||||
|
|
||||||
private async Task SendGreetEnableHint(GreetType type)
|
|
||||||
{
|
|
||||||
var cmd = $"`{prefix}{GetCmdName(type)}`";
|
var cmd = $"`{prefix}{GetCmdName(type)}`";
|
||||||
|
|
||||||
var str = type switch
|
var str = type switch
|
||||||
{
|
{
|
||||||
GreetType.Greet => strs.greetmsg_enable(cmd),
|
GreetType.Greet => strs.boostmsg_enable(cmd),
|
||||||
GreetType.Bye => strs.byemsg_enable(cmd),
|
GreetType.Bye => strs.greetmsg_enable(cmd),
|
||||||
GreetType.Boost => strs.boostmsg_enable(cmd),
|
GreetType.Boost => strs.byemsg_enable(cmd),
|
||||||
GreetType.GreetDm => strs.greetdmmsg_enable(cmd),
|
GreetType.GreetDm => strs.greetdmmsg_enable(cmd),
|
||||||
_ => strs.error
|
_ => strs.error
|
||||||
};
|
};
|
||||||
|
if (conf?.IsEnabled is not true)
|
||||||
await Response().Pending(str).SendAsync();
|
await Response().Pending(str).SendAsync();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -208,11 +208,11 @@ public class GreetService : IEService, IReadyExecutor
|
||||||
return Task.CompletedTask;
|
return Task.CompletedTask;
|
||||||
}
|
}
|
||||||
|
|
||||||
private TypedKey<GreetSettings?> GreetSettingsKey(ulong gid, GreetType type)
|
private TypedKey<GreetSettings?> GreetSettingsKey(GreetType type)
|
||||||
=> new($"greet_settings:{gid}:{type}");
|
=> new($"greet_settings:{type}");
|
||||||
|
|
||||||
public async Task<GreetSettings?> GetGreetSettingsAsync(ulong gid, GreetType type)
|
public async Task<GreetSettings?> GetGreetSettingsAsync(ulong gid, GreetType type)
|
||||||
=> await _cache.GetOrAddAsync<GreetSettings?>(GreetSettingsKey(gid, type),
|
=> await _cache.GetOrAddAsync<GreetSettings?>(GreetSettingsKey(type),
|
||||||
() => InternalGetGreetSettingsAsync(gid, type),
|
() => InternalGetGreetSettingsAsync(gid, type),
|
||||||
TimeSpan.FromSeconds(3));
|
TimeSpan.FromSeconds(3));
|
||||||
|
|
||||||
|
@ -339,7 +339,7 @@ public class GreetService : IEService, IReadyExecutor
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
Log.Warning(ex, "Unable to send Greet DM. Probably the user has closed DMs");
|
Log.Error(ex, "Error sending greet dm");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -71,7 +71,7 @@ public sealed class HoneyPotService : IHoneyPotService, IReadyExecutor, IExecNoC
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
Log.Information("Honeypot caught user {User} [{UserId}]", user, user.Id);
|
Log.Information("Honeypot caught user {User} [{UserId}]", user, user.Id);
|
||||||
await user.BanAsync(pruneDays: 1, reason: "Honeypot");
|
await user.BanAsync(pruneDays: 1);
|
||||||
await user.Guild.RemoveBanAsync(user.Id);
|
await user.Guild.RemoveBanAsync(user.Id);
|
||||||
}
|
}
|
||||||
catch (Exception e)
|
catch (Exception e)
|
||||||
|
|
|
@ -123,7 +123,7 @@ public partial class Administration
|
||||||
|
|
||||||
[Cmd]
|
[Cmd]
|
||||||
public async Task LanguagesList()
|
public async Task LanguagesList()
|
||||||
=> await Response().Embed(CreateEmbed()
|
=> await Response().Embed(_sender.CreateEmbed()
|
||||||
.WithOkColor()
|
.WithOkColor()
|
||||||
.WithTitle(GetText(strs.lang_list))
|
.WithTitle(GetText(strs.lang_list))
|
||||||
.WithDescription(string.Join("\n",
|
.WithDescription(string.Join("\n",
|
||||||
|
|
|
@ -72,18 +72,18 @@ public partial class Administration
|
||||||
[RequireContext(ContextType.Guild)]
|
[RequireContext(ContextType.Guild)]
|
||||||
[UserPerm(GuildPerm.ManageRoles | GuildPerm.MuteMembers)]
|
[UserPerm(GuildPerm.ManageRoles | GuildPerm.MuteMembers)]
|
||||||
[Priority(1)]
|
[Priority(1)]
|
||||||
public async Task Mute(ParsedTimespan timespan, IGuildUser user, [Leftover] string reason = "")
|
public async Task Mute(StoopidTime time, IGuildUser user, [Leftover] string reason = "")
|
||||||
{
|
{
|
||||||
if (timespan.Time < TimeSpan.FromMinutes(1) || timespan.Time > TimeSpan.FromDays(49))
|
if (time.Time < TimeSpan.FromMinutes(1) || time.Time > TimeSpan.FromDays(49))
|
||||||
return;
|
return;
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
if (!await VerifyMutePermissions((IGuildUser)ctx.User, user))
|
if (!await VerifyMutePermissions((IGuildUser)ctx.User, user))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
await _service.TimedMute(user, ctx.User, timespan.Time, reason: reason);
|
await _service.TimedMute(user, ctx.User, time.Time, reason: reason);
|
||||||
await Response().Confirm(strs.user_muted_time(Format.Bold(user.ToString()),
|
await Response().Confirm(strs.user_muted_time(Format.Bold(user.ToString()),
|
||||||
(int)timespan.Time.TotalMinutes)).SendAsync();
|
(int)time.Time.TotalMinutes)).SendAsync();
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
|
@ -133,18 +133,18 @@ public partial class Administration
|
||||||
[RequireContext(ContextType.Guild)]
|
[RequireContext(ContextType.Guild)]
|
||||||
[UserPerm(GuildPerm.ManageRoles)]
|
[UserPerm(GuildPerm.ManageRoles)]
|
||||||
[Priority(1)]
|
[Priority(1)]
|
||||||
public async Task ChatMute(ParsedTimespan timespan, IGuildUser user, [Leftover] string reason = "")
|
public async Task ChatMute(StoopidTime time, IGuildUser user, [Leftover] string reason = "")
|
||||||
{
|
{
|
||||||
if (timespan.Time < TimeSpan.FromMinutes(1) || timespan.Time > TimeSpan.FromDays(49))
|
if (time.Time < TimeSpan.FromMinutes(1) || time.Time > TimeSpan.FromDays(49))
|
||||||
return;
|
return;
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
if (!await VerifyMutePermissions((IGuildUser)ctx.User, user))
|
if (!await VerifyMutePermissions((IGuildUser)ctx.User, user))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
await _service.TimedMute(user, ctx.User, timespan.Time, MuteType.Chat, reason);
|
await _service.TimedMute(user, ctx.User, time.Time, MuteType.Chat, reason);
|
||||||
await Response().Confirm(strs.user_chat_mute_time(Format.Bold(user.ToString()),
|
await Response().Confirm(strs.user_chat_mute_time(Format.Bold(user.ToString()),
|
||||||
(int)timespan.Time.TotalMinutes)).SendAsync();
|
(int)time.Time.TotalMinutes)).SendAsync();
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
|
@ -193,18 +193,18 @@ public partial class Administration
|
||||||
[RequireContext(ContextType.Guild)]
|
[RequireContext(ContextType.Guild)]
|
||||||
[UserPerm(GuildPerm.MuteMembers)]
|
[UserPerm(GuildPerm.MuteMembers)]
|
||||||
[Priority(1)]
|
[Priority(1)]
|
||||||
public async Task VoiceMute(ParsedTimespan timespan, IGuildUser user, [Leftover] string reason = "")
|
public async Task VoiceMute(StoopidTime time, IGuildUser user, [Leftover] string reason = "")
|
||||||
{
|
{
|
||||||
if (timespan.Time < TimeSpan.FromMinutes(1) || timespan.Time > TimeSpan.FromDays(49))
|
if (time.Time < TimeSpan.FromMinutes(1) || time.Time > TimeSpan.FromDays(49))
|
||||||
return;
|
return;
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
if (!await VerifyMutePermissions((IGuildUser)ctx.User, user))
|
if (!await VerifyMutePermissions((IGuildUser)ctx.User, user))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
await _service.TimedMute(user, ctx.User, timespan.Time, MuteType.Voice, reason);
|
await _service.TimedMute(user, ctx.User, time.Time, MuteType.Voice, reason);
|
||||||
await Response().Confirm(strs.user_voice_mute_time(Format.Bold(user.ToString()),
|
await Response().Confirm(strs.user_voice_mute_time(Format.Bold(user.ToString()),
|
||||||
(int)timespan.Time.TotalMinutes)).SendAsync();
|
(int)time.Time.TotalMinutes)).SendAsync();
|
||||||
}
|
}
|
||||||
catch
|
catch
|
||||||
{
|
{
|
||||||
|
|
|
@ -122,7 +122,7 @@ public class MuteService : IEService
|
||||||
return;
|
return;
|
||||||
|
|
||||||
_ = Task.Run(() => _sender.Response(user)
|
_ = Task.Run(() => _sender.Response(user)
|
||||||
.Embed(_sender.CreateEmbed(user?.GuildId)
|
.Embed(_sender.CreateEmbed()
|
||||||
.WithDescription($"You've been muted in {user.Guild} server")
|
.WithDescription($"You've been muted in {user.Guild} server")
|
||||||
.AddField("Mute Type", type.ToString())
|
.AddField("Mute Type", type.ToString())
|
||||||
.AddField("Moderator", mod.ToString())
|
.AddField("Moderator", mod.ToString())
|
||||||
|
@ -140,7 +140,7 @@ public class MuteService : IEService
|
||||||
return;
|
return;
|
||||||
|
|
||||||
_ = Task.Run(() => _sender.Response(user)
|
_ = Task.Run(() => _sender.Response(user)
|
||||||
.Embed(_sender.CreateEmbed(user.GuildId)
|
.Embed(_sender.CreateEmbed()
|
||||||
.WithDescription($"You've been unmuted in {user.Guild} server")
|
.WithDescription($"You've been unmuted in {user.Guild} server")
|
||||||
.AddField("Unmute Type", type.ToString())
|
.AddField("Unmute Type", type.ToString())
|
||||||
.AddField("Moderator", mod.ToString())
|
.AddField("Moderator", mod.ToString())
|
||||||
|
|
|
@ -1,23 +0,0 @@
|
||||||
using EllieBot.Db.Models;
|
|
||||||
using System.Collections;
|
|
||||||
|
|
||||||
namespace EllieBot.Modules.Administration;
|
|
||||||
|
|
||||||
public interface INotifyModel
|
|
||||||
{
|
|
||||||
static abstract string KeyName { get; }
|
|
||||||
static abstract NotifyType NotifyType { get; }
|
|
||||||
IReadOnlyDictionary<string, Func<SocketGuild, string>> GetReplacements();
|
|
||||||
|
|
||||||
public virtual bool TryGetGuildId(out ulong guildId)
|
|
||||||
{
|
|
||||||
guildId = 0;
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
public virtual bool TryGetUserId(out ulong userId)
|
|
||||||
{
|
|
||||||
userId = 0;
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,7 +0,0 @@
|
||||||
namespace EllieBot.Modules.Administration;
|
|
||||||
|
|
||||||
public interface INotifySubscriber
|
|
||||||
{
|
|
||||||
Task NotifyAsync<T>(T data, bool isShardLocal = false)
|
|
||||||
where T : struct, INotifyModel;
|
|
||||||
}
|
|
|
@ -1,36 +0,0 @@
|
||||||
using EllieBot.Db.Models;
|
|
||||||
using EllieBot.Modules.Administration;
|
|
||||||
|
|
||||||
namespace EllieBot.Modules.Xp.Services;
|
|
||||||
|
|
||||||
public record struct AddRoleRewardNotifyModel(ulong GuildId, ulong RoleId, ulong UserId, long Level) : INotifyModel
|
|
||||||
{
|
|
||||||
public static string KeyName
|
|
||||||
=> "notify.reward.addrole";
|
|
||||||
|
|
||||||
public static NotifyType NotifyType
|
|
||||||
=> NotifyType.AddRoleReward;
|
|
||||||
|
|
||||||
public IReadOnlyDictionary<string, Func<SocketGuild, string>> GetReplacements()
|
|
||||||
{
|
|
||||||
var model = this;
|
|
||||||
return new Dictionary<string, Func<SocketGuild, string>>()
|
|
||||||
{
|
|
||||||
{ "%event.user%", g => g.GetUser(model.UserId)?.ToString() ?? model.UserId.ToString() },
|
|
||||||
{ "%event.role%", g => g.GetRole(model.RoleId)?.ToString() ?? model.RoleId.ToString() },
|
|
||||||
{ "%event.level%", g => model.Level.ToString() }
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
public bool TryGetUserId(out ulong userId)
|
|
||||||
{
|
|
||||||
userId = UserId;
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
public bool TryGetGuildId(out ulong guildId)
|
|
||||||
{
|
|
||||||
guildId = GuildId;
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,38 +0,0 @@
|
||||||
using EllieBot.Db.Models;
|
|
||||||
|
|
||||||
namespace EllieBot.Modules.Administration;
|
|
||||||
|
|
||||||
public record struct LevelUpNotifyModel(
|
|
||||||
ulong GuildId,
|
|
||||||
ulong ChannelId,
|
|
||||||
ulong UserId,
|
|
||||||
long Level) : INotifyModel
|
|
||||||
{
|
|
||||||
public static string KeyName
|
|
||||||
=> "notify.levelup";
|
|
||||||
|
|
||||||
public static NotifyType NotifyType
|
|
||||||
=> NotifyType.LevelUp;
|
|
||||||
|
|
||||||
public IReadOnlyDictionary<string, Func<SocketGuild, string>> GetReplacements()
|
|
||||||
{
|
|
||||||
var data = this;
|
|
||||||
return new Dictionary<string, Func<SocketGuild, string>>()
|
|
||||||
{
|
|
||||||
{ "%event.level%", g => data.Level.ToString() },
|
|
||||||
{ "%event.user%", g => g.GetUser(data.UserId)?.ToString() ?? data.UserId.ToString() },
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
public bool TryGetGuildId(out ulong guildId)
|
|
||||||
{
|
|
||||||
guildId = GuildId;
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
public bool TryGetUserId(out ulong userId)
|
|
||||||
{
|
|
||||||
userId = UserId;
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,34 +0,0 @@
|
||||||
#nullable disable
|
|
||||||
using EllieBot.Db.Models;
|
|
||||||
|
|
||||||
namespace EllieBot.Modules.Administration.Services;
|
|
||||||
|
|
||||||
public record struct ProtectionNotifyModel(ulong GuildId, ProtectionType ProtType, ulong UserId) : INotifyModel
|
|
||||||
{
|
|
||||||
public static string KeyName
|
|
||||||
=> "notify.protection";
|
|
||||||
|
|
||||||
public static NotifyType NotifyType
|
|
||||||
=> NotifyType.Protection;
|
|
||||||
|
|
||||||
public IReadOnlyDictionary<string, Func<SocketGuild, string>> GetReplacements()
|
|
||||||
{
|
|
||||||
var data = this;
|
|
||||||
return new Dictionary<string, Func<SocketGuild, string>>()
|
|
||||||
{
|
|
||||||
{ "%event.type%", g => data.ProtType.ToString() },
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
public bool TryGetUserId(out ulong userId)
|
|
||||||
{
|
|
||||||
userId = UserId;
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
public bool TryGetGuildId(out ulong guildId)
|
|
||||||
{
|
|
||||||
guildId = GuildId;
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,36 +0,0 @@
|
||||||
using EllieBot.Db.Models;
|
|
||||||
using EllieBot.Modules.Administration;
|
|
||||||
|
|
||||||
namespace EllieBot.Modules.Xp.Services;
|
|
||||||
|
|
||||||
public record struct RemoveRoleRewardNotifyModel(ulong GuildId, ulong RoleId, ulong UserId, long Level) : INotifyModel
|
|
||||||
{
|
|
||||||
public static string KeyName
|
|
||||||
=> "notify.reward.removerole";
|
|
||||||
|
|
||||||
public static NotifyType NotifyType
|
|
||||||
=> NotifyType.RemoveRoleReward;
|
|
||||||
|
|
||||||
public IReadOnlyDictionary<string, Func<SocketGuild, string>> GetReplacements()
|
|
||||||
{
|
|
||||||
var model = this;
|
|
||||||
return new Dictionary<string, Func<SocketGuild, string>>()
|
|
||||||
{
|
|
||||||
{ "%event.user%", g => g.GetUser(model.UserId)?.ToString() ?? model.UserId.ToString() },
|
|
||||||
{ "%event.role%", g => g.GetRole(model.RoleId)?.ToString() ?? model.RoleId.ToString() },
|
|
||||||
{ "%event.level%", g => model.Level.ToString() },
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
public bool TryGetUserId(out ulong userId)
|
|
||||||
{
|
|
||||||
userId = UserId;
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
public bool TryGetGuildId(out ulong guildId)
|
|
||||||
{
|
|
||||||
guildId = GuildId;
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,114 +0,0 @@
|
||||||
using EllieBot.Db.Models;
|
|
||||||
using System.Text;
|
|
||||||
|
|
||||||
namespace EllieBot.Modules.Administration;
|
|
||||||
|
|
||||||
public partial class Administration
|
|
||||||
{
|
|
||||||
public class NotifyCommands : EllieModule<NotifyService>
|
|
||||||
{
|
|
||||||
[Cmd]
|
|
||||||
[UserPerm(GuildPerm.Administrator)]
|
|
||||||
public async Task Notify()
|
|
||||||
{
|
|
||||||
await Response()
|
|
||||||
.Paginated()
|
|
||||||
.Items(Enum.GetValues<NotifyType>())
|
|
||||||
.PageSize(5)
|
|
||||||
.Page((items, page) =>
|
|
||||||
{
|
|
||||||
var eb = CreateEmbed()
|
|
||||||
.WithOkColor()
|
|
||||||
.WithTitle(GetText(strs.notify_available));
|
|
||||||
|
|
||||||
foreach (var item in items)
|
|
||||||
{
|
|
||||||
eb.AddField(item.ToString(), GetText(GetDescription(item)), false);
|
|
||||||
}
|
|
||||||
|
|
||||||
return eb;
|
|
||||||
})
|
|
||||||
.SendAsync();
|
|
||||||
}
|
|
||||||
|
|
||||||
private LocStr GetDescription(NotifyType item)
|
|
||||||
=> item switch
|
|
||||||
{
|
|
||||||
NotifyType.LevelUp => strs.notify_desc_levelup,
|
|
||||||
NotifyType.Protection => strs.notify_desc_protection,
|
|
||||||
NotifyType.AddRoleReward => strs.notify_desc_addrolerew,
|
|
||||||
NotifyType.RemoveRoleReward => strs.notify_desc_removerolerew,
|
|
||||||
_ => strs.notify_desc_not_found
|
|
||||||
};
|
|
||||||
|
|
||||||
[Cmd]
|
|
||||||
[UserPerm(GuildPerm.Administrator)]
|
|
||||||
public async Task Notify(NotifyType nType, [Leftover] string? message = null)
|
|
||||||
{
|
|
||||||
if (string.IsNullOrWhiteSpace(message))
|
|
||||||
{
|
|
||||||
// show msg
|
|
||||||
var conf = await _service.GetNotifyAsync(ctx.Guild.Id, nType);
|
|
||||||
if (conf is null)
|
|
||||||
{
|
|
||||||
await Response().Confirm(strs.notify_msg_not_set).SendAsync();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
var eb = CreateEmbed()
|
|
||||||
.WithOkColor()
|
|
||||||
.WithTitle(GetText(strs.notify_msg))
|
|
||||||
.WithDescription(conf.Message.TrimTo(2048))
|
|
||||||
.AddField(GetText(strs.notify_type), conf.Type.ToString(), true)
|
|
||||||
.AddField(GetText(strs.channel),
|
|
||||||
$"""
|
|
||||||
<#{conf.ChannelId}>
|
|
||||||
`{conf.ChannelId}`
|
|
||||||
""",
|
|
||||||
true);
|
|
||||||
|
|
||||||
await Response().Embed(eb).SendAsync();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
await _service.EnableAsync(ctx.Guild.Id, ctx.Channel.Id, nType, message);
|
|
||||||
await Response().Confirm(strs.notify_on($"<#{ctx.Channel.Id}>", Format.Bold(nType.ToString()))).SendAsync();
|
|
||||||
}
|
|
||||||
|
|
||||||
[Cmd]
|
|
||||||
[UserPerm(GuildPerm.Administrator)]
|
|
||||||
public async Task NotifyList(int page = 1)
|
|
||||||
{
|
|
||||||
if (--page < 0)
|
|
||||||
return;
|
|
||||||
|
|
||||||
var notifs = await _service.GetForGuildAsync(ctx.Guild.Id);
|
|
||||||
|
|
||||||
var sb = new StringBuilder();
|
|
||||||
|
|
||||||
foreach (var notif in notifs)
|
|
||||||
{
|
|
||||||
sb.AppendLine($"""
|
|
||||||
- **{notif.Type}**
|
|
||||||
<#{notif.ChannelId}> `{notif.ChannelId}`
|
|
||||||
|
|
||||||
""");
|
|
||||||
}
|
|
||||||
|
|
||||||
if (notifs.Count == 0)
|
|
||||||
sb.AppendLine(GetText(strs.notify_none));
|
|
||||||
|
|
||||||
await Response()
|
|
||||||
.Confirm(GetText(strs.notify_list), text: sb.ToString())
|
|
||||||
.SendAsync();
|
|
||||||
}
|
|
||||||
|
|
||||||
[Cmd]
|
|
||||||
[UserPerm(GuildPerm.Administrator)]
|
|
||||||
public async Task NotifyClear(NotifyType nType)
|
|
||||||
{
|
|
||||||
await _service.DisableAsync(ctx.Guild.Id, nType);
|
|
||||||
await Response().Confirm(strs.notify_off(nType)).SendAsync();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,6 +0,0 @@
|
||||||
namespace EllieBot.Modules.Administration;
|
|
||||||
|
|
||||||
public static class NotifyKeys
|
|
||||||
{
|
|
||||||
public static TypedKey<LevelUpNotifyModel> LevelUp { get; } = new("notify:levelup");
|
|
||||||
}
|
|
|
@ -1,8 +0,0 @@
|
||||||
namespace EllieBot.Modules.Administration;
|
|
||||||
|
|
||||||
public static class NotifyModelExtensions
|
|
||||||
{
|
|
||||||
public static TypedKey<T> GetTypedKey<T>(this T model)
|
|
||||||
where T : struct, INotifyModel
|
|
||||||
=> new(T.KeyName);
|
|
||||||
}
|
|
|
@ -1,227 +0,0 @@
|
||||||
using LinqToDB;
|
|
||||||
using LinqToDB.EntityFrameworkCore;
|
|
||||||
using EllieBot.Common.ModuleBehaviors;
|
|
||||||
using EllieBot.Db.Models;
|
|
||||||
using EllieBot.Generators;
|
|
||||||
|
|
||||||
namespace EllieBot.Modules.Administration;
|
|
||||||
|
|
||||||
public sealed class NotifyService : IReadyExecutor, INotifySubscriber, IEService
|
|
||||||
{
|
|
||||||
private readonly DbService _db;
|
|
||||||
private readonly IMessageSenderService _mss;
|
|
||||||
private readonly DiscordSocketClient _client;
|
|
||||||
private readonly IBotCreds _creds;
|
|
||||||
private readonly IReplacementService _repSvc;
|
|
||||||
private readonly IPubSub _pubSub;
|
|
||||||
private ConcurrentDictionary<NotifyType, ConcurrentDictionary<ulong, Notify>> _events = new();
|
|
||||||
|
|
||||||
public NotifyService(
|
|
||||||
DbService db,
|
|
||||||
IMessageSenderService mss,
|
|
||||||
DiscordSocketClient client,
|
|
||||||
IBotCreds creds,
|
|
||||||
IReplacementService repSvc,
|
|
||||||
IPubSub pubSub)
|
|
||||||
{
|
|
||||||
_db = db;
|
|
||||||
_mss = mss;
|
|
||||||
_client = client;
|
|
||||||
_creds = creds;
|
|
||||||
_repSvc = repSvc;
|
|
||||||
_pubSub = pubSub;
|
|
||||||
}
|
|
||||||
|
|
||||||
public async Task OnReadyAsync()
|
|
||||||
{
|
|
||||||
await using var uow = _db.GetDbContext();
|
|
||||||
_events = (await uow.GetTable<Notify>()
|
|
||||||
.Where(x => Linq2DbExpressions.GuildOnShard(x.GuildId,
|
|
||||||
_creds.TotalShards,
|
|
||||||
_client.ShardId))
|
|
||||||
.ToListAsyncLinqToDB())
|
|
||||||
.GroupBy(x => x.Type)
|
|
||||||
.ToDictionary(x => x.Key, x => x.ToDictionary(x => x.GuildId).ToConcurrent())
|
|
||||||
.ToConcurrent();
|
|
||||||
|
|
||||||
|
|
||||||
await SubscribeToEvent<LevelUpNotifyModel>();
|
|
||||||
}
|
|
||||||
|
|
||||||
private async Task SubscribeToEvent<T>()
|
|
||||||
where T : struct, INotifyModel
|
|
||||||
{
|
|
||||||
await _pubSub.Sub(new TypedKey<T>(T.KeyName), async (model) => await OnEvent(model));
|
|
||||||
}
|
|
||||||
|
|
||||||
public async Task NotifyAsync<T>(T data, bool isShardLocal = false)
|
|
||||||
where T : struct, INotifyModel
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
if (isShardLocal)
|
|
||||||
{
|
|
||||||
await OnEvent(data);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
await _pubSub.Pub(data.GetTypedKey(), data);
|
|
||||||
}
|
|
||||||
catch (Exception ex)
|
|
||||||
{
|
|
||||||
Log.Warning(ex,
|
|
||||||
"Unknown error occurred while trying to triger {NotifyEvent} for {NotifyModel}",
|
|
||||||
T.KeyName,
|
|
||||||
data);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private async Task OnEvent<T>(T model)
|
|
||||||
where T : struct, INotifyModel
|
|
||||||
{
|
|
||||||
if (_events.TryGetValue(T.NotifyType, out var subs))
|
|
||||||
{
|
|
||||||
if (model.TryGetGuildId(out var gid))
|
|
||||||
{
|
|
||||||
if (!subs.TryGetValue(gid, out var conf))
|
|
||||||
return;
|
|
||||||
|
|
||||||
await HandleNotifyEvent(conf, model);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
foreach (var key in subs.Keys.ToArray())
|
|
||||||
{
|
|
||||||
if (subs.TryGetValue(key, out var notif))
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
await HandleNotifyEvent(notif, model);
|
|
||||||
}
|
|
||||||
catch (Exception ex)
|
|
||||||
{
|
|
||||||
Log.Error(ex,
|
|
||||||
"Error occured while sending notification {NotifyEvent} to guild {GuildId}: {ErrorMessage}",
|
|
||||||
T.NotifyType,
|
|
||||||
key,
|
|
||||||
ex.Message);
|
|
||||||
}
|
|
||||||
|
|
||||||
await Task.Delay(500);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private async Task HandleNotifyEvent(Notify conf, INotifyModel model)
|
|
||||||
{
|
|
||||||
var guild = _client.GetGuild(conf.GuildId);
|
|
||||||
var channel = guild?.GetTextChannel(conf.ChannelId);
|
|
||||||
|
|
||||||
if (guild is null || channel is null)
|
|
||||||
return;
|
|
||||||
|
|
||||||
IUser? user = null;
|
|
||||||
if (model.TryGetUserId(out var userId))
|
|
||||||
{
|
|
||||||
user = guild.GetUser(userId) ?? _client.GetUser(userId);
|
|
||||||
}
|
|
||||||
|
|
||||||
var rctx = new ReplacementContext(guild: guild, channel: channel, user: user);
|
|
||||||
|
|
||||||
var st = SmartText.CreateFrom(conf.Message);
|
|
||||||
foreach (var modelRep in model.GetReplacements())
|
|
||||||
{
|
|
||||||
rctx.WithOverride(modelRep.Key, () => modelRep.Value(guild));
|
|
||||||
}
|
|
||||||
|
|
||||||
st = await _repSvc.ReplaceAsync(st, rctx);
|
|
||||||
if (st is SmartPlainText spt)
|
|
||||||
{
|
|
||||||
await _mss.Response(channel)
|
|
||||||
.Confirm(spt.Text)
|
|
||||||
.SendAsync();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
await _mss.Response(channel)
|
|
||||||
.Text(st)
|
|
||||||
.Sanitize(false)
|
|
||||||
.SendAsync();
|
|
||||||
}
|
|
||||||
|
|
||||||
public async Task EnableAsync(
|
|
||||||
ulong guildId,
|
|
||||||
ulong channelId,
|
|
||||||
NotifyType nType,
|
|
||||||
string message)
|
|
||||||
{
|
|
||||||
await using var uow = _db.GetDbContext();
|
|
||||||
await uow.GetTable<Notify>()
|
|
||||||
.InsertOrUpdateAsync(() => new()
|
|
||||||
{
|
|
||||||
GuildId = guildId,
|
|
||||||
ChannelId = channelId,
|
|
||||||
Type = nType,
|
|
||||||
Message = message,
|
|
||||||
},
|
|
||||||
(_) => new()
|
|
||||||
{
|
|
||||||
Message = message,
|
|
||||||
ChannelId = channelId
|
|
||||||
},
|
|
||||||
() => new()
|
|
||||||
{
|
|
||||||
GuildId = guildId,
|
|
||||||
Type = nType
|
|
||||||
});
|
|
||||||
|
|
||||||
var eventDict = _events.GetOrAdd(nType, _ => new());
|
|
||||||
eventDict[guildId] = new()
|
|
||||||
{
|
|
||||||
GuildId = guildId,
|
|
||||||
ChannelId = channelId,
|
|
||||||
Type = nType,
|
|
||||||
Message = message
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
public async Task DisableAsync(ulong guildId, NotifyType nType)
|
|
||||||
{
|
|
||||||
await using var uow = _db.GetDbContext();
|
|
||||||
var deleted = await uow.GetTable<Notify>()
|
|
||||||
.Where(x => x.GuildId == guildId && x.Type == nType)
|
|
||||||
.DeleteAsync();
|
|
||||||
|
|
||||||
if (deleted == 0)
|
|
||||||
return;
|
|
||||||
|
|
||||||
if (!_events.TryGetValue(nType, out var guildsDict))
|
|
||||||
return;
|
|
||||||
|
|
||||||
guildsDict.TryRemove(guildId, out _);
|
|
||||||
}
|
|
||||||
|
|
||||||
public async Task<IReadOnlyCollection<Notify>> GetForGuildAsync(ulong guildId, int page = 0)
|
|
||||||
{
|
|
||||||
ArgumentOutOfRangeException.ThrowIfNegative(page);
|
|
||||||
|
|
||||||
await using var ctx = _db.GetDbContext();
|
|
||||||
var list = await ctx.GetTable<Notify>()
|
|
||||||
.Where(x => x.GuildId == guildId)
|
|
||||||
.OrderBy(x => x.Type)
|
|
||||||
.Skip(page * 10)
|
|
||||||
.Take(10)
|
|
||||||
.ToListAsyncLinqToDB();
|
|
||||||
|
|
||||||
return list;
|
|
||||||
}
|
|
||||||
|
|
||||||
public async Task<Notify?> GetNotifyAsync(ulong guildId, NotifyType nType)
|
|
||||||
{
|
|
||||||
await using var ctx = _db.GetDbContext();
|
|
||||||
return await ctx.GetTable<Notify>()
|
|
||||||
.Where(x => x.GuildId == guildId && x.Type == nType)
|
|
||||||
.FirstOrDefaultAsyncLinqToDB();
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -36,7 +36,7 @@ public partial class Administration
|
||||||
[UserPerm(GuildPerm.Administrator)]
|
[UserPerm(GuildPerm.Administrator)]
|
||||||
public async Task DiscordPermOverrideReset()
|
public async Task DiscordPermOverrideReset()
|
||||||
{
|
{
|
||||||
var result = await PromptUserConfirmAsync(CreateEmbed()
|
var result = await PromptUserConfirmAsync(_sender.CreateEmbed()
|
||||||
.WithOkColor()
|
.WithOkColor()
|
||||||
.WithDescription(GetText(strs.perm_override_all_confirm)));
|
.WithDescription(GetText(strs.perm_override_all_confirm)));
|
||||||
|
|
||||||
|
@ -65,7 +65,7 @@ public partial class Administration
|
||||||
.CurrentPage(page)
|
.CurrentPage(page)
|
||||||
.Page((items, _) =>
|
.Page((items, _) =>
|
||||||
{
|
{
|
||||||
var eb = CreateEmbed().WithTitle(GetText(strs.perm_overrides)).WithOkColor();
|
var eb = _sender.CreateEmbed().WithTitle(GetText(strs.perm_overrides)).WithOkColor();
|
||||||
|
|
||||||
if (items.Count == 0)
|
if (items.Count == 0)
|
||||||
eb.WithDescription(GetText(strs.perm_override_page_none));
|
eb.WithDescription(GetText(strs.perm_override_page_none));
|
||||||
|
|
|
@ -6,7 +6,7 @@ namespace EllieBot.Modules.Administration;
|
||||||
public partial class Administration
|
public partial class Administration
|
||||||
{
|
{
|
||||||
[Group]
|
[Group]
|
||||||
public partial class PlayingRotateCommands : EllieModule<IBotActivityService>
|
public partial class PlayingRotateCommands : EllieModule<PlayingRotateService>
|
||||||
{
|
{
|
||||||
[Cmd]
|
[Cmd]
|
||||||
[OwnerOnly]
|
[OwnerOnly]
|
||||||
|
|
|
@ -5,134 +5,36 @@ using EllieBot.Db.Models;
|
||||||
|
|
||||||
namespace EllieBot.Modules.Administration.Services;
|
namespace EllieBot.Modules.Administration.Services;
|
||||||
|
|
||||||
public sealed class BotActivityService : IBotActivityService, IReadyExecutor, IEService
|
public sealed class PlayingRotateService : IEService, IReadyExecutor
|
||||||
{
|
{
|
||||||
private readonly TypedKey<ActivityPubData> _activitySetKey = new("activity.set");
|
|
||||||
|
|
||||||
private readonly IPubSub _pubSub;
|
|
||||||
private readonly DiscordSocketClient _client;
|
|
||||||
private readonly DbService _db;
|
|
||||||
private readonly IReplacementService _rep;
|
|
||||||
private readonly BotConfigService _bss;
|
private readonly BotConfigService _bss;
|
||||||
|
private readonly SelfService _selfService;
|
||||||
|
private readonly IReplacementService _repService;
|
||||||
|
// private readonly Replacer _rep;
|
||||||
|
private readonly DbService _db;
|
||||||
|
private readonly DiscordSocketClient _client;
|
||||||
|
|
||||||
public BotActivityService(
|
public PlayingRotateService(
|
||||||
IPubSub pubSub,
|
|
||||||
DiscordSocketClient client,
|
DiscordSocketClient client,
|
||||||
DbService db,
|
DbService db,
|
||||||
IReplacementService rep,
|
BotConfigService bss,
|
||||||
BotConfigService bss)
|
IEnumerable<IPlaceholderProvider> phProviders,
|
||||||
|
SelfService selfService,
|
||||||
|
IReplacementService repService)
|
||||||
{
|
{
|
||||||
_pubSub = pubSub;
|
|
||||||
_client = client;
|
|
||||||
_db = db;
|
_db = db;
|
||||||
_rep = rep;
|
|
||||||
_bss = bss;
|
_bss = bss;
|
||||||
}
|
_selfService = selfService;
|
||||||
|
_repService = repService;
|
||||||
|
_client = client;
|
||||||
|
|
||||||
public async Task<string> RemovePlayingAsync(int index)
|
|
||||||
{
|
|
||||||
ArgumentOutOfRangeException.ThrowIfNegative(index);
|
|
||||||
|
|
||||||
await using var uow = _db.GetDbContext();
|
|
||||||
var toRemove = await uow.Set<RotatingPlayingStatus>()
|
|
||||||
.AsQueryable()
|
|
||||||
.AsNoTracking()
|
|
||||||
.Skip(index)
|
|
||||||
.FirstOrDefaultAsync();
|
|
||||||
|
|
||||||
if (toRemove is null)
|
|
||||||
return null;
|
|
||||||
|
|
||||||
uow.Remove(toRemove);
|
|
||||||
await uow.SaveChangesAsync();
|
|
||||||
return toRemove.Status;
|
|
||||||
}
|
|
||||||
|
|
||||||
public async Task AddPlaying(ActivityType activityType, string status)
|
|
||||||
{
|
|
||||||
await using var uow = _db.GetDbContext();
|
|
||||||
var toAdd = new RotatingPlayingStatus
|
|
||||||
{
|
|
||||||
Status = status,
|
|
||||||
Type = (EllieBot.Db.DbActivityType)activityType
|
|
||||||
};
|
|
||||||
uow.Add(toAdd);
|
|
||||||
await uow.SaveChangesAsync();
|
|
||||||
}
|
|
||||||
|
|
||||||
public void DisableRotatePlaying()
|
|
||||||
{
|
|
||||||
_bss.ModifyConfig(bs => { bs.RotateStatuses = false; });
|
|
||||||
}
|
|
||||||
|
|
||||||
public bool ToggleRotatePlaying()
|
|
||||||
{
|
|
||||||
var enabled = false;
|
|
||||||
_bss.ModifyConfig(bs => { enabled = bs.RotateStatuses = !bs.RotateStatuses; });
|
|
||||||
return enabled;
|
|
||||||
}
|
|
||||||
|
|
||||||
public IReadOnlyList<RotatingPlayingStatus> GetRotatingStatuses()
|
|
||||||
{
|
|
||||||
using var uow = _db.GetDbContext();
|
|
||||||
return uow.Set<RotatingPlayingStatus>().AsNoTracking().ToList();
|
|
||||||
}
|
|
||||||
|
|
||||||
public Task SetActivityAsync(string game, ActivityType? type)
|
|
||||||
=> _pubSub.Pub(_activitySetKey,
|
|
||||||
new()
|
|
||||||
{
|
|
||||||
Name = game,
|
|
||||||
Link = null,
|
|
||||||
Type = type
|
|
||||||
});
|
|
||||||
|
|
||||||
public Task SetStreamAsync(string name, string link)
|
|
||||||
=> _pubSub.Pub(_activitySetKey,
|
|
||||||
new()
|
|
||||||
{
|
|
||||||
Name = name,
|
|
||||||
Link = link,
|
|
||||||
Type = ActivityType.Streaming
|
|
||||||
});
|
|
||||||
|
|
||||||
private sealed class ActivityPubData
|
|
||||||
{
|
|
||||||
public string Name { get; init; }
|
|
||||||
public string Link { get; init; }
|
|
||||||
public ActivityType? Type { get; init; }
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task OnReadyAsync()
|
public async Task OnReadyAsync()
|
||||||
{
|
{
|
||||||
await _pubSub.Sub(_activitySetKey,
|
|
||||||
async data =>
|
|
||||||
{
|
|
||||||
if (_client.ShardId == 0)
|
|
||||||
{
|
|
||||||
DisableRotatePlaying();
|
|
||||||
}
|
|
||||||
|
|
||||||
try
|
|
||||||
{
|
|
||||||
if (data.Type is { } activityType)
|
|
||||||
{
|
|
||||||
await _client.SetGameAsync(data.Name, data.Link, activityType);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
await _client.SetCustomStatusAsync(data.Name);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
catch (Exception ex)
|
|
||||||
{
|
|
||||||
Log.Warning(ex, "Error setting activity");
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
if (_client.ShardId != 0)
|
if (_client.ShardId != 0)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
using var timer = new PeriodicTimer(TimeSpan.FromMinutes(1));
|
using var timer = new PeriodicTimer(TimeSpan.FromMinutes(1));
|
||||||
var index = 0;
|
var index = 0;
|
||||||
while (await timer.WaitForNextTickAsync())
|
while (await timer.WaitForNextTickAsync())
|
||||||
|
@ -155,8 +57,8 @@ public sealed class BotActivityService : IBotActivityService, IReadyExecutor, IE
|
||||||
? rotatingStatuses[index = 0]
|
? rotatingStatuses[index = 0]
|
||||||
: rotatingStatuses[index++];
|
: rotatingStatuses[index++];
|
||||||
|
|
||||||
var statusText = await _rep.ReplaceAsync(playingStatus.Status, new(client: _client));
|
var statusText = await _repService.ReplaceAsync(playingStatus.Status, new (client: _client));
|
||||||
await SetActivityAsync(statusText, (ActivityType)playingStatus.Type);
|
await _selfService.SetActivityAsync(statusText, (ActivityType)playingStatus.Type);
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
|
@ -164,4 +66,44 @@ public sealed class BotActivityService : IBotActivityService, IReadyExecutor, IE
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public async Task<string> RemovePlayingAsync(int index)
|
||||||
|
{
|
||||||
|
ArgumentOutOfRangeException.ThrowIfNegative(index);
|
||||||
|
|
||||||
|
await using var uow = _db.GetDbContext();
|
||||||
|
var toRemove = await uow.Set<RotatingPlayingStatus>().AsQueryable().AsNoTracking().Skip(index).FirstOrDefaultAsync();
|
||||||
|
|
||||||
|
if (toRemove is null)
|
||||||
|
return null;
|
||||||
|
|
||||||
|
uow.Remove(toRemove);
|
||||||
|
await uow.SaveChangesAsync();
|
||||||
|
return toRemove.Status;
|
||||||
|
}
|
||||||
|
|
||||||
|
public async Task AddPlaying(ActivityType activityType, string status)
|
||||||
|
{
|
||||||
|
await using var uow = _db.GetDbContext();
|
||||||
|
var toAdd = new RotatingPlayingStatus
|
||||||
|
{
|
||||||
|
Status = status,
|
||||||
|
Type = (EllieBot.Db.DbActivityType)activityType
|
||||||
|
};
|
||||||
|
uow.Add(toAdd);
|
||||||
|
await uow.SaveChangesAsync();
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool ToggleRotatePlaying()
|
||||||
|
{
|
||||||
|
var enabled = false;
|
||||||
|
_bss.ModifyConfig(bs => { enabled = bs.RotateStatuses = !bs.RotateStatuses; });
|
||||||
|
return enabled;
|
||||||
|
}
|
||||||
|
|
||||||
|
public IReadOnlyList<RotatingPlayingStatus> GetRotatingStatuses()
|
||||||
|
{
|
||||||
|
using var uow = _db.GetDbContext();
|
||||||
|
return uow.Set<RotatingPlayingStatus>().AsNoTracking().ToList();
|
||||||
|
}
|
||||||
}
|
}
|
|
@ -28,17 +28,17 @@ public partial class Administration
|
||||||
[RequireContext(ContextType.Guild)]
|
[RequireContext(ContextType.Guild)]
|
||||||
[UserPerm(GuildPerm.Administrator)]
|
[UserPerm(GuildPerm.Administrator)]
|
||||||
public async Task AntiAlt(
|
public async Task AntiAlt(
|
||||||
ParsedTimespan minAge,
|
StoopidTime minAge,
|
||||||
PunishmentAction action,
|
PunishmentAction action,
|
||||||
[Leftover] ParsedTimespan punishTimespan = null)
|
[Leftover] StoopidTime punishTime = null)
|
||||||
{
|
{
|
||||||
var minAgeMinutes = (int)minAge.Time.TotalMinutes;
|
var minAgeMinutes = (int)minAge.Time.TotalMinutes;
|
||||||
var punishTimeMinutes = (int?)punishTimespan?.Time.TotalMinutes ?? 0;
|
var punishTimeMinutes = (int?)punishTime?.Time.TotalMinutes ?? 0;
|
||||||
|
|
||||||
if (minAgeMinutes < 1 || punishTimeMinutes < 0)
|
if (minAgeMinutes < 1 || punishTimeMinutes < 0)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
var minutes = (int?)punishTimespan?.Time.TotalMinutes ?? 0;
|
var minutes = (int?)punishTime?.Time.TotalMinutes ?? 0;
|
||||||
if (action is PunishmentAction.TimeOut && minutes < 1)
|
if (action is PunishmentAction.TimeOut && minutes < 1)
|
||||||
minutes = 1;
|
minutes = 1;
|
||||||
|
|
||||||
|
@ -53,7 +53,7 @@ public partial class Administration
|
||||||
[Cmd]
|
[Cmd]
|
||||||
[RequireContext(ContextType.Guild)]
|
[RequireContext(ContextType.Guild)]
|
||||||
[UserPerm(GuildPerm.Administrator)]
|
[UserPerm(GuildPerm.Administrator)]
|
||||||
public async Task AntiAlt(ParsedTimespan minAge, PunishmentAction action, [Leftover] IRole role)
|
public async Task AntiAlt(StoopidTime minAge, PunishmentAction action, [Leftover] IRole role)
|
||||||
{
|
{
|
||||||
var minAgeMinutes = (int)minAge.Time.TotalMinutes;
|
var minAgeMinutes = (int)minAge.Time.TotalMinutes;
|
||||||
|
|
||||||
|
@ -86,8 +86,8 @@ public partial class Administration
|
||||||
int userThreshold,
|
int userThreshold,
|
||||||
int seconds,
|
int seconds,
|
||||||
PunishmentAction action,
|
PunishmentAction action,
|
||||||
[Leftover] ParsedTimespan punishTimespan)
|
[Leftover] StoopidTime punishTime)
|
||||||
=> InternalAntiRaid(userThreshold, seconds, action, punishTimespan);
|
=> InternalAntiRaid(userThreshold, seconds, action, punishTime);
|
||||||
|
|
||||||
[Cmd]
|
[Cmd]
|
||||||
[RequireContext(ContextType.Guild)]
|
[RequireContext(ContextType.Guild)]
|
||||||
|
@ -100,7 +100,7 @@ public partial class Administration
|
||||||
int userThreshold,
|
int userThreshold,
|
||||||
int seconds = 10,
|
int seconds = 10,
|
||||||
PunishmentAction action = PunishmentAction.Mute,
|
PunishmentAction action = PunishmentAction.Mute,
|
||||||
ParsedTimespan punishTimespan = null)
|
StoopidTime punishTime = null)
|
||||||
{
|
{
|
||||||
if (action == PunishmentAction.AddRole)
|
if (action == PunishmentAction.AddRole)
|
||||||
{
|
{
|
||||||
|
@ -120,13 +120,13 @@ public partial class Administration
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (punishTimespan is not null)
|
if (punishTime is not null)
|
||||||
{
|
{
|
||||||
if (!_service.IsDurationAllowed(action))
|
if (!_service.IsDurationAllowed(action))
|
||||||
await Response().Error(strs.prot_cant_use_time).SendAsync();
|
await Response().Error(strs.prot_cant_use_time).SendAsync();
|
||||||
}
|
}
|
||||||
|
|
||||||
var time = (int?)punishTimespan?.Time.TotalMinutes ?? 0;
|
var time = (int?)punishTime?.Time.TotalMinutes ?? 0;
|
||||||
if (time is < 0 or > 60 * 24)
|
if (time is < 0 or > 60 * 24)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
@ -170,8 +170,8 @@ public partial class Administration
|
||||||
[RequireContext(ContextType.Guild)]
|
[RequireContext(ContextType.Guild)]
|
||||||
[UserPerm(GuildPerm.Administrator)]
|
[UserPerm(GuildPerm.Administrator)]
|
||||||
[Priority(1)]
|
[Priority(1)]
|
||||||
public Task AntiSpam(int messageCount, PunishmentAction action, [Leftover] ParsedTimespan punishTimespan)
|
public Task AntiSpam(int messageCount, PunishmentAction action, [Leftover] StoopidTime punishTime)
|
||||||
=> InternalAntiSpam(messageCount, action, punishTimespan);
|
=> InternalAntiSpam(messageCount, action, punishTime);
|
||||||
|
|
||||||
[Cmd]
|
[Cmd]
|
||||||
[RequireContext(ContextType.Guild)]
|
[RequireContext(ContextType.Guild)]
|
||||||
|
@ -183,19 +183,19 @@ public partial class Administration
|
||||||
private async Task InternalAntiSpam(
|
private async Task InternalAntiSpam(
|
||||||
int messageCount,
|
int messageCount,
|
||||||
PunishmentAction action,
|
PunishmentAction action,
|
||||||
ParsedTimespan timespanData = null,
|
StoopidTime timeData = null,
|
||||||
IRole role = null)
|
IRole role = null)
|
||||||
{
|
{
|
||||||
if (messageCount is < 2 or > 10)
|
if (messageCount is < 2 or > 10)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
if (timespanData is not null)
|
if (timeData is not null)
|
||||||
{
|
{
|
||||||
if (!_service.IsDurationAllowed(action))
|
if (!_service.IsDurationAllowed(action))
|
||||||
await Response().Error(strs.prot_cant_use_time).SendAsync();
|
await Response().Error(strs.prot_cant_use_time).SendAsync();
|
||||||
}
|
}
|
||||||
|
|
||||||
var time = (int?)timespanData?.Time.TotalMinutes ?? 0;
|
var time = (int?)timeData?.Time.TotalMinutes ?? 0;
|
||||||
if (time is < 0 or > 60 * 24)
|
if (time is < 0 or > 60 * 24)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
@ -241,7 +241,7 @@ public partial class Administration
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
var embed = CreateEmbed().WithOkColor().WithTitle(GetText(strs.prot_active));
|
var embed = _sender.CreateEmbed().WithOkColor().WithTitle(GetText(strs.prot_active));
|
||||||
|
|
||||||
if (spam is not null)
|
if (spam is not null)
|
||||||
embed.AddField("Anti-Spam", GetAntiSpamString(spam).TrimTo(1024), true);
|
embed.AddField("Anti-Spam", GetAntiSpamString(spam).TrimTo(1024), true);
|
||||||
|
|
|
@ -22,7 +22,6 @@ public class ProtectionService : IEService
|
||||||
private readonly MuteService _mute;
|
private readonly MuteService _mute;
|
||||||
private readonly DbService _db;
|
private readonly DbService _db;
|
||||||
private readonly UserPunishService _punishService;
|
private readonly UserPunishService _punishService;
|
||||||
private readonly INotifySubscriber _notifySub;
|
|
||||||
|
|
||||||
private readonly Channel<PunishQueueItem> _punishUserQueue =
|
private readonly Channel<PunishQueueItem> _punishUserQueue =
|
||||||
Channel.CreateUnbounded<PunishQueueItem>(new()
|
Channel.CreateUnbounded<PunishQueueItem>(new()
|
||||||
|
@ -36,14 +35,12 @@ public class ProtectionService : IEService
|
||||||
IBot bot,
|
IBot bot,
|
||||||
MuteService mute,
|
MuteService mute,
|
||||||
DbService db,
|
DbService db,
|
||||||
UserPunishService punishService,
|
UserPunishService punishService)
|
||||||
INotifySubscriber notifySub)
|
|
||||||
{
|
{
|
||||||
_client = client;
|
_client = client;
|
||||||
_mute = mute;
|
_mute = mute;
|
||||||
_db = db;
|
_db = db;
|
||||||
_punishService = punishService;
|
_punishService = punishService;
|
||||||
_notifySub = notifySub;
|
|
||||||
|
|
||||||
var ids = client.GetGuildIds();
|
var ids = client.GetGuildIds();
|
||||||
using (var uow = db.GetDbContext())
|
using (var uow = db.GetDbContext())
|
||||||
|
@ -178,9 +175,6 @@ public class ProtectionService : IEService
|
||||||
alts.RoleId,
|
alts.RoleId,
|
||||||
user);
|
user);
|
||||||
|
|
||||||
await _notifySub.NotifyAsync(new ProtectionNotifyModel(user.Guild.Id,
|
|
||||||
ProtectionType.Alting,
|
|
||||||
user.Id));
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -200,8 +194,6 @@ public class ProtectionService : IEService
|
||||||
var settings = stats.AntiRaidSettings;
|
var settings = stats.AntiRaidSettings;
|
||||||
|
|
||||||
await PunishUsers(settings.Action, ProtectionType.Raiding, settings.PunishDuration, null, users);
|
await PunishUsers(settings.Action, ProtectionType.Raiding, settings.PunishDuration, null, users);
|
||||||
await _notifySub.NotifyAsync(
|
|
||||||
new ProtectionNotifyModel(user.Guild.Id, ProtectionType.Raiding, users[0].Id));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
await Task.Delay(1000 * stats.AntiRaidSettings.Seconds);
|
await Task.Delay(1000 * stats.AntiRaidSettings.Seconds);
|
||||||
|
@ -254,10 +246,6 @@ public class ProtectionService : IEService
|
||||||
settings.MuteTime,
|
settings.MuteTime,
|
||||||
settings.RoleId,
|
settings.RoleId,
|
||||||
(IGuildUser)msg.Author);
|
(IGuildUser)msg.Author);
|
||||||
|
|
||||||
await _notifySub.NotifyAsync(new ProtectionNotifyModel(channel.GuildId,
|
|
||||||
ProtectionType.Spamming,
|
|
||||||
msg.Author.Id));
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Some files were not shown because too many files have changed in this diff Show more
Loading…
Reference in a new issue