namespace Degenz open System open System.Threading.Tasks open DSharpPlus open DSharpPlus.Entities open DSharpPlus.EventArgs open DSharpPlus.SlashCommands open Newtonsoft.Json [] module ResultHelpers = let (>>=) x f = Result.bind f x let () x f = Result.map f x [] module Types = [] type mins [] type GBT type BattleClass = | Network | Exploit | Penetration type HackId = | Virus = 0 | RemoteAccess = 1 | Worm = 2 type ShieldId = | Firewall = 6 | Encryption = 7 | Cypher = 8 type ItemType = | Hack | Shield type BattleItem = { Id : int Name : string Cost : int Type : ItemType Class : BattleClass Power : int Cooldown : int } type HackResult = | Strong | Weak [] type DiscordPlayer = { Id: uint64; Name: string } [] type AttackResult = { Result : bool Target : DiscordPlayer } type ActionType = | Attack of AttackResult | Defense [] type Action = { ActionId : int Type : ActionType Timestamp : DateTime } type StatAmount = int type XPAmount = int type AttributeId = | Strength = 0 | Cunning = 1 type PlayerStats = { Strength : int Focus : int } with static member empty = { Strength = 0 ; Focus = 0 } type PlayerXP = { Amount : XPAmount } [] type PlayerData = { DiscordId : uint64 Name : string Arsenal : BattleItem array Actions : Action array Stats : PlayerStats XP : int Bank : int } module Armory = let battleItems = let file = System.IO.File.ReadAllText("Items.json") JsonConvert.DeserializeObject(file) let hacks = battleItems |> Array.filter (fun bi -> match bi.Type with Hack -> true | Shield -> false) let shields = battleItems |> Array.filter (fun bi -> match bi.Type with Shield -> true | Hack -> false) let getItem itemId = battleItems |> Array.find (fun w -> w.Id = itemId) module Messaging = type InteractiveMessage = { ButtonId : string ButtonText : string Message : string } type DiscordContext = | Interaction of InteractionContext | Event of ComponentInteractionCreateEventArgs type IDiscordContext = abstract member Respond : InteractionResponseType -> DiscordInteractionResponseBuilder -> Task abstract member FollowUp : DiscordFollowupMessageBuilder -> Task abstract member GetDiscordMember : unit -> DiscordMember abstract member GetGuild : unit -> DiscordGuild abstract member GetInteractionId : unit -> string abstract member GetChannel : unit -> DiscordChannel type DiscordInteractionContext(ctx : InteractionContext) = interface IDiscordContext with member this.Respond responseType builder = async { do! ctx.Interaction.CreateResponseAsync(responseType, builder) |> Async.AwaitTask } |> Async.StartAsTask :> Task member this.FollowUp(builder) = async { do! ctx.Interaction.CreateFollowupMessageAsync(builder) |> Async.AwaitTask |> Async.Ignore } |> Async.StartAsTask :> Task member this.GetDiscordMember() = ctx.Member member this.GetGuild() = ctx.Guild member this.GetInteractionId() = string ctx.InteractionId member this.GetChannel() = ctx.Channel type DiscordEventContext(ctx : ComponentInteractionCreateEventArgs) = interface IDiscordContext with member this.Respond responseType builder = async { do! ctx.Interaction.CreateResponseAsync(responseType, builder) |> Async.AwaitTask } |> Async.StartAsTask :> Task member this.FollowUp(builder) = async { do! ctx.Interaction.CreateFollowupMessageAsync(builder) |> Async.AwaitTask |> Async.Ignore } |> Async.StartAsTask :> Task member this.GetDiscordMember() = ctx.User :?> DiscordMember member this.GetGuild() = ctx.Guild member this.GetInteractionId() = ctx.Id member this.GetChannel() = ctx.Channel let getTimeText isCooldown (timespan : TimeSpan) timestamp = let span = if isCooldown then timespan - (DateTime.UtcNow - timestamp) else (DateTime.UtcNow - timestamp) let plural amount = if amount = 1 then "" else "s" let ``and`` = if span.Hours > 0 then "and " else "" let hours = if span.Hours > 0 then $"{span.Hours} hour{plural span.Hours} {``and``}" else String.Empty let totalMins = span.Minutes let minutes = if totalMins > 0 then $"{totalMins} minute{plural totalMins}" else "1 minute" $"{hours}{minutes}" let getShortTimeText (timespan : TimeSpan) timestamp = let remaining = timespan - (DateTime.UtcNow - timestamp) let hours = if remaining.Hours > 0 then $"{remaining.Hours}h " else String.Empty let minutesRemaining = if remaining.Hours = 0 then remaining.Minutes + 1 else remaining.Minutes $"{hours}{minutesRemaining}min" let sendSimpleResponse (ctx: IDiscordContext) msg = async { let builder = DiscordInteractionResponseBuilder() builder.Content <- msg builder.AsEphemeral true |> ignore do! ctx.Respond InteractionResponseType.ChannelMessageWithSource builder |> Async.AwaitTask } let sendFollowUpMessage (ctx : IDiscordContext) msg = async { let builder = DiscordFollowupMessageBuilder() builder.IsEphemeral <- true builder.Content <- msg do! ctx.FollowUp(builder) |> Async.AwaitTask } let sendFollowUpEmbed (ctx : IDiscordContext) embed = async { let builder = DiscordFollowupMessageBuilder() .AsEphemeral(true) .AddEmbed(embed) do! ctx.FollowUp(builder) |> Async.AwaitTask } let sendFollowUpMessageWithButton (ctx : IDiscordContext) interactiveMessage = async { let builder = DiscordFollowupMessageBuilder() let button = DiscordButtonComponent(ButtonStyle.Success, interactiveMessage.ButtonId, interactiveMessage.ButtonText) :> DiscordComponent builder.AddComponents [| button |] |> ignore builder.IsEphemeral <- true builder.Content <- interactiveMessage.Message do! ctx.FollowUp(builder) |> Async.AwaitTask } let updateMessageWithGreyedOutButtons (ctx : IDiscordContext) interactiveMessage = async { let builder = DiscordInteractionResponseBuilder() let button = DiscordButtonComponent(ButtonStyle.Success, interactiveMessage.ButtonId, interactiveMessage.ButtonText, true) :> DiscordComponent builder.AddComponents [| button |] |> ignore builder.IsEphemeral <- true builder.Content <- interactiveMessage.Message do! ctx.Respond InteractionResponseType.UpdateMessage builder |> Async.AwaitTask } let handleResultWithResponse ctx fn (player : Result) = match player with | Ok p -> fn p | Error e -> async { do! sendFollowUpMessage ctx e } let handleResultWithResponseFromEvent event fn (player : Result) = match player with | Ok p -> fn p | Error e -> async { do! sendFollowUpMessage event e }