diff --git a/config/hugging-face-models.json b/config/hugging-face-models.json new file mode 100644 index 0000000..bc7b396 --- /dev/null +++ b/config/hugging-face-models.json @@ -0,0 +1,63 @@ +{ + "models": { + "runwayml/stable-diffusion-v1-5": [ + "" + ], + "prompthero/openjourney": [ + "mdjrny-v4 style, " + ], + "nitrosocke/mo-di-diffusion": [ + "modern disney style, " + ], + "wavymulder/Analog-Diffusion": [ + "analog style, " + ], + "nitrosocke/redshift-diffusion": [ + "redshift style, " + ], + "nitrosocke/Ghibli-Diffusion": [ + "ghibli style, " + ], + "Envvi/Inkpunk-Diffusion": [ + "nvinkpunk, " + ], + "ogkalu/Comic-Diffusion": [ + "charliebo artstyle, ", + "holliemengert artstyle, ", + "marioalberti artstyle, ", + "pepelarraz artstyle, ", + "andreasrocha artstyle, ", + "jamesdaly artstyle, " + ], + "nousr/robo-diffusion-2-base": [ + "nousr robot, " + ], + "Fictiverse/Stable_Diffusion_PaperCut_Model": [ + "PaperCut, " + ], + "nitrosocke/elden-ring-diffusion": [ + "elden ring style, " + ], + "dallinmackay/Van-Gogh-diffusion": [ + "lvngvncnt, " + ], + "Conflictx/Complex-Lineart": [ + "ComplexLA style, " + ], + "darkstorm2150/Protogen_x3.4_Official_Release": [ + "modelshoot style, ", + "analog style, ", + "mdjrny-v4 style, ", + "nousr robot, " + ], + "Aybeeceedee/knollingcase": [ + "knollingcase, " + ], + "wavymulder/portraitplus": [ + "portrait+ style, " + ], + "0xJustin/Dungeons-and-Diffusion": [ + "" + ] + } +} \ No newline at end of file diff --git a/kardbot/commands.go b/kardbot/commands.go index 42e0a89..a7ceb11 100644 --- a/kardbot/commands.go +++ b/kardbot/commands.go @@ -356,9 +356,22 @@ func getCommands() []*discordgo.ApplicationCommand { Options: timeCmdOpts(), }, { - Name: dalle2Cmd, - Description: "Ask an AI to generate an image from a prompt. Uses Open AI's DALL·E 2.", - Options: dalle2Opts(), + Name: renderCmd, + Description: "Ask an AI to generate an image from a prompt.", + Options: []*discordgo.ApplicationCommandOption{ + { + Name: dalle2SubCmd, + Description: "Ask Open AI's DALL·E 2 model to generate an image from a prompt.", + Type: discordgo.ApplicationCommandOptionSubCommand, + Options: dalle2Opts(), + }, + { + Name: hfSubCmd, + Description: "Ask a HuggingFace model to generate an image from a prompt.", + Type: discordgo.ApplicationCommandOptionSubCommand, + Options: hfOpts(), + }, + }, }, { Name: helpCmd, @@ -392,7 +405,7 @@ func getCommandImpls() map[string]onInteractionHandler { madlibCmd: handleMadLibCmd, timeCmd: handleTimeCmd, pollCmd: handlePollCmd, - dalle2Cmd: handleDalle2Cmd, + renderCmd: handleRenderCmd, } } diff --git a/kardbot/dalle2.go b/kardbot/dalle2.go deleted file mode 100644 index b4a99ec..0000000 --- a/kardbot/dalle2.go +++ /dev/null @@ -1,122 +0,0 @@ -package kardbot - -import ( - "bytes" - "encoding/base64" - "encoding/json" - "errors" - "fmt" - - "github.com/TannerKvarfordt/gopenai/images" - "github.com/TannerKvarfordt/gopenai/moderations" - "github.com/bwmarrin/discordgo" - - log "github.com/sirupsen/logrus" -) - -const ( - dalle2Cmd = "dalle2" - dalle2PromptOpt = "prompt" - dalle2SizeOpt = "size" -) - -func dalle2Opts() []*discordgo.ApplicationCommandOption { - return []*discordgo.ApplicationCommandOption{ - { - Type: discordgo.ApplicationCommandOptionString, - Name: dalle2PromptOpt, - Description: "A prompt to generate an image from. This can be very specific.", - Required: true, - }, - { - Type: discordgo.ApplicationCommandOptionString, - Name: dalle2SizeOpt, - Description: "The size of the image to be generated", - Required: true, - Choices: []*discordgo.ApplicationCommandOptionChoice{ - { - Name: images.SmallImage, - Value: images.SmallImage, - }, - { - Name: images.MediumImage, - Value: images.MediumImage, - }, - { - Name: images.LargeImage, - Value: images.LargeImage, - }, - }, - }, - } -} - -func handleDalle2Cmd(s *discordgo.Session, i *discordgo.InteractionCreate) { - err := s.InteractionRespond(i.Interaction, &discordgo.InteractionResponse{ - Type: discordgo.InteractionResponseDeferredChannelMessageWithSource, - }) - if err != nil { - log.Error(err) - interactionRespondEphemeralError(s, i, true, err) - return - } - - mdata, err := getInteractionMetaData(i) - if err != nil { - log.Error(err) - interactionFollowUpEphemeralError(s, i, true, err) - return - } - - prompt := i.ApplicationCommandData().Options[0].StringValue() - size := i.ApplicationCommandData().Options[1].StringValue() - imageCount := uint64(1) - resp, modr, err := images.MakeModeratedCreationRequest(&images.CreationRequest{ - Prompt: prompt, - N: &imageCount, - Size: size, - ResponseFormat: images.ResponseFormatB64JSON, - User: mdata.AuthorID, - }, nil) - if err != nil { - targetErr := moderations.NewModerationFlagError() - if errors.As(err, &targetErr) { - contentFlags, err := json.MarshalIndent(modr.Results[0].Categories, "", " ") - if err != nil { - log.Error(err) - contentFlags = []byte("Whoops, couldn't retrieve the details of your violation.") - } - interactionFollowUpEphemeralError(s, i, false, fmt.Errorf("sorry! Your prompt does not appear to conform to [Open AI's Usage Policies]()\n```JSON\n%s\n```", contentFlags)) - } else { - log.Error(err) - interactionFollowUpEphemeralError(s, i, true, err) - } - return - } - - unbased, err := base64.StdEncoding.DecodeString(resp.Data[0].B64JSON) - if err != nil { - log.Error(err) - interactionFollowUpEphemeralError(s, i, true, err) - return - } - - errMsg := fmt.Sprintf("> %s\n\nImage generated using [DALL·E 2]().", prompt) - _, err = s.InteractionResponseEdit(i.Interaction, &discordgo.WebhookEdit{ - Content: &errMsg, - Files: []*discordgo.File{ - { - Name: "Dalle-2-Output.png", - ContentType: "image/png", - Reader: bytes.NewReader(unbased), - }, - }, - AllowedMentions: &discordgo.MessageAllowedMentions{ - Users: []string{mdata.AuthorID}, - }, - }) - if err != nil { - log.Error(err) - interactionFollowUpEphemeralError(s, i, true, err) - } -} diff --git a/kardbot/render.go b/kardbot/render.go new file mode 100644 index 0000000..17d39f7 --- /dev/null +++ b/kardbot/render.go @@ -0,0 +1,277 @@ +package kardbot + +import ( + "bytes" + "encoding/base64" + "encoding/json" + "errors" + "fmt" + "image/jpeg" + "image/png" + "math/rand" + "mime" + "strings" + + "github.com/TannerKvarfordt/Kard-bot/kardbot/config" + "github.com/TannerKvarfordt/gopenai/images" + "github.com/TannerKvarfordt/gopenai/moderations" + "github.com/TannerKvarfordt/hfapigo" + "github.com/bwmarrin/discordgo" + + log "github.com/sirupsen/logrus" +) + +const ( + renderCmd = "render" + + hfSubCmd = "hugging-face" + hfPromptOpt = "prompt" + hfModelOpt = "model" + hfModelOptCustom = "custom-model" + hfModelsFilepath = "config/hugging-face-models.json" + + dalle2SubCmd = "dalle2" + dalle2PromptOpt = "prompt" + dalle2SizeOpt = "size" +) + +func handleRenderCmd(s *discordgo.Session, i *discordgo.InteractionCreate) { + err := s.InteractionRespond(i.Interaction, &discordgo.InteractionResponse{ + Type: discordgo.InteractionResponseDeferredChannelMessageWithSource, + }) + if err != nil { + log.Error(err) + interactionRespondEphemeralError(s, i, true, err) + return + } + + switch i.ApplicationCommandData().Options[0].Name { + case hfSubCmd: + handleHfSubCmd(s, i, i.ApplicationCommandData().Options[0].Options) + case dalle2SubCmd: + handleDalle2SubCmd(s, i, i.ApplicationCommandData().Options[0].Options) + default: + err = fmt.Errorf("reached unreachable case") + log.Error(err) + interactionFollowUpEphemeralError(s, i, true, err) + } +} + +var hfModels = func() []*discordgo.ApplicationCommandOptionChoice { return nil } + +// A mapping of model names to keywords for that model +var hfModelKeyWords = func() map[string][]string { return nil } + +func init() { + cfg := struct { + // A map of model names to activation words for the model + Models map[string][]string `json:"models"` + }{} + + jsonCfg, err := config.NewJsonConfig(hfModelsFilepath) + if err != nil { + log.Fatal(err) + } + + err = json.Unmarshal(jsonCfg.Raw, &cfg) + if err != nil { + log.Fatal(err) + } + + modelChoices := []*discordgo.ApplicationCommandOptionChoice{} + for model := range cfg.Models { + if strings.ToLower(model) == hfModelOptCustom { + log.Warnf(`Custom model name "%s" conflicts with a builtin model name. It will be ignored.`, hfModelOptCustom) + continue + } + modelChoices = append(modelChoices, &discordgo.ApplicationCommandOptionChoice{ + Name: model, + Value: model, + }) + } + + modelChoices[len(modelChoices)-1] = &discordgo.ApplicationCommandOptionChoice{ + Name: hfModelOptCustom, + Value: hfModelOptCustom, + } + + hfModels = func() []*discordgo.ApplicationCommandOptionChoice { return modelChoices } + hfModelKeyWords = func() map[string][]string { return cfg.Models } +} + +func hfOpts() []*discordgo.ApplicationCommandOption { + return []*discordgo.ApplicationCommandOption{ + { + Type: discordgo.ApplicationCommandOptionString, + Name: hfModelOpt, + Description: "The model to use when generating the image.", + Required: true, + Choices: hfModels(), + }, + { + Type: discordgo.ApplicationCommandOptionString, + Name: hfPromptOpt, + Description: "A prompt to generate an image from. This can be very specific.", + Required: true, + }, + { + Type: discordgo.ApplicationCommandOptionString, + Name: hfModelOptCustom, + Description: "Any text-to-image model from huggingface.co", + Required: false, + }, + } +} + +func handleHfSubCmd(s *discordgo.Session, i *discordgo.InteractionCreate, opts []*discordgo.ApplicationCommandInteractionDataOption) { + prompt := opts[1].StringValue() + model := opts[0].StringValue() + if model == hfModelOptCustom { + if len(opts) < 3 { + interactionFollowUpEphemeralError(s, i, false, fmt.Errorf(`you must specify a custom model to use when selecting the "%s" model`, hfModelOptCustom)) + return + } + model = opts[2].StringValue() + } + modelKeyWords := hfModelKeyWords()[model] + if len(modelKeyWords) == 0 { + modelKeyWords = append(modelKeyWords, "") + } + + img, imgFmt, err := hfapigo.SendTextToImageRequest(model, &hfapigo.TextToImageRequest{ + Inputs: fmt.Sprintf("%s%s", modelKeyWords[rand.Intn(len(modelKeyWords))], prompt), + Options: *hfapigo.NewOptions().SetWaitForModel(true), + }) + if err != nil { + // TODO: detect if error is invalid model, and report to user instead of owner. + log.Error(err) + interactionFollowUpEphemeralError(s, i, true, err) + return + } + + imgMimeType := mime.TypeByExtension(fmt.Sprintf(".%s", imgFmt)) + buf := new(bytes.Buffer) + switch imgMimeType { + case "image/jpeg": + err = jpeg.Encode(buf, img, &jpeg.Options{ + Quality: 100, + }) + case "image/png": + err = png.Encode(buf, img) + default: + err = fmt.Errorf("unsupported image type (%s) returned", imgFmt) + } + if err != nil { + log.Error(err) + interactionFollowUpEphemeralError(s, i, true, err) + return + } + + content := fmt.Sprintf("> %s\n\nImage generated using [%s]().", prompt, model, model) + _, err = s.InteractionResponseEdit(i.Interaction, &discordgo.WebhookEdit{ + Content: &content, + Files: []*discordgo.File{ + { + Name: fmt.Sprintf("HuggingFaceImg.%s", imgFmt), + ContentType: imgMimeType, + Reader: buf, + }, + }, + }) + if err != nil { + log.Error(err) + interactionFollowUpEphemeralError(s, i, true, err) + } +} + +func dalle2Opts() []*discordgo.ApplicationCommandOption { + return []*discordgo.ApplicationCommandOption{ + { + Type: discordgo.ApplicationCommandOptionString, + Name: dalle2PromptOpt, + Description: "A prompt to generate an image from. This can be very specific.", + Required: true, + }, + { + Type: discordgo.ApplicationCommandOptionString, + Name: dalle2SizeOpt, + Description: "The size of the image to be generated", + Required: true, + Choices: []*discordgo.ApplicationCommandOptionChoice{ + { + Name: images.SmallImage, + Value: images.SmallImage, + }, + { + Name: images.MediumImage, + Value: images.MediumImage, + }, + { + Name: images.LargeImage, + Value: images.LargeImage, + }, + }, + }, + } +} + +func handleDalle2SubCmd(s *discordgo.Session, i *discordgo.InteractionCreate, opts []*discordgo.ApplicationCommandInteractionDataOption) { + mdata, err := getInteractionMetaData(i) + if err != nil { + log.Error(err) + interactionFollowUpEphemeralError(s, i, true, err) + return + } + + prompt := opts[0].StringValue() + size := opts[1].StringValue() + imageCount := uint64(1) + resp, modr, err := images.MakeModeratedCreationRequest(&images.CreationRequest{ + Prompt: prompt, + N: &imageCount, + Size: size, + ResponseFormat: images.ResponseFormatB64JSON, + User: mdata.AuthorID, + }, nil) + if err != nil { + targetErr := moderations.NewModerationFlagError() + if errors.As(err, &targetErr) { + contentFlags, err := json.MarshalIndent(modr.Results[0].Categories, "", " ") + if err != nil { + log.Error(err) + contentFlags = []byte("Whoops, couldn't retrieve the details of your violation.") + } + interactionFollowUpEphemeralError(s, i, false, fmt.Errorf("sorry! Your prompt does not appear to conform to [Open AI's Usage Policies]()\n```JSON\n%s\n```", contentFlags)) + } else { + log.Error(err) + interactionFollowUpEphemeralError(s, i, true, err) + } + return + } + + unbased, err := base64.StdEncoding.DecodeString(resp.Data[0].B64JSON) + if err != nil { + log.Error(err) + interactionFollowUpEphemeralError(s, i, true, err) + return + } + + content := fmt.Sprintf("> %s\n\nImage generated using [DALL·E 2]().", prompt) + _, err = s.InteractionResponseEdit(i.Interaction, &discordgo.WebhookEdit{ + Content: &content, + Files: []*discordgo.File{ + { + Name: "Dalle-2-Output.png", + ContentType: "image/png", + Reader: bytes.NewReader(unbased), + }, + }, + AllowedMentions: &discordgo.MessageAllowedMentions{ + Users: []string{mdata.AuthorID}, + }, + }) + if err != nil { + log.Error(err) + interactionFollowUpEphemeralError(s, i, true, err) + } +}