Add uploader which can deal with errors

This commit is contained in:
Pieter Vander Vennet 2023-01-18 23:05:55 +01:00
parent ff134ebfba
commit 5aa1413078
5 changed files with 131 additions and 62 deletions

View file

@ -4,6 +4,11 @@ export interface MapCompleteUsageOverview {
showTopContributors?: boolean, showTopContributors?: boolean,
showTopThemes?: boolean, showTopThemes?: boolean,
/**
* Add a content warning to all posts
*/
contentWarning?: string,
/** /**
* Term to use in 'created/moved/deleted one point' * Term to use in 'created/moved/deleted one point'
*/ */

75
src/ImageUploader.ts Normal file
View file

@ -0,0 +1,75 @@
import {ChangeSetData} from "./OsmCha";
import MastodonPoster from "./Mastodon";
import OsmUserInfo from "./OsmUserInfo";
import ImgurAttribution from "./ImgurAttribution";
import Utils from "./Utils";
import Config from "./Config";
export default class ImageUploader {
private readonly _imageQueue: { image: string; changeset: ChangeSetData }[];
private _poster: MastodonPoster;
private _authors: string[] = []
private readonly _globalConfig: Config
constructor(imageQueue: {image: string, changeset: ChangeSetData}[], poster: MastodonPoster, config: Config) {
this._imageQueue = imageQueue;
this._poster = poster;
this._globalConfig = config
}
public getCurrentAuthors(){
return [...this._authors]
}
public async attemptToUpload(targetcount: number): Promise<string[]>{
const mediaIds = []
while(mediaIds.length < targetcount && this._imageQueue.length >0){
const first = this._imageQueue[0]
try {
const id = await this.uploadFirstImage()
mediaIds.push(id)
}catch (e) {
console.error("Could not upload image! ", first.image, e)
console.log("Trying again")
try {
const id = await this.uploadFirstImage()
mediaIds.push(id)
}catch (e) {
console.error("Retry could not upload image! ", first.image, e)
}
}
}
return mediaIds
}
private async uploadFirstImage(): Promise<string>{
const image = this._imageQueue.shift()
const cs = image.changeset.properties
let authorName = cs.user
try {
const authorInfo = new OsmUserInfo(Number(cs.uid), this._globalConfig)
authorName = (await authorInfo.GetMastodonUsername(this._poster)) ?? cs.user
} catch (e) {
console.log("Could not fetch more info about contributor", authorName, cs.uid, "due to", e)
}
if (this._globalConfig.mastodonAuth.dryrun) {
console.log("Not uploading/downloading image:" + image.image + " dryrun")
this._authors.push(authorName)
return "dummy_id"
}
console.log("Fetching attribution for", image.image)
const attribution = await ImgurAttribution.DownloadAttribution(image.image)
const id = image.image.substring(image.image.lastIndexOf("/") + 1)
const path = this._globalConfig.cacheDir + "/image_" + id
await Utils.DownloadBlob(image.image, path)
const mediaId = await this._poster.uploadImage(path, "Image taken by " + authorName + ", available under " + attribution.license + ". It is made with the thematic map " + image.changeset.properties.theme + " in changeset https://openstreetmap.org/changeset/" + image.changeset.id)
this._authors.push(authorName)
return mediaId
}
}

View file

@ -76,7 +76,7 @@ export default class MastodonPoster {
if (descrParts.indexOf("#nobot") >= 0 || descrParts.indexOf("#nomapcompletebot") >= 0) { if (descrParts.indexOf("#nobot") >= 0 || descrParts.indexOf("#nomapcompletebot") >= 0) {
return true return true
} }
const nobot = info.fields.find(f => f.name === "nobot").value const nobot = info.fields.find(f => f.name === "nobot")?.value ?? ""
if (nobot.toLowerCase() === "yes" || nobot.toLowerCase() === "true") { if (nobot.toLowerCase() === "yes" || nobot.toLowerCase() === "true") {
return true return true
} }

View file

@ -22,7 +22,7 @@ export default class OsmUserInfo {
private _userData: UserInfo = undefined private _userData: UserInfo = undefined
private readonly _cachingPath: string | undefined; private readonly _cachingPath: string | undefined;
constructor(userId: number, options?: constructor(userId: number, options:
{ {
osmBackend?: string, osmBackend?: string,
cacheDir?: string cacheDir?: string

View file

@ -4,8 +4,8 @@ import {ChangeSetData} from "./OsmCha";
import OsmUserInfo from "./OsmUserInfo"; import OsmUserInfo from "./OsmUserInfo";
import Config, {MapCompleteUsageOverview} from "./Config"; import Config, {MapCompleteUsageOverview} from "./Config";
import MastodonPoster from "./Mastodon"; import MastodonPoster from "./Mastodon";
import ImgurAttribution from "./ImgurAttribution";
import Overpass from "./Overpass"; import Overpass from "./Overpass";
import ImageUploader from "./ImageUploader";
type ImageInfo = { image: string, changeset: ChangeSetData } type ImageInfo = { image: string, changeset: ChangeSetData }
@ -35,13 +35,15 @@ export class Postbuilder {
} }
getStatisticsFor(changesetsMade?: ChangeSetData[]): { total: number, getStatisticsFor(changesetsMade?: ChangeSetData[]): {
total: number,
answered?: number, answered?: number,
created?: number, created?: number,
addImage?: number, addImage?: number,
deleted: number, deleted: number,
moved?: number, moved?: number,
summaryText?: string } { summaryText?: string
} {
const stats: Record<string, number> = {} const stats: Record<string, number> = {}
changesetsMade ??= this._changesetsMade changesetsMade ??= this._changesetsMade
@ -63,11 +65,11 @@ export class Postbuilder {
const linkedImages = stats["link-image"] const linkedImages = stats["link-image"]
const poi = this._config.poiName ?? "point" const poi = this._config.poiName ?? "point"
const pois = this._config.poisName ?? "points" const pois = this._config.poisName ?? "points"
if(create){ if (create) {
if (create == 1) { if (create == 1) {
overview.push("added one "+poi) overview.push("added one " + poi)
} else { } else {
overview.push("added " + create +" "+pois) overview.push("added " + create + " " + pois)
} }
} }
if (answer) { if (answer) {
@ -87,17 +89,17 @@ export class Postbuilder {
if (move) { if (move) {
if (move == 1) { if (move == 1) {
overview.push("moved one "+poi) overview.push("moved one " + poi)
} else { } else {
overview.push("moved " + move + " "+pois) overview.push("moved " + move + " " + pois)
} }
} }
if (deleted) { if (deleted) {
if (deleted == 1) { if (deleted == 1) {
overview.push("deleted one "+poi) overview.push("deleted one " + poi)
} else { } else {
overview.push("deleted " + deleted + " "+pois) overview.push("deleted " + deleted + " " + pois)
} }
} }
@ -162,11 +164,15 @@ export class Postbuilder {
return `${contribCountStr} ${statistics.summaryText} on https://mapcomplete.osm.be/${theme}` return `${contribCountStr} ${statistics.summaryText} on https://mapcomplete.osm.be/${theme}`
} }
public selectImages(images: ImageInfo[], targetCount: number = 4): /**
* Creates a new list of images, sorted by priority.
* It tries to order them in such a way that the number of contributors is as big as possible.
* However, it is biased to select pictures from certain themes too
* @param images
*/
public selectImages(images: ImageInfo[]):
ImageInfo[] { ImageInfo[] {
if (images.length <= targetCount) {
return images
}
const themeBonus = { const themeBonus = {
climbing: 1, climbing: 1,
rainbow_crossings: 1, rainbow_crossings: 1,
@ -184,7 +190,7 @@ export class Postbuilder {
const alreadyEncounteredUid = new Map<string, number>() const alreadyEncounteredUid = new Map<string, number>()
const result: ImageInfo[] = [] const result: ImageInfo[] = []
for (let i = 0; i < targetCount; i++) { for (let i = 0; i < images.length; i++) {
let bestImageScore: number = -999999999 let bestImageScore: number = -999999999
let bestImageOptions: ImageInfo[] = [] let bestImageOptions: ImageInfo[] = []
@ -228,17 +234,18 @@ export class Postbuilder {
let lastPostId: string = undefined let lastPostId: string = undefined
if(this._config.report){ if (this._config.report) {
const report = this._config.report const report = this._config.report
const overpass = new Overpass(report) const overpass = new Overpass(report)
const data = await overpass.query() const data = await overpass.query()
const total = data.elements.length const total = data.elements.length
const date = data.osm3s.timestamp_osm_base.substring(0, 10) const date = data.osm3s.timestamp_osm_base.substring(0, 10)
lastPostId = (await this._poster.writeMessage( lastPostId = (await this._poster.writeMessage(
report.post.replace(/{total}/g, ""+total).replace(/{date}/g, date) report.post.replace(/{total}/g, "" + total).replace(/{date}/g, date),
{spoilerText: this._config.contentWarning}
)).id )).id
} }
const perContributor = new Histogram(changesets, cs => cs.properties.uid) const perContributor = new Histogram(changesets, cs => cs.properties.uid)
const topContributors = perContributor.sortedByCount({ const topContributors = perContributor.sortedByCount({
countMethod: cs => { countMethod: cs => {
@ -254,12 +261,12 @@ export class Postbuilder {
const totalStats = this.getStatisticsFor() const totalStats = this.getStatisticsFor()
const { const {
totalImagesCreated, totalImagesCreated,
attachmentIds, randomImages,
imgAuthors,
totalImageContributorCount totalImageContributorCount
} = await this.prepareImages(changesets, 12) } = await this.prepareImages(changesets)
const imageUploader = new ImageUploader(randomImages, this._poster, this._globalConfig)
let timePeriod = "Yesterday" let timePeriod = "Yesterday"
if (this._config.numberOfDays > 1) { if (this._config.numberOfDays > 1) {
@ -276,7 +283,7 @@ export class Postbuilder {
const uid = topContributor.key const uid = topContributor.key
const changesetsMade = perContributor.get(uid) const changesetsMade = perContributor.get(uid)
try { try {
const userInfo = new OsmUserInfo(Number(uid)) const userInfo = new OsmUserInfo(Number(uid), this._globalConfig)
const {nobot} = await userInfo.hasNoBotTag() const {nobot} = await userInfo.hasNoBotTag()
if (nobot) { if (nobot) {
continue continue
@ -290,7 +297,11 @@ export class Postbuilder {
console.error("Could not add contributor " + uid, e) console.error("Could not add contributor " + uid, e)
} }
} }
lastPostId = (await this._poster.writeMessage(toSend.join("\n"), {mediaIds: attachmentIds.slice(0, 4)})).id lastPostId = (await this._poster.writeMessage(toSend.join("\n"),
{
mediaIds: await imageUploader.attemptToUpload(4),
spoilerText: this._config.contentWarning
})).id
toSend = [] toSend = []
} }
@ -316,30 +327,33 @@ export class Postbuilder {
lastPostId = (await this._poster.writeMessage(toSend.join("\n"), { lastPostId = (await this._poster.writeMessage(toSend.join("\n"), {
inReplyToId: lastPostId, inReplyToId: lastPostId,
mediaIds: attachmentIds.slice(4, 8) mediaIds: await imageUploader.attemptToUpload(4),
spoilerText: this._config.contentWarning
})).id })).id
toSend = [] toSend = []
} }
const authorNames = Array.from(new Set<string>(imgAuthors)) const images = await imageUploader.attemptToUpload(4)
const authors = Array.from(new Set(imageUploader.getCurrentAuthors()))
await this._poster.writeMessage([ await this._poster.writeMessage([
"In total, " + totalImageContributorCount + " different contributors uploaded " + totalImagesCreated + " images.\n", "In total, " + totalImageContributorCount + " different contributors uploaded " + totalImagesCreated + " images.\n",
"Images in this thread are randomly selected from them and were made by: ", "Images in this thread are randomly selected from them and were made by: ",
...authorNames.map(auth => "- " + auth), ...authors.map(auth => "- " + auth),
"", "",
"All changes were made on " + date + (this._config.numberOfDays > 1 ? ` or at most ${this._config.numberOfDays} days before` : "") "All changes were made on " + date + (this._config.numberOfDays > 1 ? ` or at most ${this._config.numberOfDays} days before` : "")
].join("\n"), { ].join("\n"), {
inReplyToId: lastPostId, inReplyToId: lastPostId,
mediaIds: attachmentIds.slice(8, 12) mediaIds: images,
spoilerText: this._config.contentWarning
} }
) )
} }
private async prepareImages(changesets: ChangeSetData[], targetCount: number = 4): Promise<{ imgAuthors: string[], attachmentIds: string[], totalImagesCreated: number, totalImageContributorCount: number }> { private async prepareImages(changesets: ChangeSetData[]): Promise<{ randomImages: { image: string, changeset: ChangeSetData }[], totalImagesCreated: number, totalImageContributorCount: number }> {
const withImage: ChangeSetData[] = changesets.filter(cs => cs.properties["add-image"] > 0) const withImage: ChangeSetData[] = changesets.filter(cs => cs.properties["add-image"] > 0)
const totalImagesCreated = Utils.Sum(withImage.map(cs => cs.properties["add-image"])) const totalImagesCreated = Utils.Sum(withImage.map(cs => cs.properties["add-image"]))
@ -347,7 +361,7 @@ export class Postbuilder {
const seenURLS = new Set<string>() const seenURLS = new Set<string>()
for (const changeset of withImage) { for (const changeset of withImage) {
const userinfo = new OsmUserInfo(Number(changeset.properties.uid)) const userinfo = new OsmUserInfo(Number(changeset.properties.uid), this._globalConfig)
const {nobot} = await userinfo.hasNoBotTag() const {nobot} = await userinfo.hasNoBotTag()
if (nobot) { if (nobot) {
console.log("Not indexing images of user", changeset.properties.user) console.log("Not indexing images of user", changeset.properties.user)
@ -369,35 +383,10 @@ export class Postbuilder {
} }
} }
const randomImages: ImageInfo[] = this.selectImages(images, targetCount) const randomImages: ImageInfo[] = this.selectImages(images)
const attachmentIds: string[] = []
const imgAuthors: string[] = []
for (const randomImage of randomImages) {
const cs = randomImage.changeset.properties
let authorName = cs.user
try {
const authorInfo = new OsmUserInfo(Number(cs.uid), this._globalConfig)
authorName = (await authorInfo.GetMastodonUsername(this._poster)) ?? cs.user
} catch (e) {
console.log("Could not fetch more info about contributor", authorName, cs.uid, "due to", e)
}
imgAuthors.push(authorName)
if (this._globalConfig.mastodonAuth.dryrun) {
console.log("Not uploading/downloading image:" + randomImage.image + " dryrun")
continue
}
const attribution = await ImgurAttribution.DownloadAttribution(randomImage.image)
const id = randomImage.image.substring(randomImage.image.lastIndexOf("/") + 1)
const path = this._globalConfig.cacheDir + "/image_" + id
await Utils.DownloadBlob(randomImage.image, path)
const mediaId = await this._poster.uploadImage(path, "Image taken by " + authorName + ", available under " + attribution.license + ". It is made with the thematic map " + randomImage.changeset.properties.theme + " in changeset https://openstreetmap.org/changeset/" + randomImage.changeset.id)
attachmentIds.push(mediaId)
}
return { return {
attachmentIds, randomImages,
imgAuthors,
totalImagesCreated, totalImagesCreated,
totalImageContributorCount: new Set(withImage.map(cs => cs.properties.uid)).size totalImageContributorCount: new Set(withImage.map(cs => cs.properties.uid)).size
} }