641 lines
18 KiB
Go
641 lines
18 KiB
Go
package cv
|
|
|
|
import (
|
|
"bufio"
|
|
"bytes"
|
|
"cmp"
|
|
"context"
|
|
"encoding/json"
|
|
"errors"
|
|
"fmt"
|
|
"io"
|
|
"io/fs"
|
|
"log"
|
|
"net/http"
|
|
"net/url"
|
|
"os"
|
|
"path"
|
|
"path/filepath"
|
|
"strconv"
|
|
"strings"
|
|
"sync"
|
|
"time"
|
|
|
|
"slices"
|
|
|
|
ch "gitea.narnian.us/lordwelch/comic-hasher"
|
|
)
|
|
|
|
type Download struct {
|
|
URL string
|
|
Dest string
|
|
IssueID string
|
|
Image []byte
|
|
}
|
|
|
|
type Issue struct {
|
|
ID int `json:"id"`
|
|
Image struct {
|
|
IconURL string `json:"icon_url,omitempty"`
|
|
MediumURL string `json:"medium_url,omitempty"`
|
|
ScreenURL string `json:"screen_url,omitempty"`
|
|
ScreenLargeURL string `json:"screen_large_url,omitempty"`
|
|
SmallURL string `json:"small_url,omitempty"`
|
|
SuperURL string `json:"super_url,omitempty"`
|
|
ThumbURL string `json:"thumb_url"`
|
|
TinyURL string `json:"tiny_url,omitempty"`
|
|
OriginalURL string `json:"original_url"`
|
|
ImageTags string `json:"image_tags"`
|
|
} `json:"image"`
|
|
Volume struct {
|
|
ID int `json:"id"`
|
|
} `json:"volume"`
|
|
}
|
|
|
|
type CVResult struct {
|
|
// Error string `json:"error"`
|
|
// Limit int `json:"limit"`
|
|
Offset int `json:"offset"`
|
|
NumberOfPageResults int `json:"number_of_page_results"`
|
|
NumberOfTotalResults int `json:"number_of_total_results"`
|
|
StatusCode int `json:"status_code"`
|
|
Results []Issue `json:"results"`
|
|
// Version string `json:"version"`
|
|
}
|
|
|
|
type CVDownloader struct {
|
|
APIKey string
|
|
JSONPath string
|
|
ImagePath string
|
|
ImageTypes []string
|
|
SendExistingImages bool
|
|
KeepDownloadedImages bool
|
|
Context context.Context
|
|
FinishedDownloadQueue chan Download
|
|
|
|
fileList []fs.DirEntry
|
|
totalResults int
|
|
imageWG sync.WaitGroup
|
|
downloadQueue chan *CVResult
|
|
imageDownloads chan download
|
|
notFound chan download
|
|
chdb ch.CHDB
|
|
}
|
|
|
|
var (
|
|
ErrQuit = errors.New("Quit")
|
|
ErrInvalidPage = errors.New("Invalid ComicVine Page")
|
|
)
|
|
|
|
func (c *CVDownloader) readJson() ([]*CVResult, error) {
|
|
var issues []*CVResult
|
|
for _, file_entry := range c.fileList {
|
|
if c.hasQuit() {
|
|
return nil, ErrQuit
|
|
}
|
|
result, err := c.loadIssues(file_entry)
|
|
if err != nil {
|
|
if err == ErrInvalidPage {
|
|
continue
|
|
}
|
|
return issues, err
|
|
}
|
|
|
|
c.totalResults = max(result.NumberOfTotalResults, c.totalResults)
|
|
issues = append(issues, result)
|
|
}
|
|
return issues, nil
|
|
}
|
|
func (c *CVDownloader) loadIssues(file_entry fs.DirEntry) (*CVResult, error) {
|
|
tmp := &CVResult{}
|
|
file, err := os.Open(filepath.Join(c.JSONPath, file_entry.Name()))
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
bytes, err := io.ReadAll(file)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
err = json.Unmarshal(bytes, tmp)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if getOffset(file_entry) != tmp.Offset {
|
|
return nil, ErrInvalidPage
|
|
}
|
|
return tmp, nil
|
|
}
|
|
|
|
func Get(ctx context.Context, url string) (*http.Response, error, func()) {
|
|
ctx, cancel := context.WithTimeout(ctx, time.Second*10)
|
|
req, err := http.NewRequestWithContext(ctx, "GET", url, nil)
|
|
if err != nil {
|
|
return nil, err, cancel
|
|
}
|
|
resp, err := http.DefaultClient.Do(req)
|
|
return resp, err, cancel
|
|
}
|
|
|
|
func getOffset(entry fs.DirEntry) int {
|
|
i, _ := strconv.Atoi(entry.Name()[3 : len(entry.Name())-1-4])
|
|
return i
|
|
}
|
|
|
|
// updateIssues c.downloadQueue must not be closed before this function has returned
|
|
func (c *CVDownloader) updateIssues() {
|
|
base_url, err := url.Parse("https://comicvine.gamespot.com/api/issues/?sort=date_added,id:asc&format=json&field_list=id,image,volume")
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
|
|
query := base_url.Query()
|
|
query.Add("api_key", c.APIKey)
|
|
base_url.RawQuery = query.Encode()
|
|
c.totalResults = max(c.totalResults, 1)
|
|
failCount := 0
|
|
prev := -1
|
|
offset := 0
|
|
retry := func(url string, err error) bool {
|
|
if errors.Is(err, context.Canceled) {
|
|
log.Println("Server closed")
|
|
return false
|
|
}
|
|
log.Printf("Failed to download %#v at offset %v: %v Attempt #%d", url, offset, err, failCount+1)
|
|
if prev == offset {
|
|
sleepTime := time.Second * 36
|
|
if failCount > 2 {
|
|
sleepTime = time.Minute * 10
|
|
}
|
|
log.Println("This page failed to download, lets wait for", sleepTime, "and hope it works")
|
|
select {
|
|
case <-c.Context.Done(): // allows us to return immediately even during a timeout
|
|
return false
|
|
case <-time.After(sleepTime):
|
|
}
|
|
}
|
|
prev = offset
|
|
failCount += 1
|
|
offset -= 100
|
|
return failCount < 15
|
|
}
|
|
for offset = 0; offset < c.totalResults; offset += 100 {
|
|
if c.hasQuit() {
|
|
return
|
|
}
|
|
if offset/100 < len(c.fileList) {
|
|
if getOffset(c.fileList[offset/100]) == offset { // If it's in order and it's not missing it should be here
|
|
if issue, err := c.loadIssues(c.fileList[offset/100]); err == nil && issue != nil {
|
|
c.totalResults = max(c.totalResults, issue.NumberOfTotalResults)
|
|
prev = -1
|
|
failCount = 0
|
|
// When canceled one of these will randomly be chosen, c.downloadQueue won't be closed until after this function returns
|
|
select {
|
|
case <-c.Context.Done():
|
|
case c.downloadQueue <- issue:
|
|
}
|
|
continue
|
|
} else {
|
|
log.Println("Failed to read page at offset ", offset, err)
|
|
os.Remove(filepath.Join(c.JSONPath, c.fileList[offset/100].Name()))
|
|
c.fileList = slices.Delete(c.fileList, offset/100, (offset/100)+1)
|
|
}
|
|
}
|
|
log.Printf("Expected Offset %d got Offset %d", offset, getOffset(c.fileList[offset/100]))
|
|
}
|
|
index, found := slices.BinarySearchFunc(c.fileList, offset, func(a fs.DirEntry, b int) int {
|
|
ai, _ := strconv.Atoi(a.Name()[3 : len(a.Name())-1-4])
|
|
return cmp.Compare(ai, b)
|
|
})
|
|
if found {
|
|
if issue, err := c.loadIssues(c.fileList[index]); err != nil && issue != nil {
|
|
prev = -1
|
|
failCount = 0
|
|
// When canceled one of these will randomly be chosen, c.downloadQueue won't be closed until after this function returns
|
|
select {
|
|
case <-c.Context.Done():
|
|
case c.downloadQueue <- issue:
|
|
}
|
|
continue
|
|
} else {
|
|
log.Println("Failed to read page at offset ", offset, err)
|
|
os.Remove(filepath.Join(c.JSONPath, c.fileList[offset/100].Name()))
|
|
c.fileList = slices.Delete(c.fileList, offset/100, (offset/100)+1)
|
|
}
|
|
}
|
|
|
|
log.Println("Starting download at offset", offset)
|
|
issue := &CVResult{}
|
|
URI := (*base_url)
|
|
query = base_url.Query()
|
|
query.Add("offset", strconv.Itoa(offset))
|
|
URI.RawQuery = query.Encode()
|
|
|
|
select {
|
|
case <-c.Context.Done(): // allows us to return immediately even during a timeout
|
|
return
|
|
case <-time.After(10 * time.Second):
|
|
}
|
|
resp, err, cancelDownloadCTX := Get(c.Context, URI.String())
|
|
if err != nil {
|
|
cancelDownloadCTX()
|
|
if retry(URI.String(), err) {
|
|
continue
|
|
}
|
|
return
|
|
}
|
|
if resp.StatusCode != 200 {
|
|
cancelDownloadCTX()
|
|
if retry(URI.String(), nil) {
|
|
_ = resp.Body.Close()
|
|
continue
|
|
}
|
|
log.Println("Failed to download this page, we'll wait for an hour to see if it clears up")
|
|
select {
|
|
case <-c.Context.Done(): // allows us to return immediately even during a timeout
|
|
_ = resp.Body.Close()
|
|
return
|
|
case <-time.After(1 * time.Hour):
|
|
}
|
|
}
|
|
file, err := os.Create(filepath.Join(c.JSONPath, "cv-"+strconv.Itoa(offset)+".json"))
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
body := io.TeeReader(resp.Body, file)
|
|
err = json.NewDecoder(bufio.NewReader(body)).Decode(issue)
|
|
_ = resp.Body.Close()
|
|
_ = file.Close()
|
|
if err != nil || issue.Offset != offset {
|
|
os.Remove(filepath.Join(c.JSONPath, "cv-"+strconv.Itoa(offset)+".json"))
|
|
cancelDownloadCTX()
|
|
if retry(URI.String(), err) {
|
|
continue
|
|
}
|
|
return
|
|
}
|
|
cancelDownloadCTX()
|
|
if issue.NumberOfTotalResults > c.totalResults {
|
|
c.totalResults = issue.NumberOfTotalResults
|
|
}
|
|
prev = -1
|
|
failCount = 0
|
|
// When canceled one of these will randomly be chosen, c.downloadQueue won't be closed until after this function returns
|
|
select {
|
|
case <-c.Context.Done():
|
|
return
|
|
case c.downloadQueue <- issue:
|
|
}
|
|
log.Printf("Downloaded %s/cv-%v.json", c.JSONPath, offset)
|
|
}
|
|
}
|
|
|
|
type download struct {
|
|
url string
|
|
dest string
|
|
offset int
|
|
volumeID int
|
|
issueID int
|
|
finished bool
|
|
}
|
|
|
|
func (c *CVDownloader) start_downloader() {
|
|
for i := range 5 {
|
|
go func() {
|
|
log.Println("starting downloader", i)
|
|
for dl := range c.imageDownloads {
|
|
if c.hasQuit() {
|
|
c.imageWG.Done()
|
|
continue // We must continue so that c.imageWG will complete otherwise it will hang forever
|
|
}
|
|
if dl.finished {
|
|
|
|
select {
|
|
case <-c.Context.Done():
|
|
c.imageWG.Done()
|
|
continue
|
|
case c.FinishedDownloadQueue <- Download{
|
|
URL: dl.url,
|
|
Dest: dl.dest,
|
|
IssueID: strconv.Itoa(dl.issueID),
|
|
}:
|
|
c.imageWG.Done()
|
|
}
|
|
continue
|
|
}
|
|
dir := filepath.Dir(dl.dest)
|
|
resp, err, cancelDownload := Get(c.Context, dl.url)
|
|
if err != nil {
|
|
cancelDownload()
|
|
log.Println("Failed to download", dl.volumeID, "/", dl.issueID, dl.url, err)
|
|
c.imageWG.Done()
|
|
continue
|
|
}
|
|
cleanup := func() {
|
|
resp.Body.Close()
|
|
cancelDownload()
|
|
c.imageWG.Done()
|
|
}
|
|
if resp.StatusCode == 404 {
|
|
|
|
c.notFound <- dl
|
|
cleanup()
|
|
continue
|
|
}
|
|
if resp.StatusCode != 200 {
|
|
log.Println("Failed to download", dl.url, resp.StatusCode)
|
|
cleanup()
|
|
continue
|
|
}
|
|
|
|
if c.KeepDownloadedImages {
|
|
_ = os.MkdirAll(dir, 0o755)
|
|
image, err := os.Create(dl.dest)
|
|
if err != nil {
|
|
log.Println("Unable to create image file", dl.dest, err)
|
|
os.Remove(dl.dest)
|
|
image.Close()
|
|
cleanup()
|
|
continue
|
|
}
|
|
log.Println("downloading", dl.dest)
|
|
_, err = io.Copy(image, resp.Body)
|
|
image.Close()
|
|
if err != nil {
|
|
log.Println("Failed when downloading image", err)
|
|
os.Remove(dl.dest)
|
|
cleanup()
|
|
continue
|
|
}
|
|
|
|
c.FinishedDownloadQueue <- Download{
|
|
URL: dl.url,
|
|
Dest: dl.dest,
|
|
IssueID: strconv.Itoa(dl.issueID),
|
|
}
|
|
|
|
} else {
|
|
image := &bytes.Buffer{}
|
|
log.Println("downloading", dl.dest)
|
|
_, err = io.Copy(image, resp.Body)
|
|
if err != nil {
|
|
log.Println("Failed when downloading image", err)
|
|
cleanup()
|
|
os.Remove(dl.dest)
|
|
continue
|
|
}
|
|
|
|
c.FinishedDownloadQueue <- Download{
|
|
URL: dl.url,
|
|
Dest: dl.dest,
|
|
IssueID: strconv.Itoa(dl.issueID),
|
|
Image: image.Bytes(),
|
|
}
|
|
}
|
|
cleanup()
|
|
}
|
|
}()
|
|
}
|
|
}
|
|
|
|
func (c *CVDownloader) handleNotFound() {
|
|
for failedDownload := range c.notFound {
|
|
c.chdb.AddURL(failedDownload.url)
|
|
log.Printf("Not found: volumeID: %d issueID: %d Offset: %d URL: %s\n", failedDownload.volumeID, failedDownload.issueID, failedDownload.offset, failedDownload.url)
|
|
}
|
|
}
|
|
|
|
func (c *CVDownloader) downloadImages() {
|
|
defer func() {
|
|
log.Println("Waiting for final images to complete download")
|
|
c.imageWG.Wait()
|
|
}()
|
|
go c.start_downloader()
|
|
|
|
go c.handleNotFound()
|
|
added := 0
|
|
for list := range c.downloadQueue {
|
|
log.Printf("Checking downloads at offset %v\r", list.Offset)
|
|
for _, issue := range list.Results {
|
|
type i struct {
|
|
url string
|
|
name string
|
|
}
|
|
imageURLs := []i{{issue.Image.IconURL, "icon_url"}, {issue.Image.MediumURL, "medium_url"}, {issue.Image.ScreenURL, "screen_url"}, {issue.Image.ScreenLargeURL, "screen_large_url"}, {issue.Image.SmallURL, "small_url"}, {issue.Image.SuperURL, "super_url"}, {issue.Image.ThumbURL, "thumb_url"}, {issue.Image.TinyURL, "tiny_url"}, {issue.Image.OriginalURL, "original_url"}}
|
|
for _, image := range imageURLs {
|
|
if c.hasQuit() {
|
|
return
|
|
}
|
|
if len(c.ImageTypes) > 0 && !slices.Contains(c.ImageTypes, image.name) {
|
|
continue
|
|
}
|
|
if c.chdb.CheckURL(image.url) {
|
|
log.Printf("Skipping known bad url %s", image.url)
|
|
continue
|
|
}
|
|
|
|
uri, err := url.ParseRequestURI(image.url)
|
|
if err != nil {
|
|
c.notFound <- download{
|
|
url: image.url,
|
|
offset: list.Offset,
|
|
volumeID: issue.Volume.ID,
|
|
issueID: issue.ID,
|
|
finished: true,
|
|
}
|
|
}
|
|
ext := strings.TrimSuffix(strings.ToLower(path.Ext(uri.Path)), "~original")
|
|
if ext == "" || (len(ext) > 4 && !slices.Contains([]string{".avif", ".webp", ".tiff", ".heif"}, ext)) {
|
|
ext = ".jpg"
|
|
}
|
|
dir := filepath.Join(c.ImagePath, strconv.Itoa(issue.Volume.ID), strconv.Itoa(issue.ID))
|
|
path := filepath.Join(dir, image.name+ext)
|
|
|
|
if c.chdb.PathDownloaded(path) {
|
|
if _, err = os.Stat(path); c.SendExistingImages && err == nil {
|
|
// We don't add to the count of added as these should be processed immediately
|
|
log.Printf("Sending Existing image %v/%v %v", issue.Volume.ID, issue.ID, path)
|
|
c.imageWG.Add(1)
|
|
c.imageDownloads <- download{
|
|
url: image.url,
|
|
dest: path,
|
|
offset: list.Offset,
|
|
volumeID: issue.Volume.ID,
|
|
issueID: issue.ID,
|
|
finished: true,
|
|
}
|
|
}
|
|
continue // If it exists assume it is fine, adding some basic verification might be a good idea later
|
|
}
|
|
added++
|
|
|
|
c.imageWG.Add(1)
|
|
c.imageDownloads <- download{
|
|
url: image.url,
|
|
dest: path,
|
|
offset: list.Offset,
|
|
volumeID: issue.Volume.ID,
|
|
issueID: issue.ID,
|
|
}
|
|
}
|
|
if added > 200 {
|
|
// On a clean single image type run each page would have 100 downloads of a single cover type but stuff happens so we only wait once we have sent 200 to the queue
|
|
log.Println("waiting for", added, "downloads at offset", list.Offset)
|
|
beforeWait := time.Now()
|
|
c.imageWG.Wait()
|
|
waited := time.Since(beforeWait)
|
|
added = 0
|
|
// If we had to wait for the arbitrarily picked time of 7.4 seconds it means we had a backed up queue (slow hashing can also cause it to wait longer), lets wait to give the CV servers a break
|
|
if waited > time.Duration(7.4*float64(time.Second)) {
|
|
t := 10 * time.Second
|
|
log.Println("Waiting for", t, "at offset", list.Offset, "had to wait for", waited)
|
|
select {
|
|
case <-c.Context.Done(): // allows us to return immediately even during a timeout
|
|
return
|
|
case <-time.After(t):
|
|
}
|
|
} else {
|
|
// Things are too fast we can't depend CV being slow to manage our download speed
|
|
// We sleep for 3 seconds so we don't overload CV
|
|
time.Sleep(3 * time.Second)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func (c *CVDownloader) cleanBadURLs() error {
|
|
|
|
var indexesToRemove []int
|
|
list:
|
|
for i, jsonFile := range c.fileList {
|
|
list, err := c.loadIssues(jsonFile)
|
|
if err != nil {
|
|
indexesToRemove = append(indexesToRemove, i)
|
|
os.Remove(filepath.Join(c.JSONPath, jsonFile.Name()))
|
|
continue
|
|
}
|
|
for _, issue := range list.Results {
|
|
for _, url := range []string{issue.Image.IconURL, issue.Image.MediumURL, issue.Image.ScreenURL, issue.Image.ScreenLargeURL, issue.Image.SmallURL, issue.Image.SuperURL, issue.Image.ThumbURL, issue.Image.TinyURL, issue.Image.OriginalURL} {
|
|
if c.hasQuit() {
|
|
return ErrQuit
|
|
}
|
|
if c.chdb.CheckURL(url) {
|
|
indexesToRemove = append(indexesToRemove, i)
|
|
if err := os.Remove(filepath.Join(c.JSONPath, jsonFile.Name())); err != nil {
|
|
return err
|
|
}
|
|
// We've removed the entire page, lets see if the new url works
|
|
continue list
|
|
}
|
|
}
|
|
}
|
|
}
|
|
slices.Reverse(indexesToRemove)
|
|
for _, i := range indexesToRemove {
|
|
c.fileList = slices.Delete(c.fileList, i, min(i+1, len(c.fileList)-1))
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (c *CVDownloader) hasQuit() bool {
|
|
select {
|
|
case <-c.Context.Done():
|
|
return true
|
|
default:
|
|
return false
|
|
}
|
|
}
|
|
|
|
func (c *CVDownloader) cleanDirs() {
|
|
_ = filepath.WalkDir(c.ImagePath, func(path string, d fs.DirEntry, err error) error {
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if d.IsDir() {
|
|
path, _ = filepath.Abs(path)
|
|
err := ch.RmdirP(path)
|
|
// The error is only for the first path value. EG ch.RmdirP("/test/t") will only return the error for os.Remove("/test/t") not os.Remove("test")
|
|
if err == nil {
|
|
return filepath.SkipDir
|
|
}
|
|
}
|
|
return nil
|
|
})
|
|
}
|
|
|
|
func NewCVDownloader(ctx context.Context, chdb ch.CHDB, workPath, APIKey string, imageTypes []string, keepDownloadedImages, sendExistingImages bool, finishedDownloadQueue chan Download) *CVDownloader {
|
|
return &CVDownloader{
|
|
Context: ctx,
|
|
JSONPath: filepath.Join(workPath, "_json"),
|
|
ImagePath: filepath.Join(workPath, "_image"),
|
|
APIKey: APIKey,
|
|
downloadQueue: make(chan *CVResult, 1000),
|
|
imageDownloads: make(chan download, 250),
|
|
notFound: make(chan download, 100),
|
|
FinishedDownloadQueue: finishedDownloadQueue,
|
|
SendExistingImages: sendExistingImages,
|
|
KeepDownloadedImages: keepDownloadedImages,
|
|
ImageTypes: imageTypes,
|
|
chdb: chdb,
|
|
}
|
|
}
|
|
|
|
func DownloadCovers(c *CVDownloader) {
|
|
var (
|
|
err error
|
|
)
|
|
os.MkdirAll(c.JSONPath, 0o777)
|
|
f, _ := os.Create(filepath.Join(c.ImagePath, ".keep"))
|
|
f.Close()
|
|
c.cleanDirs()
|
|
c.fileList, err = os.ReadDir(c.JSONPath)
|
|
if err != nil {
|
|
panic(fmt.Errorf("Unable to open path for json files: %w", err))
|
|
}
|
|
|
|
slices.SortFunc(c.fileList, func(x, y fs.DirEntry) int {
|
|
xi, _ := strconv.Atoi(x.Name()[3 : len(x.Name())-1-4])
|
|
yi, _ := strconv.Atoi(y.Name()[3 : len(y.Name())-1-4])
|
|
return cmp.Compare(xi, yi)
|
|
})
|
|
if len(c.fileList) > 0 {
|
|
last_file := c.fileList[len(c.fileList)-1].Name()
|
|
c.totalResults, _ = strconv.Atoi(last_file[3 : len(last_file)-1-4])
|
|
}
|
|
c.totalResults += 100
|
|
log.Println("Reading json")
|
|
log.Println("Number of pages", len(c.fileList), "Expected Pages:", c.totalResults/100)
|
|
log.Println("Updating issues now")
|
|
|
|
dwg := sync.WaitGroup{}
|
|
dwg.Add(1)
|
|
go func() {
|
|
c.downloadImages()
|
|
dwg.Done()
|
|
}()
|
|
|
|
c.updateIssues()
|
|
issueCount := len(c.fileList) * 100
|
|
|
|
log.Println("Number of issues", issueCount, " expected:", c.totalResults)
|
|
|
|
close(c.downloadQueue) // sends only happen in c.updateIssues which has already been called
|
|
// We don't drain here as we want to process them
|
|
|
|
log.Println("Waiting for downloaders")
|
|
dwg.Wait()
|
|
close(c.imageDownloads)
|
|
for range c.imageDownloads {
|
|
}
|
|
close(c.notFound)
|
|
for range c.notFound {
|
|
}
|
|
|
|
// We drain this at the end because we need to wait for the images to download
|
|
for range c.downloadQueue {
|
|
}
|
|
|
|
log.Println("Completed downloading images")
|
|
}
|