Global replace [ \t]+$, add "GB" (#1751)

* "GB"

* Replace [ \t]+$ global

Co-authored-by: Lincoln Stein <lincoln.stein@gmail.com>
This commit is contained in:
Scott Lahteine 2022-12-19 10:36:39 -06:00 committed by GitHub
parent 4fd97ceddd
commit 7d8d4bcafb
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
45 changed files with 575 additions and 148 deletions

2
.gitattributes vendored
View File

@ -1,4 +1,4 @@
# Auto normalizes line endings on commit so devs don't need to change local settings. # Auto normalizes line endings on commit so devs don't need to change local settings.
# Only affects text files and ignores other file types. # Only affects text files and ignores other file types.
# For more info see: https://www.aleksandrhovhannisyan.com/blog/crlf-vs-lf-normalizing-line-endings-in-git/ # For more info see: https://www.aleksandrhovhannisyan.com/blog/crlf-vs-lf-normalizing-line-endings-in-git/
* text=auto * text=auto

View File

@ -1,4 +1,4 @@
<img src="docs/assets/invoke_ai_banner.png" align="center"> <img src="docs/assets/invoke_ai_banner.png" align="center">
Invoke-AI is a community of software developers, researchers, and user Invoke-AI is a community of software developers, researchers, and user
interface experts who have come together on a voluntary basis to build interface experts who have come together on a voluntary basis to build
@ -81,5 +81,5 @@ area. Disputes are resolved by open and honest communication.
## Signature ## Signature
This document has been collectively crafted and approved by the current InvokeAI team members, as of 28 Nov 2022: **lstein** (Lincoln Stein), **blessedcoolant**, **hipsterusername** (Kent Keirsey), **Kyle0654** (Kyle Schouviller), **damian0815**, **mauwii** (Matthias Wild), **Netsvetaev** (Artur Netsvetaev), **psychedelicious**, **tildebyte**, and **keturn**. Although individuals within the group may hold differing views on particular details and/or their implications, we are all in agreement about its fundamental statements, as well as their significance and importance to this project moving forward. This document has been collectively crafted and approved by the current InvokeAI team members, as of 28 Nov 2022: **lstein** (Lincoln Stein), **blessedcoolant**, **hipsterusername** (Kent Keirsey), **Kyle0654** (Kyle Schouviller), **damian0815**, **mauwii** (Matthias Wild), **Netsvetaev** (Artur Netsvetaev), **psychedelicious**, **tildebyte**, and **keturn**. Although individuals within the group may hold differing views on particular details and/or their implications, we are all in agreement about its fundamental statements, as well as their significance and importance to this project moving forward.

View File

@ -53,11 +53,11 @@ For full installation and upgrade instructions, please see:
1. Go to the bottom of the [Latest Release Page](https://github.com/invoke-ai/InvokeAI/releases/latest) 1. Go to the bottom of the [Latest Release Page](https://github.com/invoke-ai/InvokeAI/releases/latest)
2. Download the .zip file for your OS (Windows/macOS/Linux). 2. Download the .zip file for your OS (Windows/macOS/Linux).
3. Unzip the file. 3. Unzip the file.
4. If you are on Windows, double-click on the `install.bat` script. On macOS, open a Terminal window, drag the file `install.sh` from Finder into the Terminal, and press return. On Linux, run `install.sh`. 4. If you are on Windows, double-click on the `install.bat` script. On macOS, open a Terminal window, drag the file `install.sh` from Finder into the Terminal, and press return. On Linux, run `install.sh`.
5. Wait a while, until it is done. 5. Wait a while, until it is done.
6. The folder where you ran the installer from will now be filled with lots of files. If you are on Windows, double-click on the `invoke.bat` file. On macOS, open a Terminal window, drag `invoke.sh` from the folder into the Terminal, and press return. On Linux, run `invoke.sh` 6. The folder where you ran the installer from will now be filled with lots of files. If you are on Windows, double-click on the `invoke.bat` file. On macOS, open a Terminal window, drag `invoke.sh` from the folder into the Terminal, and press return. On Linux, run `invoke.sh`
7. Press 2 to open the "browser-based UI", press enter/return, wait a minute or two for Stable Diffusion to start up, then open your browser and go to http://localhost:9090. 7. Press 2 to open the "browser-based UI", press enter/return, wait a minute or two for Stable Diffusion to start up, then open your browser and go to http://localhost:9090.
8. Type `banana sushi` in the box on the top left and click `Invoke`: 8. Type `banana sushi` in the box on the top left and click `Invoke`:
<div align="center"><img src="docs/assets/invoke-web-server-1.png" width=640></div> <div align="center"><img src="docs/assets/invoke-web-server-1.png" width=640></div>
@ -161,9 +161,9 @@ problems and other issues.
# Contributing # Contributing
Anyone who wishes to contribute to this project, whether documentation, features, bug fixes, code Anyone who wishes to contribute to this project, whether documentation, features, bug fixes, code
cleanup, testing, or code reviews, is very much encouraged to do so. cleanup, testing, or code reviews, is very much encouraged to do so.
To join, just raise your hand on the InvokeAI Discord server (#dev-chat) or the GitHub discussion board. To join, just raise your hand on the InvokeAI Discord server (#dev-chat) or the GitHub discussion board.
If you are unfamiliar with how If you are unfamiliar with how
to contribute to GitHub projects, here is a to contribute to GitHub projects, here is a

View File

@ -21,7 +21,7 @@ This model card focuses on the model associated with the Stable Diffusion model,
# Uses # Uses
## Direct Use ## Direct Use
The model is intended for research purposes only. Possible research areas and The model is intended for research purposes only. Possible research areas and
tasks include tasks include
@ -68,11 +68,11 @@ Using the model to generate content that is cruel to individuals is a misuse of
considerations. considerations.
### Bias ### Bias
While the capabilities of image generation models are impressive, they can also reinforce or exacerbate social biases. While the capabilities of image generation models are impressive, they can also reinforce or exacerbate social biases.
Stable Diffusion v1 was trained on subsets of [LAION-2B(en)](https://laion.ai/blog/laion-5b/), Stable Diffusion v1 was trained on subsets of [LAION-2B(en)](https://laion.ai/blog/laion-5b/),
which consists of images that are primarily limited to English descriptions. which consists of images that are primarily limited to English descriptions.
Texts and images from communities and cultures that use other languages are likely to be insufficiently accounted for. Texts and images from communities and cultures that use other languages are likely to be insufficiently accounted for.
This affects the overall output of the model, as white and western cultures are often set as the default. Further, the This affects the overall output of the model, as white and western cultures are often set as the default. Further, the
ability of the model to generate content with non-English prompts is significantly worse than with English-language prompts. ability of the model to generate content with non-English prompts is significantly worse than with English-language prompts.
@ -84,7 +84,7 @@ The model developers used the following dataset for training the model:
- LAION-2B (en) and subsets thereof (see next section) - LAION-2B (en) and subsets thereof (see next section)
**Training Procedure** **Training Procedure**
Stable Diffusion v1 is a latent diffusion model which combines an autoencoder with a diffusion model that is trained in the latent space of the autoencoder. During training, Stable Diffusion v1 is a latent diffusion model which combines an autoencoder with a diffusion model that is trained in the latent space of the autoencoder. During training,
- Images are encoded through an encoder, which turns images into latent representations. The autoencoder uses a relative downsampling factor of 8 and maps images of shape H x W x 3 to latents of shape H/f x W/f x 4 - Images are encoded through an encoder, which turns images into latent representations. The autoencoder uses a relative downsampling factor of 8 and maps images of shape H x W x 3 to latents of shape H/f x W/f x 4
- Text prompts are encoded through a ViT-L/14 text-encoder. - Text prompts are encoded through a ViT-L/14 text-encoder.
@ -108,12 +108,12 @@ filtered to images with an original size `>= 512x512`, estimated aesthetics scor
- **Batch:** 32 x 8 x 2 x 4 = 2048 - **Batch:** 32 x 8 x 2 x 4 = 2048
- **Learning rate:** warmup to 0.0001 for 10,000 steps and then kept constant - **Learning rate:** warmup to 0.0001 for 10,000 steps and then kept constant
## Evaluation Results ## Evaluation Results
Evaluations with different classifier-free guidance scales (1.5, 2.0, 3.0, 4.0, Evaluations with different classifier-free guidance scales (1.5, 2.0, 3.0, 4.0,
5.0, 6.0, 7.0, 8.0) and 50 PLMS sampling 5.0, 6.0, 7.0, 8.0) and 50 PLMS sampling
steps show the relative improvements of the checkpoints: steps show the relative improvements of the checkpoints:
![pareto](assets/v1-variants-scores.jpg) ![pareto](assets/v1-variants-scores.jpg)
Evaluated using 50 PLMS steps and 10000 random prompts from the COCO2017 validation set, evaluated at 512x512 resolution. Not optimized for FID scores. Evaluated using 50 PLMS steps and 10000 random prompts from the COCO2017 validation set, evaluated at 512x512 resolution. Not optimized for FID scores.
## Environmental Impact ## Environmental Impact

View File

@ -43,7 +43,7 @@ def get_canvas_generation_mode(
) )
""" """
Mask images are white in areas where no change should be made, black where changes Mask images are white in areas where no change should be made, black where changes
should be made. should be made.
""" """

View File

@ -31,7 +31,7 @@ stable-diffusion-1.4:
width: 512 width: 512
height: 512 height: 512
waifu-diffusion-1.3: waifu-diffusion-1.3:
description: Stable Diffusion 1.4 fine tuned on anime-styled images (4.27) description: Stable Diffusion 1.4 fine tuned on anime-styled images (4.27 GB)
repo_id: hakurei/waifu-diffusion-v1-3 repo_id: hakurei/waifu-diffusion-v1-3
config: v1-inference.yaml config: v1-inference.yaml
file: model-epoch09-float32.ckpt file: model-epoch09-float32.ckpt

View File

@ -107,4 +107,4 @@ lightning:
benchmark: True benchmark: True
max_steps: 4000000 max_steps: 4000000
# max_steps: 4000 # max_steps: 4000

View File

@ -107,4 +107,4 @@ lightning:
benchmark: False benchmark: False
max_steps: 6200 max_steps: 6200
# max_steps: 4000 # max_steps: 4000

View File

@ -1,4 +1,4 @@
The Unified Canvas is a tool designed to streamline and simplify the process of composing an image using Stable Diffusion. It offers artists all of the available Stable Diffusion generation modes (Text To Image, Image To Image, Inpainting, and Outpainting) as a single unified workflow. The flexibility of the tool allows you to tweak and edit image generations, extend images beyond their initial size, and to create new content in a freeform way both inside and outside of existing images. The Unified Canvas is a tool designed to streamline and simplify the process of composing an image using Stable Diffusion. It offers artists all of the available Stable Diffusion generation modes (Text To Image, Image To Image, Inpainting, and Outpainting) as a single unified workflow. The flexibility of the tool allows you to tweak and edit image generations, extend images beyond their initial size, and to create new content in a freeform way both inside and outside of existing images.
This document explains the basics of using the Unified Canvas, introducing you to its features and tools one by one. It also describes some of the more advanced tools available to power users of the Canvas. This document explains the basics of using the Unified Canvas, introducing you to its features and tools one by one. It also describes some of the more advanced tools available to power users of the Canvas.
@ -21,7 +21,7 @@ Accepting generations will commit the new generation to the **Base Layer**. You
The **Mask Layer** consists of any masked sections that have been created to inform Inpainting generations. You can paint a new mask, or edit an existing mask, using the Brush tool and the Eraser with the Mask layer set as your Active layer. Any masked areas will only affect generation inside of the current bounding box. The **Mask Layer** consists of any masked sections that have been created to inform Inpainting generations. You can paint a new mask, or edit an existing mask, using the Brush tool and the Eraser with the Mask layer set as your Active layer. Any masked areas will only affect generation inside of the current bounding box.
### Bounding Box ### Bounding Box
When generating a new image, Invoke will process and apply new images within the area denoted by the **Bounding Box**. The Width & Height settings of the Bounding Box, as well as its location within the Unified Canvas and pixels or empty space that it encloses, determine how new invocations are generated - see [Inpainting & Outpainting](#inpainting-and-outpainting) below. The Bounding Box can be moved and resized using the Move (V) tool. It can also be resized using the Bounding Box options in the Options Panel. By using these controls you can generate larger or smaller images, control which sections of the image are being processed, as well as control Bounding Box tools like the Bounding Box fill/erase. When generating a new image, Invoke will process and apply new images within the area denoted by the **Bounding Box**. The Width & Height settings of the Bounding Box, as well as its location within the Unified Canvas and pixels or empty space that it encloses, determine how new invocations are generated - see [Inpainting & Outpainting](#inpainting-and-outpainting) below. The Bounding Box can be moved and resized using the Move (V) tool. It can also be resized using the Bounding Box options in the Options Panel. By using these controls you can generate larger or smaller images, control which sections of the image are being processed, as well as control Bounding Box tools like the Bounding Box fill/erase.
### <a name="inpainting-and-outpainting"></a> Inpainting & Outpainting ### <a name="inpainting-and-outpainting"></a> Inpainting & Outpainting
"Inpainting" means asking the AI to refine part of an image while leaving the rest alone. For example, updating a portrait of your grandmother to have her wear a biker's jacket. "Inpainting" means asking the AI to refine part of an image while leaving the rest alone. For example, updating a portrait of your grandmother to have her wear a biker's jacket.
@ -48,9 +48,9 @@ To get started with the Unified Canvas, you will want to generate a new base lay
From there, you can consider the following techniques to augment your image: From there, you can consider the following techniques to augment your image:
* **New Images**: Move the bounding box to an empty area of the Canvas, type in your prompt, and Invoke, to generate a new image using the Text to Image function. * **New Images**: Move the bounding box to an empty area of the Canvas, type in your prompt, and Invoke, to generate a new image using the Text to Image function.
* **Image Correction**: Use the color picker and brush tool to paint corrections on the image, switch to the Mask layer, and brush a mask over your painted area to use **Inpainting**. You can also use the **ImageToImage** generation method to invoke new interpretations of the image. * **Image Correction**: Use the color picker and brush tool to paint corrections on the image, switch to the Mask layer, and brush a mask over your painted area to use **Inpainting**. You can also use the **ImageToImage** generation method to invoke new interpretations of the image.
* **Image Expansion**: Move the bounding box to include a portion of your initial image, and a portion of transparent/empty pixels, then Invoke using a prompt that describes what you'd like to see in that area. This will Outpaint the image. You'll typically find more coherent results if you keep about 50-60% of the original image in the bounding box. Make sure that the Image To Image Strength slider is set to a high value - you may need to set it higher than you are used to. * **Image Expansion**: Move the bounding box to include a portion of your initial image, and a portion of transparent/empty pixels, then Invoke using a prompt that describes what you'd like to see in that area. This will Outpaint the image. You'll typically find more coherent results if you keep about 50-60% of the original image in the bounding box. Make sure that the Image To Image Strength slider is set to a high value - you may need to set it higher than you are used to.
* **New Content on Existing Images**: If you want to add new details or objects into your image, use the brush tool to paint a sketch of what you'd like to see on the image, switch to the Mask layer, and brush a mask over your painted area to use **Inpainting**. If the masked area is small, consider using a smaller bounding box to take advantage of Invoke's automatic Scaling features, which can help to produce better details. * **New Content on Existing Images**: If you want to add new details or objects into your image, use the brush tool to paint a sketch of what you'd like to see on the image, switch to the Mask layer, and brush a mask over your painted area to use **Inpainting**. If the masked area is small, consider using a smaller bounding box to take advantage of Invoke's automatic Scaling features, which can help to produce better details.
* **And more**: There are a number of creative ways to use the Canvas, and the above are just starting points. We're excited to see what you come up with! * **And more**: There are a number of creative ways to use the Canvas, and the above are just starting points. We're excited to see what you come up with!
@ -82,27 +82,27 @@ Features with non-obvious behavior are detailed below, in order to provide clari
## Toolbar ## Toolbar
### Mask Options ### Mask Options
* **Enable Mask** - This flag can be used to Enable or Disable the currently painted mask. If you have painted a mask, but you don't want it affect the next invocation, but you *also* don't want to delete it, then you can set this option to Disable. When you want the mask back, set this back to Enable. * **Enable Mask** - This flag can be used to Enable or Disable the currently painted mask. If you have painted a mask, but you don't want it affect the next invocation, but you *also* don't want to delete it, then you can set this option to Disable. When you want the mask back, set this back to Enable.
* **Preserve Masked Area** - When enabled, Preserve Masked Area inverts the effect of the Mask on the Inpainting process. Pixels in masked areas will be kept unchanged, and unmasked areas will be regenerated. * **Preserve Masked Area** - When enabled, Preserve Masked Area inverts the effect of the Mask on the Inpainting process. Pixels in masked areas will be kept unchanged, and unmasked areas will be regenerated.
### Creative Tools ### Creative Tools
* **Brush - Base/Mask Modes** - The Brush tool switches automatically between different modes of operation for the Base and Mask layers respectively. * **Brush - Base/Mask Modes** - The Brush tool switches automatically between different modes of operation for the Base and Mask layers respectively.
* On the Base layer, the brush will directly paint on the Canvas using the color selected on the Brush Options menu. * On the Base layer, the brush will directly paint on the Canvas using the color selected on the Brush Options menu.
* On the Mask layer, the brush will create a new mask. If you're finding the mask difficult to see over the existing content of the Unified Canvas, you can change the color it is drawn with using the color selector on the Mask Options dropdown. * On the Mask layer, the brush will create a new mask. If you're finding the mask difficult to see over the existing content of the Unified Canvas, you can change the color it is drawn with using the color selector on the Mask Options dropdown.
* **Erase Bounding Box** - On the Base layer, erases all pixels within the Bounding Box. * **Erase Bounding Box** - On the Base layer, erases all pixels within the Bounding Box.
* **Fill Bounding Box** - On the Base layer, fills all pixels within the Bounding Box with the currently selected color. * **Fill Bounding Box** - On the Base layer, fills all pixels within the Bounding Box with the currently selected color.
### Canvas Tools ### Canvas Tools
* **Move Tool** - Allows for manipulation of the Canvas view (by dragging on the Canvas, outside the bounding box), the Bounding Box (by dragging the edges of the box), or the Width/Height of the Bounding Box (by dragging one of the 9 directional handles). * **Move Tool** - Allows for manipulation of the Canvas view (by dragging on the Canvas, outside the bounding box), the Bounding Box (by dragging the edges of the box), or the Width/Height of the Bounding Box (by dragging one of the 9 directional handles).
* **Reset View** - Click to re-orients the view to the center of the Bounding Box. * **Reset View** - Click to re-orients the view to the center of the Bounding Box.
* **Merge Visible** - If your browser is having performance problems drawing the image in the Unified Canvas, click this to consolidate all of the information currently being rendered by your browser into a merged copy of the image. This lowers the resource requirements and should improve performance. * **Merge Visible** - If your browser is having performance problems drawing the image in the Unified Canvas, click this to consolidate all of the information currently being rendered by your browser into a merged copy of the image. This lowers the resource requirements and should improve performance.
## Seam Correction ## Seam Correction
When doing Inpainting or Outpainting, Invoke needs to merge the pixels generated by Stable Diffusion into your existing image. To do this, the area around the `seam` at the boundary between your image and the new generation is automatically blended to produce a seamless output. In a fully automatic process, a mask is generated to cover the seam, and then the area of the seam is Inpainted. When doing Inpainting or Outpainting, Invoke needs to merge the pixels generated by Stable Diffusion into your existing image. To do this, the area around the `seam` at the boundary between your image and the new generation is automatically blended to produce a seamless output. In a fully automatic process, a mask is generated to cover the seam, and then the area of the seam is Inpainted.
Although the default options should work well most of the time, sometimes it can help to alter the parameters that control the seam Inpainting. A wider seam and a blur setting of about 1/3 of the seam have been noted as producing consistently strong results (e.g. 96 wide and 16 blur - adds up to 32 blur with both sides). Seam strength of 0.7 is best for reducing hard seams. Although the default options should work well most of the time, sometimes it can help to alter the parameters that control the seam Inpainting. A wider seam and a blur setting of about 1/3 of the seam have been noted as producing consistently strong results (e.g. 96 wide and 16 blur - adds up to 32 blur with both sides). Seam strength of 0.7 is best for reducing hard seams.
* **Seam Size** - The size of the seam masked area. Set higher to make a larger mask around the seam. * **Seam Size** - The size of the seam masked area. Set higher to make a larger mask around the seam.
* **Seam Blur** - The size of the blur that is applied on *each* side of the masked area. * **Seam Blur** - The size of the blur that is applied on *each* side of the masked area.
* **Seam Strength** - The Image To Image Strength parameter used for the Inpainting generation that is applied to the seam area. * **Seam Strength** - The Image To Image Strength parameter used for the Inpainting generation that is applied to the seam area.
* **Seam Steps** - The number of generation steps that should be used to Inpaint the seam. * **Seam Steps** - The number of generation steps that should be used to Inpaint the seam.

View File

@ -39,7 +39,7 @@ Looking for a short version? Here's a TL;DR in 3 tables.
!!! tip "suggestions" !!! tip "suggestions"
For most use cases, `K_LMS`, `K_HEUN` and `K_DPM_2` are the best choices (the latter 2 run 0.5x as quick, but tend to converge 2x as quick as `K_LMS`). At very low steps (≤ `-s8`), `K_HEUN` and `K_DPM_2` are not recommended. Use `K_LMS` instead. For most use cases, `K_LMS`, `K_HEUN` and `K_DPM_2` are the best choices (the latter 2 run 0.5x as quick, but tend to converge 2x as quick as `K_LMS`). At very low steps (≤ `-s8`), `K_HEUN` and `K_DPM_2` are not recommended. Use `K_LMS` instead.
For variability, use `K_EULER_A` (runs 2x as quick as `K_DPM_2_A`). For variability, use `K_EULER_A` (runs 2x as quick as `K_DPM_2_A`).
--- ---

View File

@ -100,7 +100,7 @@ directory
The original Stable Diffusion version 1.4 weight file (4.27 GB) The original Stable Diffusion version 1.4 weight file (4.27 GB)
Download? [n] n Download? [n] n
[4] waifu-diffusion-1.3: [4] waifu-diffusion-1.3:
Stable Diffusion 1.4 fine tuned on anime-styled images (4.27) Stable Diffusion 1.4 fine tuned on anime-styled images (4.27 GB)
Download? [n] y Download? [n] y
[5] ft-mse-improved-autoencoder-840000: [5] ft-mse-improved-autoencoder-840000:
StabilityAI improved autoencoder fine-tuned for human faces (recommended; 335 MB) (recommended) StabilityAI improved autoencoder fine-tuned for human faces (recommended; 335 MB) (recommended)

View File

@ -64,7 +64,7 @@ steps:
It should look like the follwing: It should look like the follwing:
``` ```
Python 3.9.5 (default, Nov 23 2021, 15:27:38) Python 3.9.5 (default, Nov 23 2021, 15:27:38)
[GCC 9.3.0] on linux [GCC 9.3.0] on linux
Type "help", "copyright", "credits" or "license" for more information. Type "help", "copyright", "credits" or "license" for more information.
>>> from patchmatch import patch_match >>> from patchmatch import patch_match

View File

@ -1 +0,0 @@
020_INSTALL_MANUAL.md

View File

@ -0,0 +1,429 @@
---
title: Manual Installation
---
<figure markdown>
# :fontawesome-brands-linux: Linux | :fontawesome-brands-apple: macOS | :fontawesome-brands-windows: Windows
</figure>
!!! warning "This is for advanced Users"
who are already experienced with using conda or pip
## Introduction
You have two choices for manual installation, the [first one](#Conda_method)
based on the Anaconda3 package manager (`conda`), and
[a second one](#PIP_method) which uses basic Python virtual environment (`venv`)
commands and the PIP package manager. Both methods require you to enter commands
on the terminal, also known as the "console".
On Windows systems you are encouraged to install and use the
[Powershell](https://learn.microsoft.com/en-us/powershell/scripting/install/installing-powershell-on-windows?view=powershell-7.3),
which provides compatibility with Linux and Mac shells and nice features such as
command-line completion.
### Conda method
1. Check that your system meets the
[hardware requirements](index.md#Hardware_Requirements) and has the
appropriate GPU drivers installed. In particular, if you are a Linux user
with an AMD GPU installed, you may need to install the
[ROCm driver](https://rocmdocs.amd.com/en/latest/Installation_Guide/Installation-Guide.html).
InvokeAI does not yet support Windows machines with AMD GPUs due to the lack
of ROCm driver support on this platform.
To confirm that the appropriate drivers are installed, run `nvidia-smi` on
NVIDIA/CUDA systems, and `rocm-smi` on AMD systems. These should return
information about the installed video card.
Macintosh users with MPS acceleration, or anybody with a CPU-only system,
can skip this step.
2. You will need to install Anaconda3 and Git if they are not already
available. Use your operating system's preferred package manager, or
download the installers manually. You can find them here:
- [Anaconda3](https://www.anaconda.com/)
- [git](https://git-scm.com/downloads)
3. Clone the [InvokeAI](https://github.com/invoke-ai/InvokeAI) source code from
GitHub:
```bash
git clone https://github.com/invoke-ai/InvokeAI.git
```
This will create InvokeAI folder where you will follow the rest of the
steps.
4. Enter the newly-created InvokeAI folder:
```bash
cd InvokeAI
```
From this step forward make sure that you are working in the InvokeAI
directory!
5. Select the appropriate environment file:
We have created a series of environment files suited for different operating
systems and GPU hardware. They are located in the
`environments-and-requirements` directory:
<figure markdown>
| filename | OS |
| :----------------------: | :----------------------------: |
| environment-lin-amd.yml | Linux with an AMD (ROCm) GPU |
| environment-lin-cuda.yml | Linux with an NVIDIA CUDA GPU |
| environment-mac.yml | Macintosh |
| environment-win-cuda.yml | Windows with an NVIDA CUDA GPU |
</figure>
Choose the appropriate environment file for your system and link or copy it
to `environment.yml` in InvokeAI's top-level directory. To do so, run
following command from the repository-root:
!!! Example ""
=== "Macintosh and Linux"
!!! todo "Replace `xxx` and `yyy` with the appropriate OS and GPU codes as seen in the table above"
```bash
ln -sf environments-and-requirements/environment-xxx-yyy.yml environment.yml
```
When this is done, confirm that a file `environment.yml` has been linked in
the InvokeAI root directory and that it points to the correct file in the
`environments-and-requirements`.
```bash
ls -la
```
=== "Windows"
!!! todo " Since it requires admin privileges to create links, we will use the copy command to create your `environment.yml`"
```cmd
copy environments-and-requirements\environment-win-cuda.yml environment.yml
```
Afterwards verify that the file `environment.yml` has been created, either via the
explorer or by using the command `dir` from the terminal
```cmd
dir
```
!!! warning "Do not try to run conda on directly on the subdirectory environments file. This won't work. Instead, copy or link it to the top-level directory as shown."
6. Create the conda environment:
```bash
conda env update
```
This will create a new environment named `invokeai` and install all InvokeAI
dependencies into it. If something goes wrong you should take a look at
[troubleshooting](#troubleshooting).
7. Activate the `invokeai` environment:
In order to use the newly created environment you will first need to
activate it
```bash
conda activate invokeai
```
Your command-line prompt should change to indicate that `invokeai` is active
by prepending `(invokeai)`.
8. Pre-Load the model weights files:
!!! tip
If you have already downloaded the weights file(s) for another Stable
Diffusion distribution, you may skip this step (by selecting "skip" when
prompted) and configure InvokeAI to use the previously-downloaded files. The
process for this is described in [here](INSTALLING_MODELS.md).
```bash
python scripts/configure_invokeai.py
```
The script `configure_invokeai.py` will interactively guide you through the
process of downloading and installing the weights files needed for InvokeAI.
Note that the main Stable Diffusion weights file is protected by a license
agreement that you have to agree to. The script will list the steps you need
to take to create an account on the site that hosts the weights files,
accept the agreement, and provide an access token that allows InvokeAI to
legally download and install the weights files.
If you get an error message about a module not being installed, check that
the `invokeai` environment is active and if not, repeat step 5.
9. Run the command-line- or the web- interface:
!!! example ""
!!! warning "Make sure that the conda environment is activated, which should create `(invokeai)` in front of your prompt!"
=== "CLI"
```bash
python scripts/invoke.py
```
=== "local Webserver"
```bash
python scripts/invoke.py --web
```
=== "Public Webserver"
```bash
python scripts/invoke.py --web --host 0.0.0.0
```
If you choose the run the web interface, point your browser at
http://localhost:9090 in order to load the GUI.
10. Render away!
Browse the [features](../features/CLI.md) section to learn about all the things you
can do with InvokeAI.
Note that some GPUs are slow to warm up. In particular, when using an AMD
card with the ROCm driver, you may have to wait for over a minute the first
time you try to generate an image. Fortunately, after the warm up period
rendering will be fast.
11. Subsequently, to relaunch the script, be sure to run "conda activate
invokeai", enter the `InvokeAI` directory, and then launch the invoke
script. If you forget to activate the 'invokeai' environment, the script
will fail with multiple `ModuleNotFound` errors.
## Updating to newer versions of the script
This distribution is changing rapidly. If you used the `git clone` method
(step 5) to download the InvokeAI directory, then to update to the latest and
greatest version, launch the Anaconda window, enter `InvokeAI` and type:
```bash
git pull
conda env update
python scripts/configure_invokeai.py --no-interactive #optional
```
This will bring your local copy into sync with the remote one. The last step may
be needed to take advantage of new features or released models. The
`--no-interactive` flag will prevent the script from prompting you to download
the big Stable Diffusion weights files.
## pip Install
To install InvokeAI with only the PIP package manager, please follow these
steps:
1. Make sure you are using Python 3.9 or higher. The rest of the install
procedure depends on this:
```bash
python -V
```
2. Install the `virtualenv` tool if you don't have it already:
```bash
pip install virtualenv
```
3. From within the InvokeAI top-level directory, create and activate a virtual
environment named `invokeai`:
```bash
virtualenv invokeai
source invokeai/bin/activate
```
4. Pick the correct `requirements*.txt` file for your hardware and operating
system.
We have created a series of environment files suited for different operating
systems and GPU hardware. They are located in the
`environments-and-requirements` directory:
<figure markdown>
| filename | OS |
| :---------------------------------: | :-------------------------------------------------------------: |
| requirements-lin-amd.txt | Linux with an AMD (ROCm) GPU |
| requirements-lin-arm64.txt | Linux running on arm64 systems |
| requirements-lin-cuda.txt | Linux with an NVIDIA (CUDA) GPU |
| requirements-mac-mps-cpu.txt | Macintoshes with MPS acceleration |
| requirements-lin-win-colab-cuda.txt | Windows with an NVIDA (CUDA) GPU<br>(supports Google Colab too) |
</figure>
Select the appropriate requirements file, and make a link to it from
`requirements.txt` in the top-level InvokeAI directory. The command to do
this from the top-level directory is:
!!! example ""
=== "Macintosh and Linux"
!!! info "Replace `xxx` and `yyy` with the appropriate OS and GPU codes."
```bash
ln -sf environments-and-requirements/requirements-xxx-yyy.txt requirements.txt
```
=== "Windows"
!!! info "on Windows, admin privileges are required to make links, so we use the copy command instead"
```cmd
copy environments-and-requirements\requirements-lin-win-colab-cuda.txt requirements.txt
```
!!! warning
Please do not link or copy `environments-and-requirements/requirements-base.txt`.
This is a base requirements file that does not have the platform-specific
libraries. Also, be sure to link or copy the platform-specific file to
a top-level file named `requirements.txt` as shown here. Running pip on
a requirements file in a subdirectory will not work as expected.
When this is done, confirm that a file named `requirements.txt` has been
created in the InvokeAI root directory and that it points to the correct
file in `environments-and-requirements`.
5. Run PIP
Be sure that the `invokeai` environment is active before doing this:
```bash
pip install --prefer-binary -r requirements.txt
```
---
## Troubleshooting
Here are some common issues and their suggested solutions.
### Conda
#### Conda fails before completing `conda update`
The usual source of these errors is a package incompatibility. While we have
tried to minimize these, over time packages get updated and sometimes introduce
incompatibilities.
We suggest that you search
[Issues](https://github.com/invoke-ai/InvokeAI/issues) or the "bugs-and-support"
channel of the [InvokeAI Discord](https://discord.gg/ZmtBAhwWhy).
You may also try to install the broken packages manually using PIP. To do this,
activate the `invokeai` environment, and run `pip install` with the name and
version of the package that is causing the incompatibility. For example:
```bash
pip install test-tube==0.7.5
```
You can keep doing this until all requirements are satisfied and the `invoke.py`
script runs without errors. Please report to
[Issues](https://github.com/invoke-ai/InvokeAI/issues) what you were able to do
to work around the problem so that others can benefit from your investigation.
### Create Conda Environment fails on MacOS
If conda create environment fails with lmdb error, this is most likely caused by Clang.
Run brew config to see which Clang is installed on your Mac. If Clang isn't installed, that's causing the error.
Start by installing additional XCode command line tools, followed by brew install llvm.
```bash
xcode-select --install
brew install llvm
```
If brew config has Clang installed, update to the latest llvm and try creating the environment again.
#### `configure_invokeai.py` or `invoke.py` crashes at an early stage
This is usually due to an incomplete or corrupted Conda install. Make sure you
have linked to the correct environment file and run `conda update` again.
If the problem persists, a more extreme measure is to clear Conda's caches and
remove the `invokeai` environment:
```bash
conda deactivate
conda env remove -n invokeai
conda clean -a
conda update
```
This removes all cached library files, including ones that may have been
corrupted somehow. (This is not supposed to happen, but does anyway).
#### `invoke.py` crashes at a later stage
If the CLI or web site had been working ok, but something unexpected happens
later on during the session, you've encountered a code bug that is probably
unrelated to an install issue. Please search
[Issues](https://github.com/invoke-ai/InvokeAI/issues), file a bug report, or
ask for help on [Discord](https://discord.gg/ZmtBAhwWhy)
#### My renders are running very slowly
You may have installed the wrong torch (machine learning) package, and the
system is running on CPU rather than the GPU. To check, look at the log messages
that appear when `invoke.py` is first starting up. One of the earlier lines
should say `Using device type cuda`. On AMD systems, it will also say "cuda",
and on Macintoshes, it should say "mps". If instead the message says it is
running on "cpu", then you may need to install the correct torch library.
You may be able to fix this by installing a different torch library. Here are
the magic incantations for Conda and PIP.
!!! todo "For CUDA systems"
- conda
```bash
conda install pytorch torchvision torchaudio pytorch-cuda=11.6 -c pytorch -c nvidia
```
- pip
```bash
pip3 install torch torchvision torchaudio --extra-index-url https://download.pytorch.org/whl/cu116
```
!!! todo "For AMD systems"
- conda
```bash
conda activate invokeai
pip3 install torch torchvision torchaudio --extra-index-url https://download.pytorch.org/whl/rocm5.2/
```
- pip
```bash
pip3 install torch torchvision torchaudio --extra-index-url https://download.pytorch.org/whl/rocm5.2/
```
More information and troubleshooting tips can be found at https://pytorch.org.

View File

@ -3,10 +3,10 @@ info:
title: Stable Diffusion title: Stable Diffusion
description: |- description: |-
TODO: Description Here TODO: Description Here
Some useful links: Some useful links:
- [Stable Diffusion Dream Server](https://github.com/lstein/stable-diffusion) - [Stable Diffusion Dream Server](https://github.com/lstein/stable-diffusion)
license: license:
name: MIT License name: MIT License
url: https://github.com/lstein/stable-diffusion/blob/main/LICENSE url: https://github.com/lstein/stable-diffusion/blob/main/LICENSE
@ -36,7 +36,7 @@ paths:
description: successful operation description: successful operation
content: content:
image/png: image/png:
schema: schema:
type: string type: string
format: binary format: binary
'404': '404':
@ -66,7 +66,7 @@ paths:
description: successful operation description: successful operation
content: content:
image/png: image/png:
schema: schema:
type: string type: string
format: binary format: binary
'404': '404':

View File

@ -15,7 +15,6 @@
<body> <body>
<div id="root"></div> <div id="root"></div>
<script nomodule>!function(){var e=document,t=e.createElement("script");if(!("noModule"in t)&&"onbeforeload"in t){var n=!1;e.addEventListener("beforeload",(function(e){if(e.target===t)n=!0;else if(!e.target.hasAttribute("nomodule")||!n)return;e.preventDefault()}),!0),t.type="module",t.src=".",e.head.appendChild(t),t.remove()}}();</script> <script nomodule>!function(){var e=document,t=e.createElement("script");if(!("noModule"in t)&&"onbeforeload"in t){var n=!1;e.addEventListener("beforeload",(function(e){if(e.target===t)n=!0;else if(!e.target.hasAttribute("nomodule")||!n)return;e.preventDefault()}),!0),t.type="module",t.src=".",e.head.appendChild(t),t.remove()}}();</script>
<script nomodule crossorigin id="vite-legacy-polyfill" src="./assets/polyfills-legacy-dde3a68a.js"></script> <script nomodule crossorigin id="vite-legacy-polyfill" src="./assets/polyfills-legacy-dde3a68a.js"></script>
<script nomodule crossorigin id="vite-legacy-entry" data-src="./assets/index-legacy-b98e060c.js">System.import(document.getElementById('vite-legacy-entry').getAttribute('data-src'))</script> <script nomodule crossorigin id="vite-legacy-entry" data-src="./assets/index-legacy-b98e060c.js">System.import(document.getElementById('vite-legacy-entry').getAttribute('data-src'))</script>

View File

@ -341,7 +341,7 @@ class Args(object):
if not hasattr(cmd_switches,name) and not hasattr(arg_switches,name): if not hasattr(cmd_switches,name) and not hasattr(arg_switches,name):
raise AttributeError raise AttributeError
value_arg,value_cmd = (None,None) value_arg,value_cmd = (None,None)
try: try:
value_cmd = getattr(cmd_switches,name) value_cmd = getattr(cmd_switches,name)
@ -397,7 +397,7 @@ class Args(object):
description= description=
""" """
Generate images using Stable Diffusion. Generate images using Stable Diffusion.
Use --web to launch the web interface. Use --web to launch the web interface.
Use --from_file to load prompts from a file path or standard input ("-"). Use --from_file to load prompts from a file path or standard input ("-").
Otherwise you will be dropped into an interactive command prompt (type -h for help.) Otherwise you will be dropped into an interactive command prompt (type -h for help.)
Other command-line arguments are defaults that can usually be overridden Other command-line arguments are defaults that can usually be overridden
@ -1052,7 +1052,7 @@ def metadata_dumps(opt,
Given an Args object, returns a dict containing the keys and Given an Args object, returns a dict containing the keys and
structure of the proposed stable diffusion metadata standard structure of the proposed stable diffusion metadata standard
https://github.com/lstein/stable-diffusion/discussions/392 https://github.com/lstein/stable-diffusion/discussions/392
This is intended to be turned into JSON and stored in the This is intended to be turned into JSON and stored in the
"sd "sd
''' '''
@ -1135,7 +1135,7 @@ def args_from_png(png_file_path) -> list[Args]:
meta = ldm.invoke.pngwriter.retrieve_metadata(png_file_path) meta = ldm.invoke.pngwriter.retrieve_metadata(png_file_path)
except AttributeError: except AttributeError:
return [legacy_metadata_load({},png_file_path)] return [legacy_metadata_load({},png_file_path)]
try: try:
return metadata_loads(meta) return metadata_loads(meta)
except: except:
@ -1234,4 +1234,4 @@ def legacy_metadata_load(meta,pathname) -> Args:
opt.prompt = '' opt.prompt = ''
opt.seed = 0 opt.seed = 0
return opt return opt

View File

@ -119,11 +119,11 @@ class Concepts(object):
self.download_concept(concept_name) self.download_concept(concept_name)
path = os.path.join(self._concept_path(concept_name), file_name) path = os.path.join(self._concept_path(concept_name), file_name)
return path if os.path.exists(path) else None return path if os.path.exists(path) else None
def concept_is_downloaded(self, concept_name)->bool: def concept_is_downloaded(self, concept_name)->bool:
concept_directory = self._concept_path(concept_name) concept_directory = self._concept_path(concept_name)
return os.path.exists(concept_directory) return os.path.exists(concept_directory)
def download_concept(self,concept_name)->bool: def download_concept(self,concept_name)->bool:
repo_id = self._concept_id(concept_name) repo_id = self._concept_id(concept_name)
dest = self._concept_path(concept_name) dest = self._concept_path(concept_name)
@ -136,7 +136,7 @@ class Concepts(object):
os.makedirs(dest, exist_ok=True) os.makedirs(dest, exist_ok=True)
succeeded = True succeeded = True
bytes = 0 bytes = 0
def tally_download_size(chunk, size, total): def tally_download_size(chunk, size, total):
nonlocal bytes nonlocal bytes

View File

@ -21,7 +21,7 @@ class Embiggen(Generator):
def generate(self,prompt,iterations=1,seed=None, def generate(self,prompt,iterations=1,seed=None,
image_callback=None, step_callback=None, image_callback=None, step_callback=None,
**kwargs): **kwargs):
scope = choose_autocast(self.precision) scope = choose_autocast(self.precision)
make_image = self.get_make_image( make_image = self.get_make_image(
prompt, prompt,
@ -39,7 +39,7 @@ class Embiggen(Generator):
results.append([image, seed]) results.append([image, seed])
if image_callback is not None: if image_callback is not None:
image_callback(image, seed, prompt_in=prompt) image_callback(image, seed, prompt_in=prompt)
seed = self.new_seed() seed = self.new_seed()
return results return results
@torch.no_grad() @torch.no_grad()
@ -179,9 +179,9 @@ class Embiggen(Generator):
# Clamp values to max 255 # Clamp values to max 255
if distanceToLR > 255: if distanceToLR > 255:
distanceToLR = 255 distanceToLR = 255
#Place the pixel as invert of distance #Place the pixel as invert of distance
agradientC.putpixel((x, y), round(255 - distanceToLR)) agradientC.putpixel((x, y), round(255 - distanceToLR))
# Create alternative asymmetric diagonal corner to use on "tailing" intersections to prevent hard edges # Create alternative asymmetric diagonal corner to use on "tailing" intersections to prevent hard edges
# Fits for a left-fading gradient on the bottom side and full opacity on the right side. # Fits for a left-fading gradient on the bottom side and full opacity on the right side.
agradientAsymC = Image.new('L', (256, 256)) agradientAsymC = Image.new('L', (256, 256))

View File

@ -62,7 +62,7 @@ class Omnibus(Img2Img,Txt2Img):
if init_image is not None and mask_image is not None: # inpainting if init_image is not None and mask_image is not None: # inpainting
masked_image = init_image * (1 - mask_image) # masked image is the image masked by mask - masked regions zero masked_image = init_image * (1 - mask_image) # masked image is the image masked by mask - masked regions zero
elif init_image is not None: # img2img elif init_image is not None: # img2img
scope = choose_autocast(self.precision) scope = choose_autocast(self.precision)
@ -99,7 +99,7 @@ class Omnibus(Img2Img,Txt2Img):
device=model.device, device=model.device,
num_samples=num_samples, num_samples=num_samples,
) )
c = model.cond_stage_model.encode(batch["txt"]) c = model.cond_stage_model.encode(batch["txt"])
c_cat = list() c_cat = list()
for ck in model.concat_keys: for ck in model.concat_keys:
@ -164,10 +164,10 @@ class Omnibus(Img2Img,Txt2Img):
def sample_to_image(self, samples)->Image.Image: def sample_to_image(self, samples)->Image.Image:
gen_result = super().sample_to_image(samples).convert('RGB') gen_result = super().sample_to_image(samples).convert('RGB')
if self.pil_image is None or self.pil_mask is None: if self.pil_image is None or self.pil_mask is None:
return gen_result return gen_result
corrected_result = super(Img2Img, self).repaste_and_color_correct(gen_result, self.pil_image, self.pil_mask, self.mask_blur_radius) corrected_result = super(Img2Img, self).repaste_and_color_correct(gen_result, self.pil_image, self.pil_mask, self.mask_blur_radius)
return corrected_result return corrected_result

View File

@ -9,8 +9,8 @@ class InitImageResizer():
def resize(self,width=None,height=None) -> Image: def resize(self,width=None,height=None) -> Image:
""" """
Return a copy of the image resized to fit within Return a copy of the image resized to fit within
a box width x height. The aspect ratio is a box width x height. The aspect ratio is
maintained. If neither width nor height are provided, maintained. If neither width nor height are provided,
then returns a copy of the original image. If one or the other is then returns a copy of the original image. If one or the other is
provided, then the other will be calculated from the provided, then the other will be calculated from the
aspect ratio. aspect ratio.
@ -19,7 +19,7 @@ class InitImageResizer():
that it can be passed to img2img() that it can be passed to img2img()
""" """
im = self.image im = self.image
ar = im.width/float(im.height) ar = im.width/float(im.height)
# Infer missing values from aspect ratio # Infer missing values from aspect ratio
@ -44,7 +44,7 @@ class InitImageResizer():
# no resize necessary, but return a copy # no resize necessary, but return a copy
if im.width == width and im.height == height: if im.width == width and im.height == height:
return im.copy() return im.copy()
# otherwise resize the original image so that it fits inside the bounding box # otherwise resize the original image so that it fits inside the bounding box
resized_image = self.image.resize((rw,rh),resample=Image.Resampling.LANCZOS) resized_image = self.image.resize((rw,rh),resample=Image.Resampling.LANCZOS)
return resized_image return resized_image

View File

@ -1,5 +1,5 @@
''' '''
Manage a cache of Stable Diffusion model files for fast switching. Manage a cache of Stable Diffusion model files for fast switching.
They are moved between GPU and CPU as necessary. If CPU memory falls They are moved between GPU and CPU as necessary. If CPU memory falls
below a preset minimum, the least recently used model will be below a preset minimum, the least recently used model will be
cleared and loaded from disk when next needed. cleared and loaded from disk when next needed.
@ -51,7 +51,7 @@ class ModelCache(object):
identifier. identifier.
''' '''
return model_name in self.config return model_name in self.config
def get_model(self, model_name:str): def get_model(self, model_name:str):
''' '''
Given a model named identified in models.yaml, return Given a model named identified in models.yaml, return
@ -66,7 +66,7 @@ class ModelCache(object):
if model_name not in self.models: # make room for a new one if model_name not in self.models: # make room for a new one
self._make_cache_room() self._make_cache_room()
self.offload_model(self.current_model) self.offload_model(self.current_model)
if model_name in self.models: if model_name in self.models:
requested_model = self.models[model_name]['model'] requested_model = self.models[model_name]['model']
print(f'>> Retrieving model {model_name} from system RAM cache') print(f'>> Retrieving model {model_name} from system RAM cache')
@ -92,7 +92,7 @@ class ModelCache(object):
print(f'** restoring {self.current_model}') print(f'** restoring {self.current_model}')
self.get_model(self.current_model) self.get_model(self.current_model)
return return
self.current_model = model_name self.current_model = model_name
self._push_newest_model(model_name) self._push_newest_model(model_name)
return { return {
@ -191,7 +191,7 @@ class ModelCache(object):
omega[model_name] = config omega[model_name] = config
if clobber: if clobber:
self._invalidate_cached_model(model_name) self._invalidate_cached_model(model_name)
def _load_model(self, model_name:str): def _load_model(self, model_name:str):
"""Load and initialize the model from configuration variables passed at object creation time""" """Load and initialize the model from configuration variables passed at object creation time"""
if model_name not in self.config: if model_name not in self.config:
@ -254,7 +254,7 @@ class ModelCache(object):
model.to(self.device) model.to(self.device)
# model.to doesn't change the cond_stage_model.device used to move the tokenizer output, so set it here # model.to doesn't change the cond_stage_model.device used to move the tokenizer output, so set it here
model.cond_stage_model.device = self.device model.cond_stage_model.device = self.device
model.eval() model.eval()
for module in model.modules(): for module in model.modules():
@ -274,7 +274,7 @@ class ModelCache(object):
) )
return model, width, height, model_hash return model, width, height, model_hash
def offload_model(self, model_name:str) -> None: def offload_model(self, model_name:str) -> None:
''' '''
Offload the indicated model to CPU. Will call Offload the indicated model to CPU. Will call
@ -290,7 +290,7 @@ class ModelCache(object):
gc.collect() gc.collect()
if self._has_cuda(): if self._has_cuda():
torch.cuda.empty_cache() torch.cuda.empty_cache()
def scan_model(self, model_name, checkpoint): def scan_model(self, model_name, checkpoint):
# scan model # scan model
print(f'>> Scanning Model: {model_name}') print(f'>> Scanning Model: {model_name}')
@ -320,7 +320,7 @@ class ModelCache(object):
if least_recent_model is not None: if least_recent_model is not None:
del self.models[least_recent_model] del self.models[least_recent_model]
gc.collect() gc.collect()
def print_vram_usage(self) -> None: def print_vram_usage(self) -> None:
if self._has_cuda: if self._has_cuda:
print('>> Current VRAM usage: ','%4.2fG' % (torch.cuda.memory_allocated() / 1e9)) print('>> Current VRAM usage: ','%4.2fG' % (torch.cuda.memory_allocated() / 1e9))
@ -355,12 +355,12 @@ class ModelCache(object):
if model_name in self.stack: if model_name in self.stack:
self.stack.remove(model_name) self.stack.remove(model_name)
self.models.pop(model_name,None) self.models.pop(model_name,None)
def _model_to_cpu(self,model): def _model_to_cpu(self,model):
if self.device != 'cpu': if self.device != 'cpu':
model.cond_stage_model.device = 'cpu' model.cond_stage_model.device = 'cpu'
model.first_stage_model.to('cpu') model.first_stage_model.to('cpu')
model.cond_stage_model.to('cpu') model.cond_stage_model.to('cpu')
model.model.to('cpu') model.model.to('cpu')
return model.to('cpu') return model.to('cpu')
else: else:
@ -390,7 +390,7 @@ class ModelCache(object):
with contextlib.suppress(ValueError): with contextlib.suppress(ValueError):
self.stack.remove(model_name) self.stack.remove(model_name)
self.stack.append(model_name) self.stack.append(model_name)
def _has_cuda(self) -> bool: def _has_cuda(self) -> bool:
return self.device.type == 'cuda' return self.device.type == 'cuda'

View File

@ -10,7 +10,7 @@ class Restoration():
else: else:
print('>> GFPGAN Disabled') print('>> GFPGAN Disabled')
gfpgan = None gfpgan = None
# Load CodeFormer # Load CodeFormer
codeformer = self.load_codeformer() codeformer = self.load_codeformer()
if codeformer.codeformer_model_exists: if codeformer.codeformer_model_exists:
@ -18,7 +18,7 @@ class Restoration():
else: else:
print('>> CodeFormer Disabled') print('>> CodeFormer Disabled')
codeformer = None codeformer = None
return gfpgan, codeformer return gfpgan, codeformer
# Face Restore Models # Face Restore Models

View File

@ -14,7 +14,7 @@ class CodeFormerRestoration():
if not os.path.isabs(codeformer_dir): if not os.path.isabs(codeformer_dir):
codeformer_dir = os.path.join(Globals.root, codeformer_dir) codeformer_dir = os.path.join(Globals.root, codeformer_dir)
self.model_path = os.path.join(codeformer_dir, codeformer_model_path) self.model_path = os.path.join(codeformer_dir, codeformer_model_path)
self.codeformer_model_exists = os.path.isfile(self.model_path) self.codeformer_model_exists = os.path.isfile(self.model_path)
@ -35,9 +35,9 @@ class CodeFormerRestoration():
from ldm.invoke.restoration.codeformer_arch import CodeFormer from ldm.invoke.restoration.codeformer_arch import CodeFormer
from torchvision.transforms.functional import normalize from torchvision.transforms.functional import normalize
from PIL import Image from PIL import Image
cf_class = CodeFormer cf_class = CodeFormer
cf = cf_class( cf = cf_class(
dim_embd=512, dim_embd=512,
codebook_size=1024, codebook_size=1024,

View File

@ -119,7 +119,7 @@ class TransformerSALayer(nn.Module):
tgt_mask: Optional[Tensor] = None, tgt_mask: Optional[Tensor] = None,
tgt_key_padding_mask: Optional[Tensor] = None, tgt_key_padding_mask: Optional[Tensor] = None,
query_pos: Optional[Tensor] = None): query_pos: Optional[Tensor] = None):
# self attention # self attention
tgt2 = self.norm1(tgt) tgt2 = self.norm1(tgt)
q = k = self.with_pos_embed(tgt2, query_pos) q = k = self.with_pos_embed(tgt2, query_pos)
@ -159,7 +159,7 @@ class Fuse_sft_block(nn.Module):
@ARCH_REGISTRY.register() @ARCH_REGISTRY.register()
class CodeFormer(VQAutoEncoder): class CodeFormer(VQAutoEncoder):
def __init__(self, dim_embd=512, n_head=8, n_layers=9, def __init__(self, dim_embd=512, n_head=8, n_layers=9,
codebook_size=1024, latent_size=256, codebook_size=1024, latent_size=256,
connect_list=['32', '64', '128', '256'], connect_list=['32', '64', '128', '256'],
fix_modules=['quantize','generator']): fix_modules=['quantize','generator']):
@ -179,14 +179,14 @@ class CodeFormer(VQAutoEncoder):
self.feat_emb = nn.Linear(256, self.dim_embd) self.feat_emb = nn.Linear(256, self.dim_embd)
# transformer # transformer
self.ft_layers = nn.Sequential(*[TransformerSALayer(embed_dim=dim_embd, nhead=n_head, dim_mlp=self.dim_mlp, dropout=0.0) self.ft_layers = nn.Sequential(*[TransformerSALayer(embed_dim=dim_embd, nhead=n_head, dim_mlp=self.dim_mlp, dropout=0.0)
for _ in range(self.n_layers)]) for _ in range(self.n_layers)])
# logits_predict head # logits_predict head
self.idx_pred_layer = nn.Sequential( self.idx_pred_layer = nn.Sequential(
nn.LayerNorm(dim_embd), nn.LayerNorm(dim_embd),
nn.Linear(dim_embd, codebook_size, bias=False)) nn.Linear(dim_embd, codebook_size, bias=False))
self.channels = { self.channels = {
'16': 512, '16': 512,
'32': 256, '32': 256,
@ -221,7 +221,7 @@ class CodeFormer(VQAutoEncoder):
enc_feat_dict = {} enc_feat_dict = {}
out_list = [self.fuse_encoder_block[f_size] for f_size in self.connect_list] out_list = [self.fuse_encoder_block[f_size] for f_size in self.connect_list]
for i, block in enumerate(self.encoder.blocks): for i, block in enumerate(self.encoder.blocks):
x = block(x) x = block(x)
if i in out_list: if i in out_list:
enc_feat_dict[str(x.shape[-1])] = x.clone() enc_feat_dict[str(x.shape[-1])] = x.clone()
@ -266,7 +266,7 @@ class CodeFormer(VQAutoEncoder):
fuse_list = [self.fuse_generator_block[f_size] for f_size in self.connect_list] fuse_list = [self.fuse_generator_block[f_size] for f_size in self.connect_list]
for i, block in enumerate(self.generator.blocks): for i, block in enumerate(self.generator.blocks):
x = block(x) x = block(x)
if i in fuse_list: # fuse after i-th block if i in fuse_list: # fuse after i-th block
f_size = str(x.shape[-1]) f_size = str(x.shape[-1])
if w>0: if w>0:

View File

@ -13,19 +13,19 @@ class GFPGAN():
self, self,
gfpgan_model_path='models/gfpgan/GFPGANv1.4.pth' gfpgan_model_path='models/gfpgan/GFPGANv1.4.pth'
) -> None: ) -> None:
if not os.path.isabs(gfpgan_model_path): if not os.path.isabs(gfpgan_model_path):
gfpgan_model_path=os.path.abspath(os.path.join(Globals.root,gfpgan_model_path)) gfpgan_model_path=os.path.abspath(os.path.join(Globals.root,gfpgan_model_path))
self.model_path = gfpgan_model_path self.model_path = gfpgan_model_path
self.gfpgan_model_exists = os.path.isfile(self.model_path) self.gfpgan_model_exists = os.path.isfile(self.model_path)
if not self.gfpgan_model_exists: if not self.gfpgan_model_exists:
print('## NOT FOUND: GFPGAN model not found at ' + self.model_path) print('## NOT FOUND: GFPGAN model not found at ' + self.model_path)
return None return None
def model_exists(self): def model_exists(self):
return os.path.isfile(self.model_path) return os.path.isfile(self.model_path)
def process(self, image, strength: float, seed: str = None): def process(self, image, strength: float, seed: str = None):
if seed is not None: if seed is not None:
print(f'>> GFPGAN - Restoring Faces for image seed:{seed}') print(f'>> GFPGAN - Restoring Faces for image seed:{seed}')

View File

@ -51,7 +51,7 @@ class Outcrop(object):
color_match = True, color_match = True,
force_outpaint = True, # this just stops the warning about erased regions force_outpaint = True, # this just stops the warning about erased regions
) )
# swap sampler back # swap sampler back
self.generate.sampler = curr_sampler self.generate.sampler = curr_sampler
return result return result

View File

@ -16,7 +16,7 @@ class Outpaint(object):
def wrapped_callback(img,seed,**kwargs): def wrapped_callback(img,seed,**kwargs):
image_callback(img,seed,use_prefix=prefix,**kwargs) image_callback(img,seed,use_prefix=prefix,**kwargs)
return self.generate.prompt2image( return self.generate.prompt2image(
prompt, prompt,
seed = seed, seed = seed,

View File

@ -67,7 +67,7 @@ class ESRGAN():
# REALSRGAN expects a BGR np array; make array and flip channels # REALSRGAN expects a BGR np array; make array and flip channels
bgr_image_array = np.array(image, dtype=np.uint8)[...,::-1] bgr_image_array = np.array(image, dtype=np.uint8)[...,::-1]
output, _ = upsampler.enhance( output, _ = upsampler.enhance(
bgr_image_array, bgr_image_array,
outscale=upsampler_scale, outscale=upsampler_scale,

View File

@ -13,7 +13,7 @@ from basicsr.utils.registry import ARCH_REGISTRY
def normalize(in_channels): def normalize(in_channels):
return torch.nn.GroupNorm(num_groups=32, num_channels=in_channels, eps=1e-6, affine=True) return torch.nn.GroupNorm(num_groups=32, num_channels=in_channels, eps=1e-6, affine=True)
@torch.jit.script @torch.jit.script
def swish(x): def swish(x):
@ -210,15 +210,15 @@ class AttnBlock(nn.Module):
# compute attention # compute attention
b, c, h, w = q.shape b, c, h, w = q.shape
q = q.reshape(b, c, h*w) q = q.reshape(b, c, h*w)
q = q.permute(0, 2, 1) q = q.permute(0, 2, 1)
k = k.reshape(b, c, h*w) k = k.reshape(b, c, h*w)
w_ = torch.bmm(q, k) w_ = torch.bmm(q, k)
w_ = w_ * (int(c)**(-0.5)) w_ = w_ * (int(c)**(-0.5))
w_ = F.softmax(w_, dim=2) w_ = F.softmax(w_, dim=2)
# attend to values # attend to values
v = v.reshape(b, c, h*w) v = v.reshape(b, c, h*w)
w_ = w_.permute(0, 2, 1) w_ = w_.permute(0, 2, 1)
h_ = torch.bmm(v, w_) h_ = torch.bmm(v, w_)
h_ = h_.reshape(b, c, h, w) h_ = h_.reshape(b, c, h, w)
@ -270,18 +270,18 @@ class Encoder(nn.Module):
def forward(self, x): def forward(self, x):
for block in self.blocks: for block in self.blocks:
x = block(x) x = block(x)
return x return x
class Generator(nn.Module): class Generator(nn.Module):
def __init__(self, nf, emb_dim, ch_mult, res_blocks, img_size, attn_resolutions): def __init__(self, nf, emb_dim, ch_mult, res_blocks, img_size, attn_resolutions):
super().__init__() super().__init__()
self.nf = nf self.nf = nf
self.ch_mult = ch_mult self.ch_mult = ch_mult
self.num_resolutions = len(self.ch_mult) self.num_resolutions = len(self.ch_mult)
self.num_res_blocks = res_blocks self.num_res_blocks = res_blocks
self.resolution = img_size self.resolution = img_size
self.attn_resolutions = attn_resolutions self.attn_resolutions = attn_resolutions
self.in_channels = emb_dim self.in_channels = emb_dim
self.out_channels = 3 self.out_channels = 3
@ -315,24 +315,24 @@ class Generator(nn.Module):
blocks.append(nn.Conv2d(block_in_ch, self.out_channels, kernel_size=3, stride=1, padding=1)) blocks.append(nn.Conv2d(block_in_ch, self.out_channels, kernel_size=3, stride=1, padding=1))
self.blocks = nn.ModuleList(blocks) self.blocks = nn.ModuleList(blocks)
def forward(self, x): def forward(self, x):
for block in self.blocks: for block in self.blocks:
x = block(x) x = block(x)
return x return x
@ARCH_REGISTRY.register() @ARCH_REGISTRY.register()
class VQAutoEncoder(nn.Module): class VQAutoEncoder(nn.Module):
def __init__(self, img_size, nf, ch_mult, quantizer="nearest", res_blocks=2, attn_resolutions=[16], codebook_size=1024, emb_dim=256, def __init__(self, img_size, nf, ch_mult, quantizer="nearest", res_blocks=2, attn_resolutions=[16], codebook_size=1024, emb_dim=256,
beta=0.25, gumbel_straight_through=False, gumbel_kl_weight=1e-8, model_path=None): beta=0.25, gumbel_straight_through=False, gumbel_kl_weight=1e-8, model_path=None):
super().__init__() super().__init__()
logger = get_root_logger() logger = get_root_logger()
self.in_channels = 3 self.in_channels = 3
self.nf = nf self.nf = nf
self.n_blocks = res_blocks self.n_blocks = res_blocks
self.codebook_size = codebook_size self.codebook_size = codebook_size
self.embed_dim = emb_dim self.embed_dim = emb_dim
self.ch_mult = ch_mult self.ch_mult = ch_mult
@ -363,11 +363,11 @@ class VQAutoEncoder(nn.Module):
self.kl_weight self.kl_weight
) )
self.generator = Generator( self.generator = Generator(
self.nf, self.nf,
self.embed_dim, self.embed_dim,
self.ch_mult, self.ch_mult,
self.n_blocks, self.n_blocks,
self.resolution, self.resolution,
self.attn_resolutions self.attn_resolutions
) )
@ -432,4 +432,4 @@ class VQGANDiscriminator(nn.Module):
raise ValueError(f'Wrong params!') raise ValueError(f'Wrong params!')
def forward(self, x): def forward(self, x):
return self.main(x) return self.main(x)

View File

@ -1,5 +1,5 @@
import torch.nn as nn import torch.nn as nn
def _conv_forward_asymmetric(self, input, weight, bias): def _conv_forward_asymmetric(self, input, weight, bias):
""" """
Patch for Conv2d._conv_forward that supports asymmetric padding Patch for Conv2d._conv_forward that supports asymmetric padding
@ -27,4 +27,4 @@ def configure_model_padding(model, seamless, seamless_axes):
if hasattr(m, 'asymmetric_padding_mode'): if hasattr(m, 'asymmetric_padding_mode'):
del m.asymmetric_padding_mode del m.asymmetric_padding_mode
if hasattr(m, 'asymmetric_padding'): if hasattr(m, 'asymmetric_padding'):
del m.asymmetric_padding del m.asymmetric_padding

View File

@ -61,7 +61,7 @@ def build_opt(post_data, seed, gfpgan_model_exists):
broken = True broken = True
break break
opt.with_variations.append([seed, weight]) opt.with_variations.append([seed, weight])
if broken: if broken:
raise CanceledException raise CanceledException
@ -99,7 +99,7 @@ class DreamServer(BaseHTTPRequestHandler):
self.send_header("Content-type", "application/json") self.send_header("Content-type", "application/json")
self.end_headers() self.end_headers()
output = [] output = []
log_file = os.path.join(self.outdir, "legacy_web_log.txt") log_file = os.path.join(self.outdir, "legacy_web_log.txt")
if os.path.exists(log_file): if os.path.exists(log_file):
with open(log_file, "r") as log: with open(log_file, "r") as log:

View File

@ -45,7 +45,7 @@ def build_opt(post_data, seed, gfpgan_model_exists):
broken = True broken = True
break break
opt.with_variations.append([seed, weight]) opt.with_variations.append([seed, weight])
if broken: if broken:
raise CanceledException raise CanceledException
@ -84,7 +84,7 @@ class DreamServer(BaseHTTPRequestHandler):
self.send_header("Content-type", "application/json") self.send_header("Content-type", "application/json")
self.end_headers() self.end_headers()
output = [] output = []
log_file = os.path.join(self.outdir, "dream_web_log.txt") log_file = os.path.join(self.outdir, "dream_web_log.txt")
if os.path.exists(log_file): if os.path.exists(log_file):
with open(log_file, "r") as log: with open(log_file, "r") as log:

View File

@ -2,13 +2,13 @@
assignment of masks via text prompt using clipseg. assignment of masks via text prompt using clipseg.
Here is typical usage: Here is typical usage:
from ldm.invoke.txt2mask import Txt2Mask, SegmentedGrayscale from ldm.invoke.txt2mask import Txt2Mask, SegmentedGrayscale
from PIL import Image from PIL import Image
txt2mask = Txt2Mask(self.device) txt2mask = Txt2Mask(self.device)
segmented = txt2mask.segment(Image.open('/path/to/img.png'),'a bagel') segmented = txt2mask.segment(Image.open('/path/to/img.png'),'a bagel')
# this will return a grayscale Image of the segmented data # this will return a grayscale Image of the segmented data
grayscale = segmented.to_grayscale() grayscale = segmented.to_grayscale()
@ -45,7 +45,7 @@ class SegmentedGrayscale(object):
def __init__(self, image:Image, heatmap:torch.Tensor): def __init__(self, image:Image, heatmap:torch.Tensor):
self.heatmap = heatmap self.heatmap = heatmap
self.image = image self.image = image
def to_grayscale(self,invert:bool=False)->Image: def to_grayscale(self,invert:bool=False)->Image:
return self._rescale(Image.fromarray(np.uint8(255 - self.heatmap * 255 if invert else self.heatmap * 255))) return self._rescale(Image.fromarray(np.uint8(255 - self.heatmap * 255 if invert else self.heatmap * 255)))

View File

@ -113,7 +113,7 @@ class Sampler(object):
'ddim_sigmas_for_original_num_steps', 'ddim_sigmas_for_original_num_steps',
sigmas_for_original_sampling_steps, sigmas_for_original_sampling_steps,
) )
@torch.no_grad() @torch.no_grad()
def stochastic_encode(self, x0, t, use_original_steps=False, noise=None): def stochastic_encode(self, x0, t, use_original_steps=False, noise=None):
# fast, but does not allow for exact reconstruction # fast, but does not allow for exact reconstruction
@ -340,7 +340,7 @@ class Sampler(object):
x_dec = x_latent x_dec = x_latent
x0 = init_latent x0 = init_latent
self.prepare_to_sample(t_enc=total_steps, all_timesteps_count=all_timesteps_count, **kwargs) self.prepare_to_sample(t_enc=total_steps, all_timesteps_count=all_timesteps_count, **kwargs)
for i, step in enumerate(iterator): for i, step in enumerate(iterator):
index = total_steps - i - 1 index = total_steps - i - 1
ts = torch.full( ts = torch.full(
@ -373,7 +373,7 @@ class Sampler(object):
t_next = ts_next, t_next = ts_next,
step_count=len(self.ddim_timesteps) step_count=len(self.ddim_timesteps)
) )
x_dec, pred_x0, e_t = outs x_dec, pred_x0, e_t = outs
if img_callback: if img_callback:
img_callback(x_dec,i) img_callback(x_dec,i)
@ -385,7 +385,7 @@ class Sampler(object):
return torch.randn(shape, device=self.device) return torch.randn(shape, device=self.device)
else: else:
return x_T return x_T
def p_sample( def p_sample(
self, self,
img, img,
@ -423,10 +423,10 @@ class Sampler(object):
timesteps that will be used for sampling, depending on the t_enc in img2img. timesteps that will be used for sampling, depending on the t_enc in img2img.
''' '''
return self.ddim_timesteps[:ddim_steps] return self.ddim_timesteps[:ddim_steps]
def q_sample(self,x0,ts): def q_sample(self,x0,ts):
''' '''
Returns self.model.q_sample(x0,ts). Is overridden in the k* samplers to Returns self.model.q_sample(x0,ts). Is overridden in the k* samplers to
return self.model.inner_model.q_sample(x0,ts) return self.model.inner_model.q_sample(x0,ts)
''' '''
return self.model.q_sample(x0,ts) return self.model.q_sample(x0,ts)

View File

@ -220,7 +220,7 @@ class AttnBlock(nn.Module):
if mem_required > mem_free_total: if mem_required > mem_free_total:
steps = 2**(math.ceil(math.log(mem_required / mem_free_total, 2))) steps = 2**(math.ceil(math.log(mem_required / mem_free_total, 2)))
slice_size = q.shape[1] // steps if (q.shape[1] % steps) == 0 else q.shape[1] slice_size = q.shape[1] // steps if (q.shape[1] % steps) == 0 else q.shape[1]
else: else:
@ -228,7 +228,7 @@ class AttnBlock(nn.Module):
slice_size = 1 slice_size = 1
else: else:
slice_size = min(q.shape[1], math.floor(2**30 / (q.shape[0] * q.shape[1]))) slice_size = min(q.shape[1], math.floor(2**30 / (q.shape[0] * q.shape[1])))
for i in range(0, q.shape[1], slice_size): for i in range(0, q.shape[1], slice_size):
end = i + slice_size end = i + slice_size

View File

@ -241,7 +241,7 @@ class EmbeddingManager(nn.Module):
# both will be stored in this dictionary # both will be stored in this dictionary
for term in self.string_to_param_dict.keys(): for term in self.string_to_param_dict.keys():
term = term.strip('<').strip('>') term = term.strip('<').strip('>')
self.concepts_loaded[term] = True self.concepts_loaded[term] = True
print(f'>> Current embedding manager terms: {", ".join(self.string_to_param_dict.keys())}') print(f'>> Current embedding manager terms: {", ".join(self.string_to_param_dict.keys())}')
def _expand_directories(self, paths:list[str]): def _expand_directories(self, paths:list[str]):

View File

@ -548,7 +548,7 @@ class WeightedFrozenCLIPEmbedder(FrozenCLIPEmbedder):
#print(f"assembled tokens for '{fragments}' into tensor of shape {lerped_embeddings.shape}") #print(f"assembled tokens for '{fragments}' into tensor of shape {lerped_embeddings.shape}")
# append to batch # append to batch
batch_z = lerped_embeddings.unsqueeze(0) if batch_z is None else torch.cat([batch_z, lerped_embeddings.unsqueeze(0)], dim=1) batch_z = lerped_embeddings.unsqueeze(0) if batch_z is None else torch.cat([batch_z, lerped_embeddings.unsqueeze(0)], dim=1)
batch_tokens = tokens.unsqueeze(0) if batch_tokens is None else torch.cat([batch_tokens, tokens.unsqueeze(0)], dim=1) batch_tokens = tokens.unsqueeze(0) if batch_tokens is None else torch.cat([batch_tokens, tokens.unsqueeze(0)], dim=1)

View File

@ -98,7 +98,7 @@ def _get_paths_from_images(path):
""" """
# -------------------------------------------- # --------------------------------------------
# split large images into small images # split large images into small images
# -------------------------------------------- # --------------------------------------------
""" """

View File

@ -221,7 +221,7 @@ def rand_perlin_2d(shape, res, device, fade = lambda t: 6*t**5 - 15*t**4 + 10*t*
grid = torch.stack(torch.meshgrid(torch.arange(0, res[0], delta[0]), torch.arange(0, res[1], delta[1]), indexing='ij'), dim = -1).to(device) % 1 grid = torch.stack(torch.meshgrid(torch.arange(0, res[0], delta[0]), torch.arange(0, res[1], delta[1]), indexing='ij'), dim = -1).to(device) % 1
rand_val = torch.rand(res[0]+1, res[1]+1) rand_val = torch.rand(res[0]+1, res[1]+1)
angles = 2*math.pi*rand_val angles = 2*math.pi*rand_val
gradients = torch.stack((torch.cos(angles), torch.sin(angles)), dim = -1).to(device) gradients = torch.stack((torch.cos(angles), torch.sin(angles)), dim = -1).to(device)
@ -249,7 +249,7 @@ def ask_user(question: str, answers: list):
def debug_image(debug_image, debug_text, debug_show=True, debug_result=False, debug_status=False ): def debug_image(debug_image, debug_text, debug_show=True, debug_result=False, debug_status=False ):
if not debug_status: if not debug_status:
return return
image_copy = debug_image.copy() image_copy = debug_image.copy()
ImageDraw.Draw(image_copy).text( ImageDraw.Draw(image_copy).text(
(5, 5), (5, 5),
@ -261,4 +261,4 @@ def debug_image(debug_image, debug_text, debug_show=True, debug_result=False, de
image_copy.show() image_copy.show()
if debug_result: if debug_result:
return image_copy return image_copy

View File

@ -474,7 +474,7 @@ class ImageLogger(Callback):
self.check_frequency(check_idx) self.check_frequency(check_idx)
and hasattr( # batch_idx % self.batch_freq == 0 and hasattr( # batch_idx % self.batch_freq == 0
pl_module, 'log_images' pl_module, 'log_images'
) )
and callable(pl_module.log_images) and callable(pl_module.log_images)
and self.max_images > 0 and self.max_images > 0
): ):
@ -868,7 +868,7 @@ if __name__ == '__main__':
if hasattr(torch.backends, 'mps') and torch.backends.mps.is_available(): if hasattr(torch.backends, 'mps') and torch.backends.mps.is_available():
trainer_opt.accelerator = 'mps' trainer_opt.accelerator = 'mps'
trainer_opt.detect_anomaly = False trainer_opt.detect_anomaly = False
trainer = Trainer.from_argparse_args(trainer_opt, **trainer_kwargs) trainer = Trainer.from_argparse_args(trainer_opt, **trainer_kwargs)
trainer.logdir = logdir ### trainer.logdir = logdir ###

View File

@ -24,7 +24,7 @@ for f in filenames:
except PermissionError: except PermissionError:
sys.stderr.write(f'{f} could not be opened due to inadequate permissions\n') sys.stderr.write(f'{f} could not be opened due to inadequate permissions\n')
continue continue

View File

@ -398,7 +398,7 @@ SAMPLER_CHOICES = [
def create_argv_parser(): def create_argv_parser():
parser = argparse.ArgumentParser( parser = argparse.ArgumentParser(
description="""Generate images using Stable Diffusion. description="""Generate images using Stable Diffusion.
Use --web to launch the web interface. Use --web to launch the web interface.
Use --from_file to load prompts from a file path or standard input ("-"). Use --from_file to load prompts from a file path or standard input ("-").
Otherwise you will be dropped into an interactive command prompt (type -h for help.) Otherwise you will be dropped into an interactive command prompt (type -h for help.)
Other command-line arguments are defaults that can usually be overridden Other command-line arguments are defaults that can usually be overridden

View File

@ -8,9 +8,9 @@ from functools import partial
import torch import torch
def get_placeholder_loop(placeholder_string, embedder, use_bert): def get_placeholder_loop(placeholder_string, embedder, use_bert):
new_placeholder = None new_placeholder = None
while True: while True:
if new_placeholder is None: if new_placeholder is None:
new_placeholder = input(f"Placeholder string {placeholder_string} was already used. Please enter a replacement string: ") new_placeholder = input(f"Placeholder string {placeholder_string} was already used. Please enter a replacement string: ")
@ -21,7 +21,7 @@ def get_placeholder_loop(placeholder_string, embedder, use_bert):
if token is not None: if token is not None:
return new_placeholder, token return new_placeholder, token
def get_clip_token_for_string(tokenizer, string): def get_clip_token_for_string(tokenizer, string):
batch_encoding = tokenizer( batch_encoding = tokenizer(
string, string,
@ -37,7 +37,7 @@ def get_clip_token_for_string(tokenizer, string):
if torch.count_nonzero(tokens - 49407) == 2: if torch.count_nonzero(tokens - 49407) == 2:
return tokens[0, 1] return tokens[0, 1]
return None return None
def get_bert_token_for_string(tokenizer, string): def get_bert_token_for_string(tokenizer, string):
@ -53,16 +53,16 @@ if __name__ == "__main__":
parser = argparse.ArgumentParser() parser = argparse.ArgumentParser()
parser.add_argument( parser.add_argument(
"--root_dir", "--root_dir",
type=str, type=str,
default='.', default='.',
help="Path to the InvokeAI install directory containing 'models', 'outputs' and 'configs'." help="Path to the InvokeAI install directory containing 'models', 'outputs' and 'configs'."
) )
parser.add_argument( parser.add_argument(
"--manager_ckpts", "--manager_ckpts",
type=str, type=str,
nargs="+", nargs="+",
required=True, required=True,
help="Paths to a set of embedding managers to be merged." help="Paths to a set of embedding managers to be merged."
) )
@ -90,7 +90,7 @@ if __name__ == "__main__":
EmbeddingManager = partial(EmbeddingManager, embedder, ["*"]) EmbeddingManager = partial(EmbeddingManager, embedder, ["*"])
string_to_token_dict = {} string_to_token_dict = {}
string_to_param_dict = torch.nn.ParameterDict() string_to_param_dict = torch.nn.ParameterDict()
placeholder_to_src = {} placeholder_to_src = {}

View File

@ -54,7 +54,7 @@ function loadPriorResults() {
appendOutput(src, seed, metadata, true); appendOutput(src, seed, metadata, true);
}); });
}); });
// Load until page is full // Load until page is full
if (!priorResultsLoadState.initialized) { if (!priorResultsLoadState.initialized) {
if (document.body.scrollHeight <= window.innerHeight) { if (document.body.scrollHeight <= window.innerHeight) {

View File

@ -9,7 +9,7 @@ function toBase64(file) {
function appendOutput(src, seed, config) { function appendOutput(src, seed, config) {
let outputNode = document.createElement("figure"); let outputNode = document.createElement("figure");
let variations = config.with_variations; let variations = config.with_variations;
if (config.variation_amount > 0) { if (config.variation_amount > 0) {
variations = (variations ? variations + ',' : '') + seed + ':' + config.variation_amount; variations = (variations ? variations + ',' : '') + seed + ':' + config.variation_amount;