Compare commits
139 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 008f36b1d2 | |||
| 704450bc44 | |||
|
|
230bbf3d53 | ||
|
|
b3b1aac63e | ||
|
|
ef68a8f36b | ||
|
|
734d94291b | ||
|
|
f90158c1e1 | ||
|
|
09184861e3 | ||
|
|
2a520c648d | ||
| 287b9995fe | |||
|
|
0b53e95e5e | ||
|
|
e95bdadb12 | ||
|
|
9959ad2468 | ||
|
|
8da83085c8 | ||
|
|
fdbec2f393 | ||
|
|
25bf602bba | ||
|
|
5bfabe8fd8 | ||
|
|
063aa8ba86 | ||
|
|
0654150d83 | ||
|
|
519d6f350a | ||
|
|
3c071d0447 | ||
|
|
cd9d79eda7 | ||
|
|
89b7f551d8 | ||
|
|
85bfb0be00 | ||
|
|
212c338ec9 | ||
| dd45dc996b | |||
| 1d5aa50423 | |||
|
|
15b2e7f2a7 | ||
|
|
4030d04b06 | ||
| eff0af0b34 | |||
| 34b1ad9d1e | |||
|
|
6675c4d232 | ||
|
|
46be48af38 | ||
|
|
09948a6b39 | ||
|
|
bc01391872 | ||
| cd3d00adbe | |||
| 528ecdd43d | |||
|
|
5cb928518d | ||
|
|
d8c7409c7a | ||
|
|
1cd6eb5134 | ||
| 9e2983a85c | |||
| 6eddd08e4a | |||
| 34dfc25e87 | |||
|
|
0bc6a0ae59 | ||
|
|
14c554563a | ||
| 475a9a2fd3 | |||
| fbd9a5a621 | |||
| 7014c97dee | |||
| 91edfeb19a | |||
| 3b3435515c | |||
| 2239a83f76 | |||
| 31b27beee5 | |||
|
|
a4be51f498 | ||
|
|
6b5af984cc | ||
|
|
f5d3853388 | ||
| 07ebd7afad | |||
|
|
5e82881b69 | ||
| c9a75e3f91 | |||
|
|
2fd2028cbe | ||
|
|
766f35d1d5 | ||
| e4edde082b | |||
|
|
6b95a01591 | ||
|
|
2f66758c9f | ||
|
|
f21b47e14f | ||
|
|
14f1c89f05 | ||
|
|
b7530843c5 | ||
|
|
44c5e40466 | ||
|
|
7b4dd50ebf | ||
|
|
5a6fe0f2fe | ||
| d7f5fbbe58 | |||
| ee550b1ed7 | |||
| e35815f27f | |||
| 9138770d6e | |||
| 834fbcd9c5 | |||
| 4c10ae89f8 | |||
| df6fc47ca3 | |||
| 5956a68e72 | |||
| eee0167574 | |||
| a3b2748479 | |||
| 8cc5a9e3ed | |||
| d2bcd7b505 | |||
| c59c2e2931 | |||
| 2288806105 | |||
| 5a2c277f0e | |||
| 612ed2aa79 | |||
| ed78731b3c | |||
| a84f0e1436 | |||
| d8d5be5c7d | |||
| bcf920c91e | |||
|
|
14f8c3fd43 | ||
| 9cd1fc0596 | |||
| 8df3ebfa6d | |||
| de82127223 | |||
| a79a28f4a1 | |||
| 95b1172b7f | |||
| 6f3954f1c0 | |||
| a465bbfaa8 | |||
| ddae6ed8be | |||
| 8cced1f9e1 | |||
| 3245438564 | |||
| 71c528114f | |||
| e2afc1350b | |||
| e971070f9f | |||
|
|
8771ec1108 | ||
|
|
1896209b96 | ||
|
|
3c93c9fce0 | ||
|
|
42a958fc4c | ||
|
|
5fa46ff106 | ||
|
|
cca3211023 | ||
|
|
dce7604afb | ||
| 0fd9b519d1 | |||
| 6b7f727449 | |||
| 5abbab9527 | |||
|
|
e03c7debb6 | ||
| 1d2483d18f | |||
| b17f4f0f3e | |||
| 4a91167871 | |||
| 0baaf6f38b | |||
| 42182bb6c9 | |||
| 8455c193eb | |||
| d1faa10a52 | |||
| e28137bf9b | |||
| 1ff5c240c8 | |||
|
|
f893f9c3d7 | ||
|
|
c4e88d98f2 | ||
|
|
0cca387982 | ||
|
|
9e619fb3c5 | ||
|
|
0fdfed2929 | ||
|
|
73c09ec38b | ||
|
|
f901af4f0c | ||
|
|
2a8f53ca98 | ||
|
|
4a95cb1f5e | ||
|
|
482612f889 | ||
|
|
b8977df986 | ||
|
|
ae5e3daea3 | ||
|
|
1c6d832b20 | ||
| 52b39d91a3 | |||
| 3748cd35e5 | |||
| 4d4577c9f8 |
31
.github/ISSUE_TEMPLATE/bug_report.md
vendored
Normal file
@@ -0,0 +1,31 @@
|
|||||||
|
---
|
||||||
|
name: Bug report
|
||||||
|
about: Create a report to help us improve
|
||||||
|
title: "[bug-report]"
|
||||||
|
labels: bug
|
||||||
|
assignees: TLINDEN
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
**Description**
|
||||||
|
<!-- Please provide a clear and concise description of the issue: -->
|
||||||
|
|
||||||
|
|
||||||
|
**Steps To Reproduce**
|
||||||
|
<!-- Please detail the steps to reproduce the behavior, execute kleingebaeck with the -d option: -->
|
||||||
|
|
||||||
|
|
||||||
|
**Expected behavior**
|
||||||
|
<!-- What do you expected to happen instead? -->
|
||||||
|
|
||||||
|
|
||||||
|
**Version information**
|
||||||
|
<!--
|
||||||
|
Please provide as much version information as possible:
|
||||||
|
- if you have just installed a binary, provide the output of: kleingebaeck --version
|
||||||
|
- if you installed from source, provide the output of: make show-version
|
||||||
|
- provide additional details: operating system and version and shell environment
|
||||||
|
-->
|
||||||
|
|
||||||
|
|
||||||
|
**Additional informations**
|
||||||
32
.github/ISSUE_TEMPLATE/bug_report_de.md
vendored
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
---
|
||||||
|
name: Bugreport Deutsch
|
||||||
|
about: Erzeuge einen Bugreport
|
||||||
|
title: "[bug-report-de]"
|
||||||
|
labels: bug
|
||||||
|
assignees: TLINDEN
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
**Beschreibung**
|
||||||
|
<!-- Bitte beschreibe den Fehler klar und möglichst präzise: -->
|
||||||
|
|
||||||
|
|
||||||
|
**Schritte um den Fehler zu reproduzieren**
|
||||||
|
<!-- Bitte gib detailiert an, welche konkreten Schritte zum Fehler
|
||||||
|
geführt haben, führe kleingebaeck mit der Option -d option aus: -->
|
||||||
|
|
||||||
|
|
||||||
|
**Erwartetes Verhalten**
|
||||||
|
<!-- Welches Verhalten hast Du ursprünglich erwartet? -->
|
||||||
|
|
||||||
|
|
||||||
|
**Versionsinformation**
|
||||||
|
<!--
|
||||||
|
Bitte gib uns so viel Versionsinfos wie möglich:
|
||||||
|
- wenn Du nur das Programm installiert hast: kleingebaeck --version
|
||||||
|
- wenn Du von Source installiert hast: make show-version
|
||||||
|
- bitte gib zusätzliche Details an: Betriebssystem + Version, Shellumgebung etc.
|
||||||
|
-->
|
||||||
|
|
||||||
|
|
||||||
|
**Zusätzliche Informationen**
|
||||||
23
.github/ISSUE_TEMPLATE/feature_request.md
vendored
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
---
|
||||||
|
name: Feature request
|
||||||
|
about: Suggest a feature
|
||||||
|
title: "[feature-request]"
|
||||||
|
labels: feature-request
|
||||||
|
assignees: TLINDEN
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
**Description**
|
||||||
|
<!-- Please provide a clear and concise description of the feature you desire: -->
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
**Version information**
|
||||||
|
<!--
|
||||||
|
Just in case the feature is already present, please provide as
|
||||||
|
much version information as possible:
|
||||||
|
- if you have just installed a binary, provide the output of: tablizer --version
|
||||||
|
- if you installed from source, provide the output of: make show-version
|
||||||
|
- provide additional details: operating system and version and shell environment
|
||||||
|
-->
|
||||||
|
|
||||||
20
.github/ISSUE_TEMPLATE/feature_request_de.md
vendored
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
---
|
||||||
|
name: Featurerequest Deutsch
|
||||||
|
about: Empfehle ein neues Feature
|
||||||
|
title: "[feature-request-de]"
|
||||||
|
labels: feature-request
|
||||||
|
assignees: TLINDEN
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
**Beschreibung**
|
||||||
|
<!-- Bitte beschreibe das gewünschte Feature klar und möglichst präzise: -->
|
||||||
|
|
||||||
|
|
||||||
|
**Versionsinformation**
|
||||||
|
<!--
|
||||||
|
Bitte gib uns so viel Versionsinfos wie möglich:
|
||||||
|
- wenn Du nur das Programm installiert hast: kleingebaeck --version
|
||||||
|
- wenn Du von Source installiert hast: make show-version
|
||||||
|
- bitte gib zusätzliche Details an: Betriebssystem + Version, Shellumgebung etc.
|
||||||
|
-->
|
||||||
8
.github/ISSUE_TEMPLATE/note_to_self.md
vendored
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
---
|
||||||
|
name: Note to self
|
||||||
|
about: Internal bugs and wishes
|
||||||
|
title: "[bug-report]"
|
||||||
|
labels: bug
|
||||||
|
assignees: TLINDEN
|
||||||
|
|
||||||
|
---
|
||||||
BIN
.github/assets/adlisting-windows.jpg
vendored
Executable file
|
After Width: | Height: | Size: 139 KiB |
BIN
.github/assets/cmd-windows.jpg
vendored
Executable file
|
After Width: | Height: | Size: 33 KiB |
BIN
.github/assets/english.png
vendored
Normal file
|
After Width: | Height: | Size: 2.5 KiB |
BIN
.github/assets/english.xcf
vendored
Normal file
BIN
.github/assets/german.png
vendored
Normal file
|
After Width: | Height: | Size: 1.9 KiB |
BIN
.github/assets/german.xcf
vendored
Normal file
BIN
.github/assets/kleinanzeigen-ad.png
vendored
Normal file
|
After Width: | Height: | Size: 199 KiB |
BIN
.github/assets/kleinanzeigen-backup.png
vendored
Normal file
|
After Width: | Height: | Size: 263 KiB |
BIN
.github/assets/kleinanzeigen-download.png
vendored
Normal file
|
After Width: | Height: | Size: 10 KiB |
BIN
.github/assets/kleinanzeigen-index.png
vendored
Normal file
|
After Width: | Height: | Size: 232 KiB |
BIN
.github/assets/kleingebaecklogo.xcf
vendored
Normal file
BIN
.github/assets/liste-windows.jpg
vendored
Executable file
|
After Width: | Height: | Size: 90 KiB |
10
.github/dependabot.yml
vendored
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
version: 2
|
||||||
|
updates:
|
||||||
|
- package-ecosystem: "gomod"
|
||||||
|
directory: "/"
|
||||||
|
schedule:
|
||||||
|
interval: "monthly"
|
||||||
|
- package-ecosystem: "github-actions"
|
||||||
|
directory: "/"
|
||||||
|
schedule:
|
||||||
|
interval: "monthly"
|
||||||
5
.gitignore
vendored
@@ -1,3 +1,8 @@
|
|||||||
test
|
test
|
||||||
kleingebaeck
|
kleingebaeck
|
||||||
releases
|
releases
|
||||||
|
t/out
|
||||||
|
.bak
|
||||||
|
t/httproot/out
|
||||||
|
t/httproot/kleinanzeigen
|
||||||
|
t/httproot/favicon.ico
|
||||||
|
|||||||
88
Makefile
@@ -1,88 +0,0 @@
|
|||||||
# Copyright © 2023 Thomas von Dein
|
|
||||||
|
|
||||||
# This program is free software: you can redistribute it and/or modify
|
|
||||||
# it under the terms of the GNU General Public License as published by
|
|
||||||
# the Free Software Foundation, either version 3 of the License, or
|
|
||||||
# (at your option) any later version.
|
|
||||||
|
|
||||||
# This program is distributed in the hope that it will be useful,
|
|
||||||
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
||||||
# GNU General Public License for more details.
|
|
||||||
|
|
||||||
# You should have received a copy of the GNU General Public License
|
|
||||||
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
|
|
||||||
|
|
||||||
#
|
|
||||||
# no need to modify anything below
|
|
||||||
tool = kleingebaeck
|
|
||||||
VERSION = $(shell grep VERSION main.go | head -1 | cut -d '"' -f2)
|
|
||||||
archs = darwin freebsd linux windows
|
|
||||||
PREFIX = /usr/local
|
|
||||||
UID = root
|
|
||||||
GID = 0
|
|
||||||
HAVE_POD := $(shell pod2text -h 2>/dev/null)
|
|
||||||
|
|
||||||
all: $(tool).1 $(tool).go buildlocal
|
|
||||||
|
|
||||||
%.1: %.pod
|
|
||||||
ifdef HAVE_POD
|
|
||||||
pod2man -c "User Commands" -r 1 -s 1 $*.pod > $*.1
|
|
||||||
endif
|
|
||||||
|
|
||||||
%.go: %.pod
|
|
||||||
ifdef HAVE_POD
|
|
||||||
echo "package main" > $*.go
|
|
||||||
echo >> $*.go
|
|
||||||
echo "var manpage = \`" >> $*.go
|
|
||||||
pod2text $*.pod >> $*.go
|
|
||||||
echo "\`" >> $*.go
|
|
||||||
endif
|
|
||||||
|
|
||||||
buildlocal:
|
|
||||||
CGO_LDFLAGS='-static' go build -tags osusergo,netgo -ldflags "-extldflags=-static" -o $(tool)
|
|
||||||
|
|
||||||
install: buildlocal
|
|
||||||
install -d -o $(UID) -g $(GID) $(PREFIX)/bin
|
|
||||||
install -d -o $(UID) -g $(GID) $(PREFIX)/man/man1
|
|
||||||
install -o $(UID) -g $(GID) -m 555 $(tool) $(PREFIX)/sbin/
|
|
||||||
install -o $(UID) -g $(GID) -m 444 $(tool).1 $(PREFIX)/man/man1/
|
|
||||||
|
|
||||||
clean:
|
|
||||||
rm -rf $(tool) coverage.out testdata
|
|
||||||
|
|
||||||
test: clean
|
|
||||||
go test ./... $(ARGS)
|
|
||||||
|
|
||||||
testfuzzy: clean
|
|
||||||
go test -fuzz ./... $(ARGS)
|
|
||||||
|
|
||||||
singletest:
|
|
||||||
@echo "Call like this: make singletest TEST=TestPrepareColumns ARGS=-v"
|
|
||||||
go test -run $(TEST) $(ARGS)
|
|
||||||
|
|
||||||
cover-report:
|
|
||||||
go test ./... -cover -coverprofile=coverage.out
|
|
||||||
go tool cover -html=coverage.out
|
|
||||||
|
|
||||||
goupdate:
|
|
||||||
go get -t -u=patch ./...
|
|
||||||
|
|
||||||
buildall:
|
|
||||||
./mkrel.sh $(tool) $(VERSION)
|
|
||||||
|
|
||||||
release: buildall
|
|
||||||
gh release create v$(VERSION) --generate-notes releases/*
|
|
||||||
|
|
||||||
show-versions: buildlocal
|
|
||||||
@echo "### kleingebaeck version:"
|
|
||||||
@./kleingebaeck -v
|
|
||||||
|
|
||||||
@echo
|
|
||||||
@echo "### go module versions:"
|
|
||||||
@go list -m all
|
|
||||||
|
|
||||||
@echo
|
|
||||||
@echo "### go version used for building:"
|
|
||||||
@grep -m 1 go go.mod
|
|
||||||
357
README-de.md
Normal file
@@ -0,0 +1,357 @@
|
|||||||
|
## Kleingebäck - kleinanzeigen.de Backup
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
[](https://goreportcard.com/report/codeberg.org/scip/kleingebaeck)
|
||||||
|
[](https://ci.codeberg.org/repos/15530)
|
||||||
|

|
||||||
|
[](https://codeberg.org/scip/kleingebaeck/releases)
|
||||||
|
[](https://codeberg.org/scip/kleingebaeck/raw/branch/main/README.md)
|
||||||
|
|
||||||
|
Mit diesem Tool kann man seine Anzeigen bei https://kleinanzeigen.de sichern.
|
||||||
|
|
||||||
|
> [!IMPORTANT]
|
||||||
|
> Diese Software wird jetzt bei Codeberg weitergepflegt: [Codeberg](https://codeberg.org/scip/kleingebaeck/).
|
||||||
|
|
||||||
|
Es kann alle Anzeigen eines Users (oder nur eine Ausgewählte)
|
||||||
|
inklusive der Bilder herunterladen, die in einem Verzeichnis pro
|
||||||
|
Anzeige gespeichert werden. In dem Verzeichnis wird eine Datei
|
||||||
|
`Adlisting.txt` erstellt, in der sich die Inhalte der Anzeige wie
|
||||||
|
Titel, Preis, Text etc befinden. Bilder werden natürlich auch heruntergeladen.
|
||||||
|
|
||||||
|
## ACHTUNG - SICHERHEITS-UPDATE
|
||||||
|
|
||||||
|
Fertige vorcompilierte Programme älter als Version `v0.3.12` sind von
|
||||||
|
Schwachstellen in der Behandlung von HTTP und Zertifikaten
|
||||||
|
betroffen. Falls Du eine ältere Kleingebäck-Version im Einsatz hast,
|
||||||
|
bitte update auf Version `v0.3.12` oder höher. Bitte lies auch die [Release Notes für
|
||||||
|
v0.3.12](https://codeberg.org/scip/kleingebaeck/releases/tag/v0.3.12)
|
||||||
|
für mehr Details.
|
||||||
|
|
||||||
|
## Screenshots
|
||||||
|
|
||||||
|
Das ist die Hauptseite meines kleinanzeigen.de Accounts:
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
Sichern ich meine Anzeigen:
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
Backupverzeichnis nach dem Download:
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
Verzeichnis einer Anzeige:
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
**Das gleiche unter Windows:**
|
||||||
|
|
||||||
|
Anzeigen Sichern:
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
Backupverzeichnis nach dem Download
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
Und eine Anzeige:
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
## Installation
|
||||||
|
|
||||||
|
Das Tool hat keine weiteren Abhängigkeiten und erfordert auch keine
|
||||||
|
Anmeldung oder ähnliches. Man kädt sich einfach die ausführbare Datei
|
||||||
|
für seine Plattform herunter und kann direkt loslegen.
|
||||||
|
|
||||||
|
### Installation des vorcompilierten Programms
|
||||||
|
|
||||||
|
Auf der Seite [des letzten Releases](https://codeberg.org/scip/kleingebaeck/releases) findet man das Program für sein Betriebssystem und die Plattform (z.b. Windows + Intel)
|
||||||
|
|
||||||
|
Es gibt 2 Varianten:
|
||||||
|
|
||||||
|
1. Direkt das fertige Program für seine Plattform+OS herunterladen,
|
||||||
|
z.B. `kleingebaeck-linux-amd64-0.0.5`, nach `kleingebaeck`
|
||||||
|
umbenennen und in ein Verzeichnis kopieren, das im `PATH` ist,
|
||||||
|
(z.B. nach `$HOME/bin` oder als root nach `/usr/local/bin`).
|
||||||
|
|
||||||
|
Um sicher zu gehen, dass an dem Program nicht verändert wurde, kann
|
||||||
|
man die Signatur vergleichen. Für jeden Download gibt es eine dazu
|
||||||
|
passende Signatur, in unserem Beispiel wäre das
|
||||||
|
`kleingebaeck-linux-amd64-0.0.5.sha256`.
|
||||||
|
|
||||||
|
Zum Verifizieren ausführen:
|
||||||
|
|
||||||
|
```shell
|
||||||
|
cat kleingebaeck-linux-amd64-0.0.5.sha25 && sha256sum kleingebaeck-linux-amd64-0.0.5
|
||||||
|
```
|
||||||
|
Man sollte zweimal den gleichen SHA256 Hash sehen.
|
||||||
|
|
||||||
|
2. Man kann auch einen Tarball (tgz Dateiendung) herunterladen,
|
||||||
|
auspacken und mit GNU Make installieren:
|
||||||
|
|
||||||
|
```shell
|
||||||
|
tar xvfz kleingebaeck-linux-amd64-0.0.5.tar.gz
|
||||||
|
cd kleingebaeck-linux-amd64-0.0.5
|
||||||
|
sudo make install
|
||||||
|
```
|
||||||
|
|
||||||
|
### Installation aus dem Sourcecode
|
||||||
|
|
||||||
|
Man muss eine funktionierende Go Buildumgebung in der Version 1.21
|
||||||
|
installiert haben, um das Programm selber zu compilieren. GNU Make ist
|
||||||
|
hilfreich, aber nicht unbedingt erforderlich.
|
||||||
|
|
||||||
|
Um das Programm zu compilieren, muss man folgende Schritte ausführen:
|
||||||
|
|
||||||
|
```shell
|
||||||
|
git clone https://codeberg.org/scip/kleingebaeck.git
|
||||||
|
cd kleingebaeck
|
||||||
|
go mod tidy
|
||||||
|
make # (oder make)
|
||||||
|
sudo make install
|
||||||
|
```
|
||||||
|
|
||||||
|
### Docker image benutzen
|
||||||
|
|
||||||
|
Ein fertiges Dockerimage mit der aktuellen Programmversion ist immer
|
||||||
|
verfügbar. Man kann damit z.B. das Tool testen, bevor man es dauerhaft
|
||||||
|
benutzen möchte.
|
||||||
|
|
||||||
|
Um das Image herunterzuladen:
|
||||||
|
```
|
||||||
|
docker pull ghcr.io/tlinden/kleingebaeck:latest
|
||||||
|
```
|
||||||
|
|
||||||
|
Um kleingebäck im Image auszuführen und Daten ins lokale Filesystem zu
|
||||||
|
sichern, kann man so vorgehen:
|
||||||
|
|
||||||
|
```shell
|
||||||
|
mkdir anzeigen
|
||||||
|
docker run -u `id -u $USER` -v ./anzeigen:/backup ghcr.io/tlinden/kleingebaeck:latest -u XXX -v
|
||||||
|
ls -l anzeigen/ein-buch-mit-leeren-seiten
|
||||||
|
total 792
|
||||||
|
drwxr-xr-x 2 scip root 4096 Jan 23 12:58 ./
|
||||||
|
drwxr-xr-x 3 scip scip 4096 Jan 23 12:58 ../
|
||||||
|
-rw-r--r-- 1 scip root 131650 Jan 23 12:58 1.jpg
|
||||||
|
-rw-r--r-- 1 scip root 81832 Jan 23 12:58 2.jpg
|
||||||
|
-rw-r--r-- 1 scip root 134050 Jan 23 12:58 3.jpg
|
||||||
|
-rw-r--r-- 1 scip root 1166 Jan 23 12:58 Adlisting.txt
|
||||||
|
```
|
||||||
|
|
||||||
|
Hier wird der aktuelle User auf den User im Image gemappt und das
|
||||||
|
lokale Verzeichnis `anzeigen` nach `/backup` innerhalb des Images
|
||||||
|
gemountet.
|
||||||
|
|
||||||
|
Die Optionen `-u XXX -v` sind kleingebäck Optionen. Ersetze `XXX`
|
||||||
|
durch Deine tatsächliche kleinanzeigen.de Userid.
|
||||||
|
|
||||||
|
Eine Liste verfügbarer Images findet man [hier](https://codeberg.org/scip/kleingebaeck/pkgs/container/kleingebaeck/versions?filters%5Bversion_type%5D=tagged)
|
||||||
|
|
||||||
|
## Kommandozeilen Optionen:
|
||||||
|
|
||||||
|
```
|
||||||
|
Usage: kleingebaeck [-dvVhmoc] [<ad-listing-url>,...]
|
||||||
|
Options:
|
||||||
|
-u --user <uid> Backup ads from user with uid <uid>.
|
||||||
|
-d --debug Enable debug output.
|
||||||
|
-v --verbose Enable verbose output.
|
||||||
|
-o --outdir <dir> Set output dir (default: current directory)
|
||||||
|
-l --limit <num> Limit the ads to download to <num>, default: load all.
|
||||||
|
-c --config <file> Use config file <file> (default: ~/.kleingebaeck).
|
||||||
|
--ignoreerrors Ignore HTTP errors, may lead to incomplete ad backup.
|
||||||
|
-m --manual Show manual.
|
||||||
|
-h --help Show usage.
|
||||||
|
-V --version Show program version.
|
||||||
|
|
||||||
|
If one or more <ad-listing-url>'s are specified, only backup those,
|
||||||
|
otherwise backup all ads of the given user.
|
||||||
|
```
|
||||||
|
|
||||||
|
## Konfiguration
|
||||||
|
|
||||||
|
Man kann anstelle von Kommandlineoptionen auch eine
|
||||||
|
Konfigurationsdatei verwenden. Sie befindet sich standardmäßig in
|
||||||
|
`~/.kleingebaeck` aber man kann mit dem Parameter `-c` auch eine
|
||||||
|
andere Datei angeben.
|
||||||
|
|
||||||
|
Das Format (TOML) ist einfach:
|
||||||
|
|
||||||
|
```
|
||||||
|
user = 1010101
|
||||||
|
loglevel = verbose
|
||||||
|
outdir = "test"
|
||||||
|
```
|
||||||
|
|
||||||
|
Im Source gibt es eine Beispieldatei `example.conf` mit Kommentaren.
|
||||||
|
|
||||||
|
## Umgebungsvariablen
|
||||||
|
|
||||||
|
Man kann darüber hinaus auch Umgebungsvariablen verwenden. Sie
|
||||||
|
entsprechen den Konfigurationsoptionen, aber gross geschrieben mit dem
|
||||||
|
Präfix `KLEINGEBAECK_`, z.B.
|
||||||
|
|
||||||
|
```shell
|
||||||
|
% KLEINGEBAECK_OUTDIR=/backup kleingebaeck -v
|
||||||
|
```
|
||||||
|
|
||||||
|
## Benutzung
|
||||||
|
|
||||||
|
Um das Tool einsetzen zu können, muss man zunächst seine Userid bei
|
||||||
|
kleinanzeigen.de herausfinden. Dazu ruft man am besten die Liste
|
||||||
|
seiner Anzeigen auf, während man NICHT eingeloggt ist:
|
||||||
|
|
||||||
|
https://www.kleinanzeigen.de/s-bestandsliste.html?userId=XXXXXX
|
||||||
|
|
||||||
|
Der `XXXXX` Teil der URL ist die Userid.
|
||||||
|
|
||||||
|
Trage diese Userid in der Konfigurationsdatei ein wie oben
|
||||||
|
beschrieben. Gib ausserdem das Ausgabeverzeichnis an. Dann einfach nur
|
||||||
|
`kleingebaeck` ausführen.
|
||||||
|
|
||||||
|
Innerhalb des Ausgabeverzeichnisses wird sich dann pro Anzeige ein
|
||||||
|
Unterverzeichnis befinden. Pro Anzeige gibt es eine Datei
|
||||||
|
`Adlisting.txt`, die etwa so aussieht:
|
||||||
|
|
||||||
|
```default
|
||||||
|
Title: A book I sell
|
||||||
|
Price: 99 € VB
|
||||||
|
Id: 1919191919
|
||||||
|
Category: Sachbücher
|
||||||
|
Condition: Sehr Gut
|
||||||
|
Created: 10.12.2023
|
||||||
|
|
||||||
|
This is the description text.
|
||||||
|
|
||||||
|
Pay with paypal.
|
||||||
|
```
|
||||||
|
|
||||||
|
Sowie alle Bilder.
|
||||||
|
|
||||||
|
Das Format kann man mit der Variable `template` in der Konfiguration
|
||||||
|
ändern. Die `example.conf` enthält ein Beispiel für das Standard Template.
|
||||||
|
|
||||||
|
## Verhalten des Tools
|
||||||
|
|
||||||
|
Es gibt einige Dinge über das Verhalten von kleingebäck, über die Du
|
||||||
|
Bescheid wissen solltest:
|
||||||
|
|
||||||
|
- alle HTML Seiten und Bilder werden immer heruntergeladen
|
||||||
|
- es wird ein (konfigurierbarer) Useragent verwendet
|
||||||
|
- HTTP Cookies werden beachtet
|
||||||
|
- bei Fehlern wird dreimal mit unterschiedlichem Abstand erneut
|
||||||
|
versucht
|
||||||
|
- Bilder Downloads laufen parallelisiert mit leicht unterschiedlichen
|
||||||
|
zeitlichen Abständen ab
|
||||||
|
- Gleich aussehende Bilder werden nicht überschrieben
|
||||||
|
|
||||||
|
Der letzte Punkt muss genauer erläutert werden:
|
||||||
|
|
||||||
|
Wenn man bei Kleinanzeigen.de eine Anzeige einstellt und Bilder
|
||||||
|
postet, werden diese dort in ihrer Grösse reduziert (durch Kompression
|
||||||
|
und Verkleinerung der Bilder usw.). Diese reduzierten Bilder werden
|
||||||
|
dann von kleingebäck heruntergeladen. Falls Du Deine original Bilder
|
||||||
|
behalten hast, kannst Du diese danach in das Backupverzeichnis
|
||||||
|
kopieren. Bei einem erneuten kleingebäck-Lauf werden diese Bilder dann
|
||||||
|
nicht überschrieben.
|
||||||
|
|
||||||
|
Wir verwenden dafür einen Algorythmus namens [distance
|
||||||
|
hashing](https://github.com/corona10/goimagehash). Dieser Algorithmus
|
||||||
|
prüft die Ähnlichkeit von Bildern. Diese können in ihrer Auflösung,
|
||||||
|
Kompression, Farbtiefe und vielem mehr manipuliert worden sein und
|
||||||
|
trotzdem als das "gleiche Bild" erkannt werden (wohlgemerkt nicht "das
|
||||||
|
selbe": die Dateien sind durchaus unterschiedlich!). Bis zu einer
|
||||||
|
Distance von 5 überschreiben wir keine Bilder, weil wir dann davon
|
||||||
|
ausgehen, dass das lokal Vorhandene das Original ist.
|
||||||
|
|
||||||
|
Bitte beachte aber, dass dies KEIN Cachingmechanismus ist: die Bilder
|
||||||
|
werden trotzdem immer alle heruntergeladen. Das muss so sein, da wir
|
||||||
|
uns nicht die Dateinamen anschauen können, da kleinanzeigen.de diese
|
||||||
|
nämlich zu Zahlen umbenennt. Und die Dateinamen können sich auch
|
||||||
|
ändern, wenn der User in der Anzeige die Bilder umarrangiert hat.
|
||||||
|
|
||||||
|
Du kannst dieses Verhalten mit der Option **--force** ausschalten. Du
|
||||||
|
kannst ausserdem mit der Option **--ignoreerrors** auch alle Fehler
|
||||||
|
ignorieren, die beim Bilderdownload auftreten könnten.
|
||||||
|
|
||||||
|
## Documentation
|
||||||
|
|
||||||
|
Die Dokumentation kann man
|
||||||
|
[online](https://codeberg.org/scip/kleingebaeck/raw/branch/main/kleingebaeck.pod)
|
||||||
|
oder lokal lesen mit: `kleingebaeck --manual`. Hat man das Tool mit
|
||||||
|
dem Tarball installiert, funktioniert auch `man kleingebaeck`.
|
||||||
|
|
||||||
|
## Kleingebäck?
|
||||||
|
|
||||||
|
Der Name kommt von "kleinanzeigen backup", verkürzt "klein back", das
|
||||||
|
englisch ausgesprochene "back" (deutsch bäck) führt dann zu "Kleingebäck".
|
||||||
|
|
||||||
|
## Wo bekommt man Hilfe
|
||||||
|
|
||||||
|
Obwohl ich gerne von kleingebäck Benutzern in privaten Mails höre, ist
|
||||||
|
das doch der beste Weg, die Anfrage zu übersehen und zu vergessen.
|
||||||
|
|
||||||
|
Um einen Fehler, ein unerwartetes Verhalten, eine Feature Request oder
|
||||||
|
einen Patch zu übermitteln, eröffne daher bitte einen Issue unter:
|
||||||
|
https://codeberg.org/scip/kleingebaeck/issues. Danke!
|
||||||
|
|
||||||
|
Bitte gebe den fehlgeschlagenen Befehl an, rufe es auch mit Debugging
|
||||||
|
`-d` auf.
|
||||||
|
|
||||||
|
## Ähnliche Projekte
|
||||||
|
|
||||||
|
Ich konnte kein Projekt finden, das speziell dafür geeignet ist,
|
||||||
|
Anzeigen bei kleinanzeigen.de zu sichern.
|
||||||
|
|
||||||
|
Aber es gibt ein Projekt, mit dem man ebenfalls Backups erstellen
|
||||||
|
kann: [kleinanzeigen-bot](https://github.com/Second-Hand-Friends/kleinanzeigen-bot/).
|
||||||
|
Aber Vorsicht: kleinanzeigen.de bekämpft Bots aktiv, mit diesem hier
|
||||||
|
gibt es regelmäßige Probleme, z.B.:
|
||||||
|
[issue](https://github.com/Second-Hand-Friends/kleinanzeigen-bot/issues/219).
|
||||||
|
Das Hauptproblem ist, dass diese Art von Bot sich mit Deinem Account
|
||||||
|
aktiv einloggt und mit der Seite interagiert. Damit kann die Firma die
|
||||||
|
Aktivitäten recht einfach Deinem User zuordnen und diesen **sperren**!
|
||||||
|
Also sei bitte vorsichtig!
|
||||||
|
|
||||||
|
**Kleingebäck** erfordert keinen Login, es verwendet lediglich die
|
||||||
|
öffentlich verfügbare Webseite und ruft diese auf, wie ein normaler
|
||||||
|
Browser. Tatsächlich gibt es meiner Meinung nach keinen Unterschied zu
|
||||||
|
einem Browserclient: beide laufen auf Anwenderseite auf Initiative
|
||||||
|
eines Benutzers. Und mit welchen Browser ich eine Webseite aufrufe,
|
||||||
|
bleibt immer noch mir überlassen und muss mir nicht von irgendwem
|
||||||
|
vorgeschrieben werden. Das schliesst die Verwendung von Kleingebäck
|
||||||
|
mit ein.
|
||||||
|
|
||||||
|
Hinzu kommt, dass dieses Tool nicht dazu gedacht ist, rund um die Uhr
|
||||||
|
zu laufen. Man ruft es ab und zu mal auf, wenn man halt neue Anzeigen
|
||||||
|
eingestellt hat, vielleicht einmal die Woche oder so. Man weiss ja
|
||||||
|
selber, wann man was geändert hat. Man benötigt trotzdem den Zugriff
|
||||||
|
mit dem Browser oder der mobilen App um Kleinanzeigen.de verwalten zu
|
||||||
|
können.
|
||||||
|
|
||||||
|
Meiner Ansicht nach ist das Risiko also sehr minimal, es handelt sich
|
||||||
|
meiner Meinung nach auch nicht um eine Verletzung der AGBs dort. Aber
|
||||||
|
das ist nur meine persönliche Meinung, bitte beachtet das. Am Ende
|
||||||
|
müsst Ihr selbst einschätzen und beurteilen wie hoch Ihr das Risiko
|
||||||
|
seht und ob Ohr es eingehen möchtet. Für eventuell auftretende
|
||||||
|
Konsequenzen bin ich nicht verantwortlich. Siehe auch [GPL Lizenz](LICENSE).
|
||||||
|
|
||||||
|
Es gibt noch ein weiteres Tool namens
|
||||||
|
[kleinanzeigen-enhanded](https://kleinanzeigen-enhanced.de/). Das ist
|
||||||
|
eine kostenpflichtige vollständige Anzeigenverwaltung für
|
||||||
|
Profinutzer. Man muss eine monatliche Abogebühr bezahlen. Das Tool
|
||||||
|
ist als Browsererweiterung für Google Chrome implementiert, was
|
||||||
|
erklärt, warum sie Anzeigen erstellen, ändern und löschen können,
|
||||||
|
obwohl es gar keine öffentliche API gibt. Sieht nach einer netten
|
||||||
|
ausgereiften Lösung aus. Mit Backups.
|
||||||
|
|
||||||
|
## Copyright und License
|
||||||
|
|
||||||
|
Lizensiert unter der GNU GENERAL PUBLIC LICENSE Version 3.
|
||||||
|
|
||||||
|
## Autor
|
||||||
|
|
||||||
|
T.v.Dein <tom AT vondein DOT org>
|
||||||
|
|
||||||
241
README.md
@@ -1,12 +1,17 @@
|
|||||||
## Kleingebäck - kleinanzeigen.de Backup
|
## Kleingebäck - kleinanzeigen.de Backup
|
||||||
|
|
||||||

|

|
||||||
|
|
||||||
[](https://github.com/tlinden/kleingebaeck/blob/master/LICENSE)
|
[](https://goreportcard.com/report/codeberg.org/scip/kleingebaeck)
|
||||||
[](https://goreportcard.com/report/github.com/tlinden/kleingebaeck)
|
[](https://ci.codeberg.org/repos/15530)
|
||||||

|

|
||||||
[](https://github.com/TLINDEN/kleingebaeck/releases/latest)
|
[](https://codeberg.org/scip/kleingebaeck/releases)
|
||||||
|
[](https://codeberg.org/scip/kleingebaeck/raw/branch/main/README-de.md)
|
||||||
|
|
||||||
|
[Die deutsche Version des READMEs findet Ihr hier](README-de.md).
|
||||||
|
|
||||||
|
> [!IMPORTANT]
|
||||||
|
> This software is now being maintained on [Codeberg](https://codeberg.org/scip/kleingebaeck/).
|
||||||
|
|
||||||
This tool can be used to backup ads on the german ad page https://kleinanzeigen.de
|
This tool can be used to backup ads on the german ad page https://kleinanzeigen.de
|
||||||
|
|
||||||
@@ -15,25 +20,143 @@ directory, each ad into its own subdirectory. The backup will contain
|
|||||||
a textfile `Adlisting.txt` which contains the ad contents as the
|
a textfile `Adlisting.txt` which contains the ad contents as the
|
||||||
title, body, price etc. All images will be downloaded as well.
|
title, body, price etc. All images will be downloaded as well.
|
||||||
|
|
||||||
|
## CAUTION - SECURITY UPDATE
|
||||||
|
|
||||||
|
Binary releases prior to version `v0.3.11` are affected by
|
||||||
|
vulnerabilities in HTTP and certificate handling. If you are using
|
||||||
|
such a binary, please update to `v0.3.12` or higher. Please also refer
|
||||||
|
to the [Release Notes of
|
||||||
|
v0.3.12](https://codeberg.org/scip/kleingebaeck/releases/tag/v0.3.12)
|
||||||
|
for more details.
|
||||||
|
|
||||||
|
## Screenshots
|
||||||
|
|
||||||
|
This is the index of my kleinanzeigen.de Account:
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
Here I download my ads on the commandline:
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
And this is the backup directory after download:
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
Here's a directory for one ad:
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
**The same thing under windows:**
|
||||||
|
|
||||||
|
Downloading ads:
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
Backup directory after download:
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
And one ad listing directory:
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
## Installation
|
||||||
|
|
||||||
The tool doesn't need authentication and doesn't have any
|
The tool doesn't need authentication and doesn't have any
|
||||||
dependencies. Just download the binary for your platform from the
|
dependencies. Just download the binary for your platform from the
|
||||||
releases page and you're good to go.
|
releases page and you're good to go.
|
||||||
|
|
||||||
The releases also include a handy tarball which you can use to install
|
### Installation using a pre-compiled binary
|
||||||
the tool system-wide including the manual page. Just extract it and
|
|
||||||
type: `make install`.
|
Go to the [latest release
|
||||||
|
page](https://codeberg.org/scip/kleingebaeck/releases) and
|
||||||
|
look for your OS and platform. There are two options to install the binary:
|
||||||
|
|
||||||
|
1. Directly download the binary for your platform,
|
||||||
|
e.g. `kleingebaeck-linux-amd64-0.0.5`, rename it to `kleingebaeck`
|
||||||
|
(or whatever you like more!) and put it into your bin dir
|
||||||
|
(e.g. `$HOME/bin` or as root to `/usr/local/bin`).
|
||||||
|
|
||||||
|
Be sure to verify the signature of the binary file. For this also download the matching `kleingebaeck-linux-amd64-0.0.5.sha256` file and:
|
||||||
|
|
||||||
|
```shell
|
||||||
|
cat kleingebaeck-linux-amd64-0.0.5.sha25 && sha256sum kleingebaeck-linux-amd64-0.0.5
|
||||||
|
```
|
||||||
|
You should see the same SHA256 hash.
|
||||||
|
|
||||||
|
2. You may also download a binary tarball for your platform,
|
||||||
|
e.g. `kleingebaeck-linux-amd64-0.0.5.tar.gz`, unpack and install
|
||||||
|
it. GNU Make is required for this:
|
||||||
|
|
||||||
|
```shell
|
||||||
|
tar xvfz kleingebaeck-linux-amd64-0.0.5.tar.gz
|
||||||
|
cd kleingebaeck-linux-amd64-0.0.5
|
||||||
|
sudo make install
|
||||||
|
```
|
||||||
|
|
||||||
|
### Installation from source
|
||||||
|
|
||||||
|
You will need the Golang toolchain in order to build from source. GNU
|
||||||
|
Make will also help but is not strictly neccessary.
|
||||||
|
|
||||||
|
If you want to compile the tool yourself, use `git clone` to clone the
|
||||||
|
repository. Then execute `go mod tidy` to install all
|
||||||
|
dependencies. Then just enter `go build` or - if you have GNU Make
|
||||||
|
installed - `make`.
|
||||||
|
|
||||||
|
To install after building either copy the binary or execute `sudo make install`.
|
||||||
|
|
||||||
|
### Using the docker image
|
||||||
|
|
||||||
|
A pre-built docker image is available, which you can use to test the
|
||||||
|
app without installing it. To download:
|
||||||
|
|
||||||
|
```shell
|
||||||
|
docker pull ghcr.io/tlinden/kleingebaeck:latest
|
||||||
|
```
|
||||||
|
|
||||||
|
To execute kleingebaeck inside the image and download ads to a local
|
||||||
|
directory, do something like this:
|
||||||
|
|
||||||
|
```shell
|
||||||
|
mkdir myads
|
||||||
|
docker run -u `id -u $USER` -v ./myads:/backup ghcr.io/tlinden/kleingebaeck:latest -u XXX -v
|
||||||
|
ls -l myads/ein-buch-mit-leeren-seiten
|
||||||
|
total 792
|
||||||
|
drwxr-xr-x 2 scip root 4096 Jan 23 12:58 ./
|
||||||
|
drwxr-xr-x 3 scip scip 4096 Jan 23 12:58 ../
|
||||||
|
-rw-r--r-- 1 scip root 131650 Jan 23 12:58 1.jpg
|
||||||
|
-rw-r--r-- 1 scip root 81832 Jan 23 12:58 2.jpg
|
||||||
|
-rw-r--r-- 1 scip root 134050 Jan 23 12:58 3.jpg
|
||||||
|
-rw-r--r-- 1 scip root 1166 Jan 23 12:58 Adlisting.txt
|
||||||
|
```
|
||||||
|
|
||||||
|
We map the local user to the one inside the image so the permission
|
||||||
|
will match. You'll need to create the directory first before executing
|
||||||
|
docker run. And the local directory `myads` will be mapped to
|
||||||
|
`/backup` inside the container.
|
||||||
|
|
||||||
|
The options `-u XXX -v` are kleingebaeck options, replace `XXX` with
|
||||||
|
your actual kleinanzeigen.de user id.
|
||||||
|
|
||||||
|
A list of available images is [here](https://codeberg.org/scip/kleingebaeck/pkgs/container/kleingebaeck/versions?filters%5Bversion_type%5D=tagged)
|
||||||
|
|
||||||
## Commandline options:
|
## Commandline options:
|
||||||
|
|
||||||
```
|
```
|
||||||
Usage: kleingebaeck [-dvVhmoc] [<ad-listing-url>,...]
|
Usage: kleingebaeck [-dvVhmoc] [<ad-listing-url>,...]
|
||||||
Options:
|
Options:
|
||||||
--user,-u <uid> Backup ads from user with uid <uid>.
|
-u --user <uid> Backup ads from user with uid <uid>.
|
||||||
--debug, -d Enable debug output.
|
-d --debug Enable debug output.
|
||||||
--verbose,-v Enable verbose output.
|
-v --verbose Enable verbose output.
|
||||||
--output-dir,-o <dir> Set output dir (default: current directory)
|
-o --outdir <dir> Set output dir (default: current directory)
|
||||||
--manual,-m Show manual.
|
-l --limit <num> Limit the ads to download to <num>, default: load all.
|
||||||
--config,-c <file> Use config file <file> (default: ~/.kleingebaeck).
|
-c --config <file> Use config file <file> (default: ~/.kleingebaeck).
|
||||||
|
--ignoreerrors Ignore HTTP errors, may lead to incomplete ad backup.
|
||||||
|
-m --manual Show manual.
|
||||||
|
-h --help Show usage.
|
||||||
|
-V --version Show program version.
|
||||||
|
|
||||||
If one or more <ad-listing-url>'s are specified, only backup those,
|
If one or more <ad-listing-url>'s are specified, only backup those,
|
||||||
otherwise backup all ads of the given user.
|
otherwise backup all ads of the given user.
|
||||||
@@ -42,16 +165,22 @@ otherwise backup all ads of the given user.
|
|||||||
## Configfile
|
## Configfile
|
||||||
|
|
||||||
You can create a config file to save typing. By default
|
You can create a config file to save typing. By default
|
||||||
`~/.kleingebaeck.hcl` is being used but you can specify one with
|
`~/.kleingebaeck` is being used but you can specify one with
|
||||||
`-c` as well.
|
`-c` as well.
|
||||||
|
|
||||||
Format is simple:
|
Format is simple:
|
||||||
|
|
||||||
```
|
```
|
||||||
user = 1010101
|
user = 1010101
|
||||||
verbose = true
|
loglevel = verbose
|
||||||
outdir = "test"
|
outdir = "test"
|
||||||
template = ""
|
```
|
||||||
|
|
||||||
|
## Environment Variables
|
||||||
|
|
||||||
|
Kleingebaeck can also be configured using environment variables. Just prefix the config variables with `KLEINGEBAECK_` and put them to upper case. Eg:
|
||||||
|
```shell
|
||||||
|
% KLEINGEBAECK_OUTDIR=/backup kleingebaeck -v
|
||||||
```
|
```
|
||||||
|
|
||||||
## Usage
|
## Usage
|
||||||
@@ -74,9 +203,11 @@ somewhat like this:
|
|||||||
```default
|
```default
|
||||||
Title: A book I sell
|
Title: A book I sell
|
||||||
Price: 99 € VB
|
Price: 99 € VB
|
||||||
|
Shipping: 6,90 €
|
||||||
Id: 1919191919
|
Id: 1919191919
|
||||||
Category: Sachbücher
|
Category: Sachbücher
|
||||||
Condition: Sehr Gut
|
Condition: Sehr Gut
|
||||||
|
Type: Buch
|
||||||
Created: 10.12.2023
|
Created: 10.12.2023
|
||||||
|
|
||||||
This is the description text.
|
This is the description text.
|
||||||
@@ -89,6 +220,52 @@ variable. The supplied sample config contains the default template.
|
|||||||
|
|
||||||
All images will be stored in the same directory.
|
All images will be stored in the same directory.
|
||||||
|
|
||||||
|
## Tool Behavior
|
||||||
|
|
||||||
|
There are a bunch of things you might want to know about the behavior
|
||||||
|
of the kleingebäck tool:
|
||||||
|
|
||||||
|
- all HTML pages and IMAGEs are always being downloaded
|
||||||
|
- we use a (customizable) user agent
|
||||||
|
- we respect HTTP cookies
|
||||||
|
- in the case of an error, the tool does 3 retries, the time it waits
|
||||||
|
between tries is longer for each retry
|
||||||
|
- image download is parallized using small time differences to look
|
||||||
|
more natural
|
||||||
|
- same images are not being overwritten on subsequent download
|
||||||
|
|
||||||
|
|
||||||
|
The latter needs to be elaborated a bit more:
|
||||||
|
|
||||||
|
If you publish an ad on kleinanzeigen.de and post images, those images
|
||||||
|
will be reduced in size by the site (by compressing and down sizing
|
||||||
|
them). This reduced images will be downloaded by kleingebäck. However,
|
||||||
|
you may still own the original images and may want to put them into
|
||||||
|
that backup directory so that you have all things for one ad together.
|
||||||
|
|
||||||
|
You can easily do that, because kleingebäck won't overwrite those
|
||||||
|
original images. It uses something called a distance hash using
|
||||||
|
[goimagehash](https://github.com/corona10/goimagehash). This
|
||||||
|
algorithmus checks the similarity of images. If an image has been
|
||||||
|
resized it is still very similar to the original one. We accept a
|
||||||
|
maximum of a distance of 5, everything above leads to overwrite.
|
||||||
|
|
||||||
|
This works with resizes, cropped and otherwise manipulated images as
|
||||||
|
long as the image still shows the original contents good enough.
|
||||||
|
|
||||||
|
Also note, that this is NOT a caching mechanism: the images will be
|
||||||
|
downloaded anyway during each run. We also can't look at the file
|
||||||
|
names because kleinanzeigen.de renames all images to numbers. And
|
||||||
|
those might even change if the user re-arranges the images.
|
||||||
|
|
||||||
|
You can override this behavior using the **--force** option. Another
|
||||||
|
option, **--ignoreerrors**, can be used to ignore all kinds of image
|
||||||
|
errors.
|
||||||
|
|
||||||
|
## Documentation
|
||||||
|
|
||||||
|
You can read the documentation [online](https://codeberg.org/scip/kleingebaeck/raw/branch/main/kleingebaeck.pod) or locally once you have installed kleingebaeck with: `kleingebaeck --manual`.
|
||||||
|
|
||||||
## Kleingebäck?
|
## Kleingebäck?
|
||||||
|
|
||||||
The name is derived from "kleinanzeigen backup": "klein" (german for
|
The name is derived from "kleinanzeigen backup": "klein" (german for
|
||||||
@@ -103,12 +280,40 @@ that's the best way for me to forget to do something.
|
|||||||
|
|
||||||
In order to report a bug, unexpected behavior, feature requests or to
|
In order to report a bug, unexpected behavior, feature requests or to
|
||||||
submit a patch, please open an issue on github:
|
submit a patch, please open an issue on github:
|
||||||
https://github.com/TLINDEN/kleingebaeck/issues.
|
https://codeberg.org/scip/kleingebaeck/issues.
|
||||||
|
|
||||||
Please repeat the failing command with debugging enabled `-d` and
|
Please repeat the failing command with debugging enabled `-d` and
|
||||||
include the output in the issue.
|
include the output in the issue.
|
||||||
|
|
||||||
## Copyright und License
|
## Related projects
|
||||||
|
|
||||||
|
I could not find any projects specifically designed to backup
|
||||||
|
kleinanzeigen.de ads, however there's a bot project which is also able
|
||||||
|
to download ads:
|
||||||
|
[kleinanzeigen-bot](https://github.com/Second-Hand-Friends/kleinanzeigen-bot/). However,
|
||||||
|
be aware that kleinanzeigen.de is actively fighting bots! Look at this
|
||||||
|
[issue](https://github.com/Second-Hand-Friends/kleinanzeigen-bot/issues/219). The
|
||||||
|
problem with these kind of bots is, that they login into your account
|
||||||
|
using your credentials. If the company is able to detect bot activity
|
||||||
|
they can associate it easily with your account and **lock you
|
||||||
|
out**. So be careful.
|
||||||
|
|
||||||
|
**kleingebäck** doesn't need to login, it just accesses public
|
||||||
|
available web pages. Kleinanzeigen.de could hardly do anything against
|
||||||
|
it, once because it is legal. There's no difference between a browser
|
||||||
|
and a commandline client. Both run on the clientside and it is not
|
||||||
|
kleinanzeigen.de's decision which software one uses to access their
|
||||||
|
pages. And second: because you can use it to download any ads, not
|
||||||
|
just yours. So it is not really clear if the activity is associated in
|
||||||
|
any way with the ad owner. In addition to that comes the fact that
|
||||||
|
kleingebäck is just a backup tool. It is not intendet to be used on a
|
||||||
|
daily basis. You cannot use it to view regular ads or maintain your
|
||||||
|
own ads. You'll need to use the mobile app or the browser page with a
|
||||||
|
login. So, in my point of view, the risk is very minimal.
|
||||||
|
|
||||||
|
There is another Tool available named [kleinanzeigen-enhanced](https://kleinanzeigen-enhanced.de/). It is a complete Ad management system targeting primarily commercial users. You have to pay a monthly fee, perhaps there's also a free version available, but I haven't checked. The tool is implemented as a Chrome browser extension, which explains why it was possible to implement it without an API. It seems to be a nice solution for power users by the looks of it. And it includes backups.
|
||||||
|
|
||||||
|
## Copyright and License
|
||||||
|
|
||||||
Licensed under the GNU GENERAL PUBLIC LICENSE version 3.
|
Licensed under the GNU GENERAL PUBLIC LICENSE version 3.
|
||||||
|
|
||||||
|
|||||||
64
config.go
@@ -1,64 +0,0 @@
|
|||||||
/*
|
|
||||||
Copyright © 2023 Thomas von Dein
|
|
||||||
|
|
||||||
This program is free software: you can redistribute it and/or modify
|
|
||||||
it under the terms of the GNU General Public License as published by
|
|
||||||
the Free Software Foundation, either version 3 of the License, or
|
|
||||||
(at your option) any later version.
|
|
||||||
|
|
||||||
This program is distributed in the hope that it will be useful,
|
|
||||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
||||||
GNU General Public License for more details.
|
|
||||||
|
|
||||||
You should have received a copy of the GNU General Public License
|
|
||||||
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
*/
|
|
||||||
package main
|
|
||||||
|
|
||||||
import (
|
|
||||||
"os"
|
|
||||||
|
|
||||||
"github.com/hashicorp/hcl/v2/hclsimple"
|
|
||||||
)
|
|
||||||
|
|
||||||
const (
|
|
||||||
VERSION string = "0.0.4"
|
|
||||||
Baseuri string = "https://www.kleinanzeigen.de"
|
|
||||||
Listuri string = "/s-bestandsliste.html"
|
|
||||||
Defaultdir string = "."
|
|
||||||
DefaultTemplate string = "Title: {{.Title}}\nPrice: {{.Price}}\nId: {{.Id}}\n" +
|
|
||||||
"Category: {{.Category}}\nCondition: {{.Condition}}\nCreated: {{.Created}}\n\n{{.Text}}\n"
|
|
||||||
Useragent string = "Mozilla/5.0 (Windows NT 10.0; Win64; x64) " +
|
|
||||||
"AppleWebKit/537.36 (KHTML, like Gecko) Chrome/120.0.0.0 Safari/537.36"
|
|
||||||
)
|
|
||||||
|
|
||||||
type Config struct {
|
|
||||||
Verbose bool `hcl:"verbose"`
|
|
||||||
User int `hcl:"user"`
|
|
||||||
Outdir string `hcl:"outdir"`
|
|
||||||
Template string `hcl:"template"`
|
|
||||||
}
|
|
||||||
|
|
||||||
func ParseConfigfile(file string) (*Config, error) {
|
|
||||||
c := Config{}
|
|
||||||
if path, err := os.Stat(file); !os.IsNotExist(err) {
|
|
||||||
if !path.IsDir() {
|
|
||||||
configstring, err := os.ReadFile(file)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
err = hclsimple.Decode(
|
|
||||||
path.Name(), configstring,
|
|
||||||
nil, &c,
|
|
||||||
)
|
|
||||||
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return &c, nil
|
|
||||||
}
|
|
||||||
19
example.hcl
@@ -1,19 +0,0 @@
|
|||||||
#
|
|
||||||
# kleingebaeck sample configuration file.
|
|
||||||
# put this to ~/.kleingebaeck.hcl.
|
|
||||||
#
|
|
||||||
# Comments start with the '#' character.
|
|
||||||
|
|
||||||
# kleinanzeigen.de user-id. must be an unquoted number
|
|
||||||
user = 00000000
|
|
||||||
|
|
||||||
# enable verbose output (same as -v), may be true or false.
|
|
||||||
verbose = true
|
|
||||||
|
|
||||||
# directory where to store downloaded ads. kleingebaeck will try to
|
|
||||||
# create it. must be a quoted string.
|
|
||||||
outdir = "test"
|
|
||||||
|
|
||||||
# template. leave empty to use the default one, which is:
|
|
||||||
# "Title: {{.Title}}\nPrice: {{.Price}}\nId: {{.Id}}\nCategory: {{.Category}}\nCondition: {{.Condition}}\nCreated: {{.Created}}\n\n{{.Text}}\n"
|
|
||||||
template = ""
|
|
||||||
21
go.mod
@@ -1,21 +0,0 @@
|
|||||||
module kleingebaeck
|
|
||||||
|
|
||||||
go 1.21
|
|
||||||
|
|
||||||
require (
|
|
||||||
astuart.co/goq v1.0.0 // indirect
|
|
||||||
github.com/PuerkitoBio/goquery v1.5.0 // indirect
|
|
||||||
github.com/agext/levenshtein v1.2.1 // indirect
|
|
||||||
github.com/andybalholm/cascadia v1.0.0 // indirect
|
|
||||||
github.com/apparentlymart/go-textseg/v13 v13.0.0 // indirect
|
|
||||||
github.com/apparentlymart/go-textseg/v15 v15.0.0 // indirect
|
|
||||||
github.com/google/go-cmp v0.3.1 // indirect
|
|
||||||
github.com/hashicorp/hcl/v2 v2.19.1 // indirect
|
|
||||||
github.com/lmittmann/tint v1.0.3 // indirect
|
|
||||||
github.com/mitchellh/go-wordwrap v0.0.0-20150314170334-ad45545899c7 // indirect
|
|
||||||
github.com/spf13/pflag v1.0.5 // indirect
|
|
||||||
github.com/zclconf/go-cty v1.13.0 // indirect
|
|
||||||
golang.org/x/net v0.0.0-20190606173856-1492cefac77f // indirect
|
|
||||||
golang.org/x/text v0.11.0 // indirect
|
|
||||||
|
|
||||||
)
|
|
||||||
37
go.sum
@@ -1,37 +0,0 @@
|
|||||||
astuart.co/goq v1.0.0 h1:nnYIhu/Z/j0VaX9Dp+pmh2Uh7ldEz6XfgSg+bAY5Yrw=
|
|
||||||
astuart.co/goq v1.0.0/go.mod h1:+fokcnFrO8Pw2fj8drdStJvzoMFebJH69rw8IC21rno=
|
|
||||||
github.com/PuerkitoBio/goquery v1.5.0 h1:uGvmFXOA73IKluu/F84Xd1tt/z07GYm8X49XKHP7EJk=
|
|
||||||
github.com/PuerkitoBio/goquery v1.5.0/go.mod h1:qD2PgZ9lccMbQlc7eEOjaeRlFQON7xY8kdmcsrnKqMg=
|
|
||||||
github.com/agext/levenshtein v1.2.1 h1:QmvMAjj2aEICytGiWzmxoE0x2KZvE0fvmqMOfy2tjT8=
|
|
||||||
github.com/agext/levenshtein v1.2.1/go.mod h1:JEDfjyjHDjOF/1e4FlBE/PkbqA9OfWu2ki2W0IB5558=
|
|
||||||
github.com/andybalholm/cascadia v1.0.0 h1:hOCXnnZ5A+3eVDX8pvgl4kofXv2ELss0bKcqRySc45o=
|
|
||||||
github.com/andybalholm/cascadia v1.0.0/go.mod h1:GsXiBklL0woXo1j/WYWtSYYC4ouU9PqHO0sqidkEA4Y=
|
|
||||||
github.com/apparentlymart/go-textseg/v13 v13.0.0 h1:Y+KvPE1NYz0xl601PVImeQfFyEy6iT90AvPUL1NNfNw=
|
|
||||||
github.com/apparentlymart/go-textseg/v13 v13.0.0/go.mod h1:ZK2fH7c4NqDTLtiYLvIkEghdlcqw7yxLeM89kiTRPUo=
|
|
||||||
github.com/apparentlymart/go-textseg/v15 v15.0.0 h1:uYvfpb3DyLSCGWnctWKGj857c6ew1u1fNQOlOtuGxQY=
|
|
||||||
github.com/apparentlymart/go-textseg/v15 v15.0.0/go.mod h1:K8XmNZdhEBkdlyDdvbmmsvpAG721bKi0joRfFdHIWJ4=
|
|
||||||
github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
|
||||||
github.com/google/go-cmp v0.3.1 h1:Xye71clBPdm5HgqGwUkwhbynsUJZhDbS20FvLhQ2izg=
|
|
||||||
github.com/google/go-cmp v0.3.1/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU=
|
|
||||||
github.com/hashicorp/hcl/v2 v2.19.1 h1://i05Jqznmb2EXqa39Nsvyan2o5XyMowW5fnCKW5RPI=
|
|
||||||
github.com/hashicorp/hcl/v2 v2.19.1/go.mod h1:ThLC89FV4p9MPW804KVbe/cEXoQ8NZEh+JtMeeGErHE=
|
|
||||||
github.com/lmittmann/tint v1.0.3 h1:W5PHeA2D8bBJVvabNfQD/XW9HPLZK1XoPZH0cq8NouQ=
|
|
||||||
github.com/lmittmann/tint v1.0.3/go.mod h1:HIS3gSy7qNwGCj+5oRjAutErFBl4BzdQP6cJZ0NfMwE=
|
|
||||||
github.com/mitchellh/go-wordwrap v0.0.0-20150314170334-ad45545899c7 h1:DpOJ2HYzCv8LZP15IdmG+YdwD2luVPHITV96TkirNBM=
|
|
||||||
github.com/mitchellh/go-wordwrap v0.0.0-20150314170334-ad45545899c7/go.mod h1:ZXFpozHsX6DPmq2I0TCekCxypsnAUbP2oI0UX1GXzOo=
|
|
||||||
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
|
|
||||||
github.com/spf13/pflag v1.0.5 h1:iy+VFUOCP1a+8yFto/drg2CJ5u0yRoB7fZw3DKv/JXA=
|
|
||||||
github.com/spf13/pflag v1.0.5/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg=
|
|
||||||
github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
|
|
||||||
github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI=
|
|
||||||
github.com/zclconf/go-cty v1.13.0 h1:It5dfKTTZHe9aeppbNOda3mN7Ag7sg6QkBNm6TkyFa0=
|
|
||||||
github.com/zclconf/go-cty v1.13.0/go.mod h1:YKQzy/7pZ7iq2jNFzy5go57xdxdWoLLpaEp4u238AE0=
|
|
||||||
golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w=
|
|
||||||
golang.org/x/net v0.0.0-20180218175443-cbe0f9307d01/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
|
|
||||||
golang.org/x/net v0.0.0-20181114220301-adae6a3d119a/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
|
|
||||||
golang.org/x/net v0.0.0-20190606173856-1492cefac77f h1:IWHgpgFqnL5AhBUBZSgBdjl2vkQUEzcY+JNKWfcgAU0=
|
|
||||||
golang.org/x/net v0.0.0-20190606173856-1492cefac77f/go.mod h1:HSz+uSET+XFnRR8LxR5pz3Of3rY3CfYBVs4xY44aLks=
|
|
||||||
golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
|
|
||||||
golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
|
|
||||||
golang.org/x/text v0.11.0 h1:LAntKIrcmeSKERyiOh0XMV39LXS8IE9UL2yP7+f5ij4=
|
|
||||||
golang.org/x/text v0.11.0/go.mod h1:TvPlkZtksWOMsz7fbANvkp4WM8x/WCo/om8BMLbz+aE=
|
|
||||||
225
kleingebaeck.1
@@ -1,225 +0,0 @@
|
|||||||
.\" Automatically generated by Pod::Man 4.14 (Pod::Simple 3.42)
|
|
||||||
.\"
|
|
||||||
.\" Standard preamble:
|
|
||||||
.\" ========================================================================
|
|
||||||
.de Sp \" Vertical space (when we can't use .PP)
|
|
||||||
.if t .sp .5v
|
|
||||||
.if n .sp
|
|
||||||
..
|
|
||||||
.de Vb \" Begin verbatim text
|
|
||||||
.ft CW
|
|
||||||
.nf
|
|
||||||
.ne \\$1
|
|
||||||
..
|
|
||||||
.de Ve \" End verbatim text
|
|
||||||
.ft R
|
|
||||||
.fi
|
|
||||||
..
|
|
||||||
.\" Set up some character translations and predefined strings. \*(-- will
|
|
||||||
.\" give an unbreakable dash, \*(PI will give pi, \*(L" will give a left
|
|
||||||
.\" double quote, and \*(R" will give a right double quote. \*(C+ will
|
|
||||||
.\" give a nicer C++. Capital omega is used to do unbreakable dashes and
|
|
||||||
.\" therefore won't be available. \*(C` and \*(C' expand to `' in nroff,
|
|
||||||
.\" nothing in troff, for use with C<>.
|
|
||||||
.tr \(*W-
|
|
||||||
.ds C+ C\v'-.1v'\h'-1p'\s-2+\h'-1p'+\s0\v'.1v'\h'-1p'
|
|
||||||
.ie n \{\
|
|
||||||
. ds -- \(*W-
|
|
||||||
. ds PI pi
|
|
||||||
. if (\n(.H=4u)&(1m=24u) .ds -- \(*W\h'-12u'\(*W\h'-12u'-\" diablo 10 pitch
|
|
||||||
. if (\n(.H=4u)&(1m=20u) .ds -- \(*W\h'-12u'\(*W\h'-8u'-\" diablo 12 pitch
|
|
||||||
. ds L" ""
|
|
||||||
. ds R" ""
|
|
||||||
. ds C` ""
|
|
||||||
. ds C' ""
|
|
||||||
'br\}
|
|
||||||
.el\{\
|
|
||||||
. ds -- \|\(em\|
|
|
||||||
. ds PI \(*p
|
|
||||||
. ds L" ``
|
|
||||||
. ds R" ''
|
|
||||||
. ds C`
|
|
||||||
. ds C'
|
|
||||||
'br\}
|
|
||||||
.\"
|
|
||||||
.\" Escape single quotes in literal strings from groff's Unicode transform.
|
|
||||||
.ie \n(.g .ds Aq \(aq
|
|
||||||
.el .ds Aq '
|
|
||||||
.\"
|
|
||||||
.\" If the F register is >0, we'll generate index entries on stderr for
|
|
||||||
.\" titles (.TH), headers (.SH), subsections (.SS), items (.Ip), and index
|
|
||||||
.\" entries marked with X<> in POD. Of course, you'll have to process the
|
|
||||||
.\" output yourself in some meaningful fashion.
|
|
||||||
.\"
|
|
||||||
.\" Avoid warning from groff about undefined register 'F'.
|
|
||||||
.de IX
|
|
||||||
..
|
|
||||||
.nr rF 0
|
|
||||||
.if \n(.g .if rF .nr rF 1
|
|
||||||
.if (\n(rF:(\n(.g==0)) \{\
|
|
||||||
. if \nF \{\
|
|
||||||
. de IX
|
|
||||||
. tm Index:\\$1\t\\n%\t"\\$2"
|
|
||||||
..
|
|
||||||
. if !\nF==2 \{\
|
|
||||||
. nr % 0
|
|
||||||
. nr F 2
|
|
||||||
. \}
|
|
||||||
. \}
|
|
||||||
.\}
|
|
||||||
.rr rF
|
|
||||||
.\"
|
|
||||||
.\" Accent mark definitions (@(#)ms.acc 1.5 88/02/08 SMI; from UCB 4.2).
|
|
||||||
.\" Fear. Run. Save yourself. No user-serviceable parts.
|
|
||||||
. \" fudge factors for nroff and troff
|
|
||||||
.if n \{\
|
|
||||||
. ds #H 0
|
|
||||||
. ds #V .8m
|
|
||||||
. ds #F .3m
|
|
||||||
. ds #[ \f1
|
|
||||||
. ds #] \fP
|
|
||||||
.\}
|
|
||||||
.if t \{\
|
|
||||||
. ds #H ((1u-(\\\\n(.fu%2u))*.13m)
|
|
||||||
. ds #V .6m
|
|
||||||
. ds #F 0
|
|
||||||
. ds #[ \&
|
|
||||||
. ds #] \&
|
|
||||||
.\}
|
|
||||||
. \" simple accents for nroff and troff
|
|
||||||
.if n \{\
|
|
||||||
. ds ' \&
|
|
||||||
. ds ` \&
|
|
||||||
. ds ^ \&
|
|
||||||
. ds , \&
|
|
||||||
. ds ~ ~
|
|
||||||
. ds /
|
|
||||||
.\}
|
|
||||||
.if t \{\
|
|
||||||
. ds ' \\k:\h'-(\\n(.wu*8/10-\*(#H)'\'\h"|\\n:u"
|
|
||||||
. ds ` \\k:\h'-(\\n(.wu*8/10-\*(#H)'\`\h'|\\n:u'
|
|
||||||
. ds ^ \\k:\h'-(\\n(.wu*10/11-\*(#H)'^\h'|\\n:u'
|
|
||||||
. ds , \\k:\h'-(\\n(.wu*8/10)',\h'|\\n:u'
|
|
||||||
. ds ~ \\k:\h'-(\\n(.wu-\*(#H-.1m)'~\h'|\\n:u'
|
|
||||||
. ds / \\k:\h'-(\\n(.wu*8/10-\*(#H)'\z\(sl\h'|\\n:u'
|
|
||||||
.\}
|
|
||||||
. \" troff and (daisy-wheel) nroff accents
|
|
||||||
.ds : \\k:\h'-(\\n(.wu*8/10-\*(#H+.1m+\*(#F)'\v'-\*(#V'\z.\h'.2m+\*(#F'.\h'|\\n:u'\v'\*(#V'
|
|
||||||
.ds 8 \h'\*(#H'\(*b\h'-\*(#H'
|
|
||||||
.ds o \\k:\h'-(\\n(.wu+\w'\(de'u-\*(#H)/2u'\v'-.3n'\*(#[\z\(de\v'.3n'\h'|\\n:u'\*(#]
|
|
||||||
.ds d- \h'\*(#H'\(pd\h'-\w'~'u'\v'-.25m'\f2\(hy\fP\v'.25m'\h'-\*(#H'
|
|
||||||
.ds D- D\\k:\h'-\w'D'u'\v'-.11m'\z\(hy\v'.11m'\h'|\\n:u'
|
|
||||||
.ds th \*(#[\v'.3m'\s+1I\s-1\v'-.3m'\h'-(\w'I'u*2/3)'\s-1o\s+1\*(#]
|
|
||||||
.ds Th \*(#[\s+2I\s-2\h'-\w'I'u*3/5'\v'-.3m'o\v'.3m'\*(#]
|
|
||||||
.ds ae a\h'-(\w'a'u*4/10)'e
|
|
||||||
.ds Ae A\h'-(\w'A'u*4/10)'E
|
|
||||||
. \" corrections for vroff
|
|
||||||
.if v .ds ~ \\k:\h'-(\\n(.wu*9/10-\*(#H)'\s-2\u~\d\s+2\h'|\\n:u'
|
|
||||||
.if v .ds ^ \\k:\h'-(\\n(.wu*10/11-\*(#H)'\v'-.4m'^\v'.4m'\h'|\\n:u'
|
|
||||||
. \" for low resolution devices (crt and lpr)
|
|
||||||
.if \n(.H>23 .if \n(.V>19 \
|
|
||||||
\{\
|
|
||||||
. ds : e
|
|
||||||
. ds 8 ss
|
|
||||||
. ds o a
|
|
||||||
. ds d- d\h'-1'\(ga
|
|
||||||
. ds D- D\h'-1'\(hy
|
|
||||||
. ds th \o'bp'
|
|
||||||
. ds Th \o'LP'
|
|
||||||
. ds ae ae
|
|
||||||
. ds Ae AE
|
|
||||||
.\}
|
|
||||||
.rm #[ #] #H #V #F C
|
|
||||||
.\" ========================================================================
|
|
||||||
.\"
|
|
||||||
.IX Title "KLEINGEBAECK 1"
|
|
||||||
.TH KLEINGEBAECK 1 "2023-12-17" "1" "User Commands"
|
|
||||||
.\" For nroff, turn off justification. Always turn off hyphenation; it makes
|
|
||||||
.\" way too many mistakes in technical documents.
|
|
||||||
.if n .ad l
|
|
||||||
.nh
|
|
||||||
.SH "NAME"
|
|
||||||
kleingebaeck \- kleinanzeigen.de backup tool
|
|
||||||
.SH "SYNOPSYS"
|
|
||||||
.IX Header "SYNOPSYS"
|
|
||||||
.Vb 9
|
|
||||||
\& This is kleingebaeck, the kleinanzeigen.de backup tool.
|
|
||||||
\& Usage: kleingebaeck [\-dvVhmoc] [<ad\-listing\-url>,...]
|
|
||||||
\& Options:
|
|
||||||
\& \-\-user,\-u <uid> Backup ads from user with uid <uid>.
|
|
||||||
\& \-\-debug, \-d Enable debug output.
|
|
||||||
\& \-\-verbose,\-v Enable verbose output.
|
|
||||||
\& \-\-output\-dir,\-o <dir> Set output dir (default: current directory)
|
|
||||||
\& \-\-manual,\-m Show manual.
|
|
||||||
\& \-\-config,\-c <file> Use config file <file> (default: ~/.kleingebaeck).
|
|
||||||
.Ve
|
|
||||||
.SH "DESCRIPTION"
|
|
||||||
.IX Header "DESCRIPTION"
|
|
||||||
This tool can be used to backup ads on the german ad page <https://kleinanzeigen.de>.
|
|
||||||
.PP
|
|
||||||
It downloads all (or only the specified ones) ads of one user into a
|
|
||||||
directory, each ad into its own subdirectory. The backup will contain
|
|
||||||
a textfile \fBAdlisting.txt\fR which contains the ad contents such as
|
|
||||||
title, body, price etc. All images will be downloaded as well.
|
|
||||||
.SH "CONFIGURATION"
|
|
||||||
.IX Header "CONFIGURATION"
|
|
||||||
You can create a config file to save typing. By default
|
|
||||||
\&\f(CW\*(C`~/.kleingebaeck.hcl\*(C'\fR is being used but you can specify one with
|
|
||||||
\&\f(CW\*(C`\-c\*(C'\fR as well.
|
|
||||||
.PP
|
|
||||||
Format is simple:
|
|
||||||
.PP
|
|
||||||
.Vb 4
|
|
||||||
\& user = 1010101
|
|
||||||
\& verbose = true
|
|
||||||
\& outdir = "test"
|
|
||||||
\& template = ""
|
|
||||||
.Ve
|
|
||||||
.PP
|
|
||||||
Be carefull if you want to change the template. The default one looks like this:
|
|
||||||
.PP
|
|
||||||
.Vb 1
|
|
||||||
\& Title: {{.Title}}\enPrice: {{.Price}}\enId: {{.Id}}\enCategory: {{.Category}}\enCondition: {{.Condition}}\enCreated: {{.Created}}\en\en{{.Text}}\en
|
|
||||||
.Ve
|
|
||||||
.PP
|
|
||||||
You can left out certain fields and use any formatting you like. Refer
|
|
||||||
to <https://pkg.go.dev/text/template> for details how to write a template.
|
|
||||||
.SH "SETUP"
|
|
||||||
.IX Header "SETUP"
|
|
||||||
To setup the tool, you need to lookup your userid on
|
|
||||||
kleinanzeigen.de. Go to your ad overview page while \s-1NOT\s0 being logged
|
|
||||||
in:
|
|
||||||
.PP
|
|
||||||
.Vb 1
|
|
||||||
\& https://www.kleinanzeigen.de/s\-bestandsliste.html?userId=XXXXXX
|
|
||||||
.Ve
|
|
||||||
.PP
|
|
||||||
The \fB\s-1XXXXX\s0\fR part is your userid.
|
|
||||||
.PP
|
|
||||||
Put it into the configfile as outlined above. Also specify an output
|
|
||||||
directory. Then just execute \f(CW\*(C`kleingebaeck\*(C'\fR.
|
|
||||||
.PP
|
|
||||||
You can use the \fB\-v\fR option to get verbose output or \fB\-d\fR to enable
|
|
||||||
debugging.
|
|
||||||
.SH "BUGS"
|
|
||||||
.IX Header "BUGS"
|
|
||||||
In order to report a bug, unexpected behavior, feature requests
|
|
||||||
or to submit a patch, please open an issue on github:
|
|
||||||
<https://github.com/TLINDEN/kleingebaeck/issues>.
|
|
||||||
.PP
|
|
||||||
Please repeat the failing command with debugging enabled \f(CW\*(C`\-d\*(C'\fR and
|
|
||||||
include the output in the issue.
|
|
||||||
.SH "LIMITATIONS"
|
|
||||||
.IX Header "LIMITATIONS"
|
|
||||||
The \f(CW\*(C`kleingebaeck\*(C'\fR doesn't currently check if it has downloaded a
|
|
||||||
file already, so it downloads everything again every time you execute
|
|
||||||
it. Be aware of it. This will change in the future.
|
|
||||||
.PP
|
|
||||||
Also there's currently no parallelization implemented. This will
|
|
||||||
change in the future.
|
|
||||||
.SH "LICENSE"
|
|
||||||
.IX Header "LICENSE"
|
|
||||||
Licensed under the \s-1GNU GENERAL PUBLIC LICENSE\s0 version 3.
|
|
||||||
.SH "Author"
|
|
||||||
.IX Header "Author"
|
|
||||||
T.v.Dein <tom \s-1AT\s0 vondein \s-1DOT\s0 org>
|
|
||||||
@@ -1,84 +0,0 @@
|
|||||||
package main
|
|
||||||
|
|
||||||
var manpage = `
|
|
||||||
NAME
|
|
||||||
kleingebaeck - kleinanzeigen.de backup tool
|
|
||||||
|
|
||||||
SYNOPSYS
|
|
||||||
This is kleingebaeck, the kleinanzeigen.de backup tool.
|
|
||||||
Usage: kleingebaeck [-dvVhmoc] [<ad-listing-url>,...]
|
|
||||||
Options:
|
|
||||||
--user,-u <uid> Backup ads from user with uid <uid>.
|
|
||||||
--debug, -d Enable debug output.
|
|
||||||
--verbose,-v Enable verbose output.
|
|
||||||
--output-dir,-o <dir> Set output dir (default: current directory)
|
|
||||||
--manual,-m Show manual.
|
|
||||||
--config,-c <file> Use config file <file> (default: ~/.kleingebaeck).
|
|
||||||
|
|
||||||
DESCRIPTION
|
|
||||||
This tool can be used to backup ads on the german ad page
|
|
||||||
<https://kleinanzeigen.de>.
|
|
||||||
|
|
||||||
It downloads all (or only the specified ones) ads of one user into a
|
|
||||||
directory, each ad into its own subdirectory. The backup will contain a
|
|
||||||
textfile Adlisting.txt which contains the ad contents such as title,
|
|
||||||
body, price etc. All images will be downloaded as well.
|
|
||||||
|
|
||||||
CONFIGURATION
|
|
||||||
You can create a config file to save typing. By default
|
|
||||||
"~/.kleingebaeck.hcl" is being used but you can specify one with "-c" as
|
|
||||||
well.
|
|
||||||
|
|
||||||
Format is simple:
|
|
||||||
|
|
||||||
user = 1010101
|
|
||||||
verbose = true
|
|
||||||
outdir = "test"
|
|
||||||
template = ""
|
|
||||||
|
|
||||||
Be carefull if you want to change the template. The default one looks
|
|
||||||
like this:
|
|
||||||
|
|
||||||
Title: {{.Title}}\nPrice: {{.Price}}\nId: {{.Id}}\nCategory: {{.Category}}\nCondition: {{.Condition}}\nCreated: {{.Created}}\n\n{{.Text}}\n
|
|
||||||
|
|
||||||
You can left out certain fields and use any formatting you like. Refer
|
|
||||||
to <https://pkg.go.dev/text/template> for details how to write a
|
|
||||||
template.
|
|
||||||
|
|
||||||
SETUP
|
|
||||||
To setup the tool, you need to lookup your userid on kleinanzeigen.de.
|
|
||||||
Go to your ad overview page while NOT being logged in:
|
|
||||||
|
|
||||||
https://www.kleinanzeigen.de/s-bestandsliste.html?userId=XXXXXX
|
|
||||||
|
|
||||||
The XXXXX part is your userid.
|
|
||||||
|
|
||||||
Put it into the configfile as outlined above. Also specify an output
|
|
||||||
directory. Then just execute "kleingebaeck".
|
|
||||||
|
|
||||||
You can use the -v option to get verbose output or -d to enable
|
|
||||||
debugging.
|
|
||||||
|
|
||||||
BUGS
|
|
||||||
In order to report a bug, unexpected behavior, feature requests or to
|
|
||||||
submit a patch, please open an issue on github:
|
|
||||||
<https://github.com/TLINDEN/kleingebaeck/issues>.
|
|
||||||
|
|
||||||
Please repeat the failing command with debugging enabled "-d" and
|
|
||||||
include the output in the issue.
|
|
||||||
|
|
||||||
LIMITATIONS
|
|
||||||
The "kleingebaeck" doesn't currently check if it has downloaded a file
|
|
||||||
already, so it downloads everything again every time you execute it. Be
|
|
||||||
aware of it. This will change in the future.
|
|
||||||
|
|
||||||
Also there's currently no parallelization implemented. This will change
|
|
||||||
in the future.
|
|
||||||
|
|
||||||
LICENSE
|
|
||||||
Licensed under the GNU GENERAL PUBLIC LICENSE version 3.
|
|
||||||
|
|
||||||
Author
|
|
||||||
T.v.Dein <tom AT vondein DOT org>
|
|
||||||
|
|
||||||
`
|
|
||||||
@@ -1,90 +0,0 @@
|
|||||||
=head1 NAME
|
|
||||||
|
|
||||||
kleingebaeck - kleinanzeigen.de backup tool
|
|
||||||
|
|
||||||
=head1 SYNOPSYS
|
|
||||||
|
|
||||||
This is kleingebaeck, the kleinanzeigen.de backup tool.
|
|
||||||
Usage: kleingebaeck [-dvVhmoc] [<ad-listing-url>,...]
|
|
||||||
Options:
|
|
||||||
--user,-u <uid> Backup ads from user with uid <uid>.
|
|
||||||
--debug, -d Enable debug output.
|
|
||||||
--verbose,-v Enable verbose output.
|
|
||||||
--output-dir,-o <dir> Set output dir (default: current directory)
|
|
||||||
--manual,-m Show manual.
|
|
||||||
--config,-c <file> Use config file <file> (default: ~/.kleingebaeck).
|
|
||||||
|
|
||||||
=head1 DESCRIPTION
|
|
||||||
|
|
||||||
This tool can be used to backup ads on the german ad page L<https://kleinanzeigen.de>.
|
|
||||||
|
|
||||||
It downloads all (or only the specified ones) ads of one user into a
|
|
||||||
directory, each ad into its own subdirectory. The backup will contain
|
|
||||||
a textfile B<Adlisting.txt> which contains the ad contents such as
|
|
||||||
title, body, price etc. All images will be downloaded as well.
|
|
||||||
|
|
||||||
=head1 CONFIGURATION
|
|
||||||
|
|
||||||
You can create a config file to save typing. By default
|
|
||||||
C<~/.kleingebaeck.hcl> is being used but you can specify one with
|
|
||||||
C<-c> as well.
|
|
||||||
|
|
||||||
Format is simple:
|
|
||||||
|
|
||||||
user = 1010101
|
|
||||||
verbose = true
|
|
||||||
outdir = "test"
|
|
||||||
template = ""
|
|
||||||
|
|
||||||
Be carefull if you want to change the template. The default one looks like this:
|
|
||||||
|
|
||||||
Title: {{.Title}}\nPrice: {{.Price}}\nId: {{.Id}}\nCategory: {{.Category}}\nCondition: {{.Condition}}\nCreated: {{.Created}}\n\n{{.Text}}\n
|
|
||||||
|
|
||||||
You can left out certain fields and use any formatting you like. Refer
|
|
||||||
to L<https://pkg.go.dev/text/template> for details how to write a template.
|
|
||||||
|
|
||||||
=head1 SETUP
|
|
||||||
|
|
||||||
To setup the tool, you need to lookup your userid on
|
|
||||||
kleinanzeigen.de. Go to your ad overview page while NOT being logged
|
|
||||||
in:
|
|
||||||
|
|
||||||
https://www.kleinanzeigen.de/s-bestandsliste.html?userId=XXXXXX
|
|
||||||
|
|
||||||
The B<XXXXX> part is your userid.
|
|
||||||
|
|
||||||
Put it into the configfile as outlined above. Also specify an output
|
|
||||||
directory. Then just execute C<kleingebaeck>.
|
|
||||||
|
|
||||||
You can use the B<-v> option to get verbose output or B<-d> to enable
|
|
||||||
debugging.
|
|
||||||
|
|
||||||
=head1 BUGS
|
|
||||||
|
|
||||||
In order to report a bug, unexpected behavior, feature requests
|
|
||||||
or to submit a patch, please open an issue on github:
|
|
||||||
L<https://github.com/TLINDEN/kleingebaeck/issues>.
|
|
||||||
|
|
||||||
Please repeat the failing command with debugging enabled C<-d> and
|
|
||||||
include the output in the issue.
|
|
||||||
|
|
||||||
=head1 LIMITATIONS
|
|
||||||
|
|
||||||
The C<kleingebaeck> doesn't currently check if it has downloaded a
|
|
||||||
file already, so it downloads everything again every time you execute
|
|
||||||
it. Be aware of it. This will change in the future.
|
|
||||||
|
|
||||||
Also there's currently no parallelization implemented. This will
|
|
||||||
change in the future.
|
|
||||||
|
|
||||||
=head1 LICENSE
|
|
||||||
|
|
||||||
Licensed under the GNU GENERAL PUBLIC LICENSE version 3.
|
|
||||||
|
|
||||||
=head1 Author
|
|
||||||
|
|
||||||
T.v.Dein <tom AT vondein DOT org>
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
=cut
|
|
||||||
189
main.go
@@ -1,189 +0,0 @@
|
|||||||
/*
|
|
||||||
Copyright © 2023 Thomas von Dein
|
|
||||||
|
|
||||||
This program is free software: you can redistribute it and/or modify
|
|
||||||
it under the terms of the GNU General Public License as published by
|
|
||||||
the Free Software Foundation, either version 3 of the License, or
|
|
||||||
(at your option) any later version.
|
|
||||||
|
|
||||||
This program is distributed in the hope that it will be useful,
|
|
||||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
||||||
GNU General Public License for more details.
|
|
||||||
|
|
||||||
You should have received a copy of the GNU General Public License
|
|
||||||
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
*/
|
|
||||||
|
|
||||||
package main
|
|
||||||
|
|
||||||
import (
|
|
||||||
"errors"
|
|
||||||
"fmt"
|
|
||||||
"log/slog"
|
|
||||||
"os"
|
|
||||||
"runtime/debug"
|
|
||||||
|
|
||||||
"github.com/lmittmann/tint"
|
|
||||||
flag "github.com/spf13/pflag"
|
|
||||||
)
|
|
||||||
|
|
||||||
const Usage string = `This is kleingebaeck, the kleinanzeigen.de backup tool.
|
|
||||||
Usage: kleingebaeck [-dvVhmoc] [<ad-listing-url>,...]
|
|
||||||
Options:
|
|
||||||
--user,-u <uid> Backup ads from user with uid <uid>.
|
|
||||||
--debug, -d Enable debug output.
|
|
||||||
--verbose,-v Enable verbose output.
|
|
||||||
--output-dir,-o <dir> Set output dir (default: current directory)
|
|
||||||
--manual,-m Show manual.
|
|
||||||
--config,-c <file> Use config file <file> (default: ~/.kleingebaeck).
|
|
||||||
|
|
||||||
If one or more <ad-listing-url>'s are specified, only backup those,
|
|
||||||
otherwise backup all ads of the given user.`
|
|
||||||
|
|
||||||
const LevelNotice = slog.Level(2)
|
|
||||||
|
|
||||||
func main() {
|
|
||||||
os.Exit(Main())
|
|
||||||
}
|
|
||||||
|
|
||||||
func Main() int {
|
|
||||||
logLevel := &slog.LevelVar{}
|
|
||||||
opts := &tint.Options{
|
|
||||||
Level: logLevel,
|
|
||||||
AddSource: false,
|
|
||||||
ReplaceAttr: func(groups []string, a slog.Attr) slog.Attr {
|
|
||||||
// Remove time from the output
|
|
||||||
if a.Key == slog.TimeKey {
|
|
||||||
return slog.Attr{}
|
|
||||||
}
|
|
||||||
return a
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
logLevel.Set(LevelNotice)
|
|
||||||
var handler slog.Handler = tint.NewHandler(os.Stdout, opts)
|
|
||||||
logger := slog.New(handler)
|
|
||||||
slog.SetDefault(logger)
|
|
||||||
|
|
||||||
showversion := false
|
|
||||||
showhelp := false
|
|
||||||
showmanual := false
|
|
||||||
enabledebug := false
|
|
||||||
enableverbose := false
|
|
||||||
uid := 0
|
|
||||||
configfile := os.Getenv("HOME") + "/.kleingebaeck.hcl"
|
|
||||||
dir := ""
|
|
||||||
|
|
||||||
flag.BoolVarP(&enabledebug, "debug", "d", false, "debug mode")
|
|
||||||
flag.BoolVarP(&enableverbose, "verbose", "v", false, "be verbose")
|
|
||||||
flag.BoolVarP(&showversion, "version", "V", false, "show version")
|
|
||||||
flag.BoolVarP(&showhelp, "help", "h", false, "show usage")
|
|
||||||
flag.BoolVarP(&showmanual, "manual", "m", false, "show manual")
|
|
||||||
flag.IntVarP(&uid, "user", "u", uid, "user id")
|
|
||||||
flag.StringVarP(&dir, "output-dir", "o", dir, "where to store ads")
|
|
||||||
flag.StringVarP(&configfile, "config", "c", configfile, "config file")
|
|
||||||
|
|
||||||
flag.Parse()
|
|
||||||
|
|
||||||
if showversion {
|
|
||||||
fmt.Printf("This is kleingebaeck version %s\n", VERSION)
|
|
||||||
return 0
|
|
||||||
}
|
|
||||||
|
|
||||||
if showhelp {
|
|
||||||
fmt.Println(Usage)
|
|
||||||
return 0
|
|
||||||
}
|
|
||||||
|
|
||||||
if showmanual {
|
|
||||||
err := man()
|
|
||||||
if err != nil {
|
|
||||||
return Die(err)
|
|
||||||
}
|
|
||||||
return 0
|
|
||||||
}
|
|
||||||
|
|
||||||
conf, err := ParseConfigfile(configfile)
|
|
||||||
if err != nil {
|
|
||||||
return Die(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
if enableverbose || conf.Verbose {
|
|
||||||
logLevel.Set(slog.LevelInfo)
|
|
||||||
}
|
|
||||||
|
|
||||||
if enabledebug {
|
|
||||||
// we're using a more verbose logger in debug mode
|
|
||||||
buildInfo, _ := debug.ReadBuildInfo()
|
|
||||||
opts := &tint.Options{
|
|
||||||
Level: logLevel,
|
|
||||||
AddSource: true,
|
|
||||||
}
|
|
||||||
|
|
||||||
logLevel.Set(slog.LevelDebug)
|
|
||||||
var handler slog.Handler = tint.NewHandler(os.Stdout, opts)
|
|
||||||
debuglogger := slog.New(handler).With(
|
|
||||||
slog.Group("program_info",
|
|
||||||
slog.Int("pid", os.Getpid()),
|
|
||||||
slog.String("go_version", buildInfo.GoVersion),
|
|
||||||
),
|
|
||||||
)
|
|
||||||
slog.SetDefault(debuglogger)
|
|
||||||
}
|
|
||||||
|
|
||||||
slog.Debug("config", "conf", conf)
|
|
||||||
|
|
||||||
if len(dir) == 0 {
|
|
||||||
if len(conf.Outdir) > 0 {
|
|
||||||
dir = conf.Outdir
|
|
||||||
} else {
|
|
||||||
dir = Defaultdir
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// prepare output dir
|
|
||||||
err = Mkdir(dir)
|
|
||||||
if err != nil {
|
|
||||||
return Die(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// which template to use
|
|
||||||
template := DefaultTemplate
|
|
||||||
if len(conf.Template) > 0 {
|
|
||||||
template = conf.Template
|
|
||||||
}
|
|
||||||
|
|
||||||
// directly backup ad listing[s]
|
|
||||||
if len(flag.Args()) >= 1 {
|
|
||||||
for _, uri := range flag.Args() {
|
|
||||||
err := Scrape(uri, dir, template)
|
|
||||||
if err != nil {
|
|
||||||
return Die(err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return 0
|
|
||||||
}
|
|
||||||
|
|
||||||
// backup all ads of the given user (via config or cmdline)
|
|
||||||
if uid == 0 && conf.User > 0 {
|
|
||||||
uid = conf.User
|
|
||||||
}
|
|
||||||
|
|
||||||
if uid > 0 {
|
|
||||||
err := Start(fmt.Sprintf("%d", uid), dir, template)
|
|
||||||
if err != nil {
|
|
||||||
return Die(err)
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
return Die(errors.New("invalid or no user id specified"))
|
|
||||||
}
|
|
||||||
|
|
||||||
return 0
|
|
||||||
}
|
|
||||||
|
|
||||||
func Die(err error) int {
|
|
||||||
slog.Error("Failure", "error", err.Error())
|
|
||||||
return 1
|
|
||||||
}
|
|
||||||
65
mkrel.sh
@@ -1,65 +0,0 @@
|
|||||||
#!/bin/bash
|
|
||||||
|
|
||||||
# Copyright © 2023 Thomas von Dein
|
|
||||||
|
|
||||||
# This program is free software: you can redistribute it and/or modify
|
|
||||||
# it under the terms of the GNU General Public License as published by
|
|
||||||
# the Free Software Foundation, either version 3 of the License, or
|
|
||||||
# (at your option) any later version.
|
|
||||||
|
|
||||||
# This program is distributed in the hope that it will be useful,
|
|
||||||
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
||||||
# GNU General Public License for more details.
|
|
||||||
|
|
||||||
# You should have received a copy of the GNU General Public License
|
|
||||||
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
|
|
||||||
|
|
||||||
# get list with: go tool dist list
|
|
||||||
DIST="darwin/amd64
|
|
||||||
freebsd/amd64
|
|
||||||
linux/amd64
|
|
||||||
netbsd/amd64
|
|
||||||
openbsd/amd64
|
|
||||||
windows/amd64"
|
|
||||||
|
|
||||||
tool="$1"
|
|
||||||
version="$2"
|
|
||||||
|
|
||||||
if test -z "$version"; then
|
|
||||||
echo "Usage: $0 <tool name> <release version>"
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
rm -rf releases
|
|
||||||
mkdir -p releases
|
|
||||||
|
|
||||||
|
|
||||||
for D in $DIST; do
|
|
||||||
os=${D/\/*/}
|
|
||||||
arch=${D/*\//}
|
|
||||||
binfile="releases/${tool}-${os}-${arch}-${version}"
|
|
||||||
tardir="${tool}-${os}-${arch}-${version}"
|
|
||||||
tarfile="releases/${tool}-${os}-${arch}-${version}.tar.gz"
|
|
||||||
set -x
|
|
||||||
GOOS=${os} GOARCH=${arch} go build -tags osusergo,netgo -ldflags "-extldflags=-static" -o ${binfile}
|
|
||||||
mkdir -p ${tardir}
|
|
||||||
cp ${binfile} README.md LICENSE ${tardir}/
|
|
||||||
echo 'tool = kleingebaeck
|
|
||||||
PREFIX = /usr/local
|
|
||||||
UID = root
|
|
||||||
GID = 0
|
|
||||||
|
|
||||||
install:
|
|
||||||
install -d -o $(UID) -g $(GID) $(PREFIX)/bin
|
|
||||||
install -d -o $(UID) -g $(GID) $(PREFIX)/man/man1
|
|
||||||
install -o $(UID) -g $(GID) -m 555 $(tool) $(PREFIX)/sbin/
|
|
||||||
install -o $(UID) -g $(GID) -m 444 $(tool).1 $(PREFIX)/man/man1/' > ${tardir}/Makefile
|
|
||||||
tar cpzf ${tarfile} ${tardir}
|
|
||||||
sha256sum ${binfile} | cut -d' ' -f1 > ${binfile}.sha256
|
|
||||||
sha256sum ${tarfile} | cut -d' ' -f1 > ${tarfile}.sha256
|
|
||||||
rm -rf ${tardir}
|
|
||||||
set +x
|
|
||||||
done
|
|
||||||
|
|
||||||
222
scrape.go
@@ -1,222 +0,0 @@
|
|||||||
/*
|
|
||||||
Copyright © 2023 Thomas von Dein
|
|
||||||
|
|
||||||
This program is free software: you can redistribute it and/or modify
|
|
||||||
it under the terms of the GNU General Public License as published by
|
|
||||||
the Free Software Foundation, either version 3 of the License, or
|
|
||||||
(at your option) any later version.
|
|
||||||
|
|
||||||
This program is distributed in the hope that it will be useful,
|
|
||||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
||||||
GNU General Public License for more details.
|
|
||||||
|
|
||||||
You should have received a copy of the GNU General Public License
|
|
||||||
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
*/
|
|
||||||
|
|
||||||
package main
|
|
||||||
|
|
||||||
import (
|
|
||||||
"errors"
|
|
||||||
"fmt"
|
|
||||||
"io"
|
|
||||||
"log/slog"
|
|
||||||
"net/http"
|
|
||||||
"strings"
|
|
||||||
"sync"
|
|
||||||
|
|
||||||
"astuart.co/goq"
|
|
||||||
)
|
|
||||||
|
|
||||||
type Index struct {
|
|
||||||
Links []string `goquery:".text-module-begin a,[href]"`
|
|
||||||
}
|
|
||||||
|
|
||||||
type Ad struct {
|
|
||||||
Title string `goquery:"h1"`
|
|
||||||
Slug string
|
|
||||||
Id string
|
|
||||||
Condition string
|
|
||||||
Category string
|
|
||||||
Price string `goquery:"h2#viewad-price"`
|
|
||||||
Created string `goquery:"#viewad-extra-info,text"`
|
|
||||||
Text string `goquery:"p#viewad-description-text,html"`
|
|
||||||
Images []string `goquery:".galleryimage-element img,[src]"`
|
|
||||||
Meta []string `goquery:".addetailslist--detail--value,text"`
|
|
||||||
}
|
|
||||||
|
|
||||||
func (ad *Ad) LogValue() slog.Value {
|
|
||||||
return slog.GroupValue(
|
|
||||||
slog.String("title", ad.Title),
|
|
||||||
slog.String("price", ad.Price),
|
|
||||||
slog.String("id", ad.Id),
|
|
||||||
slog.Int("imagecount", len(ad.Images)),
|
|
||||||
slog.Int("bodysize", len(ad.Text)),
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
// fetch some web page content
|
|
||||||
func Get(uri string, client *http.Client) (io.ReadCloser, error) {
|
|
||||||
req, err := http.NewRequest("GET", uri, nil)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
req.Header.Set("User-Agent", Useragent)
|
|
||||||
|
|
||||||
res, err := client.Do(req)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
slog.Debug("response", "code", res.StatusCode, "status",
|
|
||||||
res.Status, "size", res.ContentLength)
|
|
||||||
|
|
||||||
return res.Body, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// extract links from all ad listing pages (that is: use pagination)
|
|
||||||
// and scrape every page
|
|
||||||
func Start(uid string, dir string, template string) error {
|
|
||||||
client := &http.Client{}
|
|
||||||
adlinks := []string{}
|
|
||||||
|
|
||||||
baseuri := Baseuri + Listuri + "?userId=" + uid
|
|
||||||
page := 1
|
|
||||||
uri := baseuri
|
|
||||||
|
|
||||||
slog.Info("fetching ad pages", "user", uid)
|
|
||||||
|
|
||||||
for {
|
|
||||||
var index Index
|
|
||||||
slog.Debug("fetching page", "uri", uri)
|
|
||||||
body, err := Get(uri, client)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
defer body.Close()
|
|
||||||
|
|
||||||
err = goq.NewDecoder(body).Decode(&index)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(index.Links) == 0 {
|
|
||||||
break
|
|
||||||
}
|
|
||||||
|
|
||||||
slog.Debug("extracted ad links", "count", len(index.Links))
|
|
||||||
|
|
||||||
for _, href := range index.Links {
|
|
||||||
adlinks = append(adlinks, href)
|
|
||||||
slog.Debug("ad link", "href", href)
|
|
||||||
}
|
|
||||||
|
|
||||||
page++
|
|
||||||
uri = baseuri + "&pageNum=" + fmt.Sprintf("%d", page)
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, adlink := range adlinks {
|
|
||||||
err := Scrape(Baseuri+adlink, dir, template)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// scrape an ad. uri is the full uri of the ad, dir is the basedir
|
|
||||||
func Scrape(uri string, dir string, template string) error {
|
|
||||||
client := &http.Client{}
|
|
||||||
ad := &Ad{}
|
|
||||||
|
|
||||||
// extract slug and id from uri
|
|
||||||
uriparts := strings.Split(uri, "/")
|
|
||||||
if len(uriparts) < 6 {
|
|
||||||
return errors.New("invalid uri")
|
|
||||||
}
|
|
||||||
ad.Slug = uriparts[4]
|
|
||||||
ad.Id = uriparts[5]
|
|
||||||
|
|
||||||
// get the ad
|
|
||||||
slog.Debug("fetching ad page", "uri", uri)
|
|
||||||
body, err := Get(uri, client)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
defer body.Close()
|
|
||||||
|
|
||||||
// extract ad contents with goquery/goq
|
|
||||||
err = goq.NewDecoder(body).Decode(&ad)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if len(ad.Meta) == 2 {
|
|
||||||
ad.Category = ad.Meta[0]
|
|
||||||
ad.Condition = ad.Meta[1]
|
|
||||||
}
|
|
||||||
slog.Debug("extracted ad listing", "ad", ad)
|
|
||||||
|
|
||||||
// write listing
|
|
||||||
err = WriteAd(dir, ad, template)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
return ScrapeImages(dir, ad)
|
|
||||||
}
|
|
||||||
|
|
||||||
func ScrapeImages(dir string, ad *Ad) error {
|
|
||||||
// fetch images
|
|
||||||
img := 1
|
|
||||||
var wg sync.WaitGroup
|
|
||||||
wg.Add(len(ad.Images))
|
|
||||||
failure := make(chan string)
|
|
||||||
|
|
||||||
for _, imguri := range ad.Images {
|
|
||||||
file := fmt.Sprintf("%s/%d.jpg", dir, img)
|
|
||||||
go func() {
|
|
||||||
defer wg.Done()
|
|
||||||
err := Getimage(imguri, file)
|
|
||||||
if err != nil {
|
|
||||||
failure <- err.Error()
|
|
||||||
return
|
|
||||||
}
|
|
||||||
slog.Info("wrote ad image", "image", file)
|
|
||||||
}()
|
|
||||||
img++
|
|
||||||
}
|
|
||||||
|
|
||||||
close(failure)
|
|
||||||
wg.Wait()
|
|
||||||
goterr := <-failure
|
|
||||||
|
|
||||||
if goterr != "" {
|
|
||||||
return errors.New(goterr)
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// fetch an image
|
|
||||||
func Getimage(uri, fileName string) error {
|
|
||||||
slog.Debug("fetching ad image", "uri", uri)
|
|
||||||
response, err := http.Get(uri)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
defer response.Body.Close()
|
|
||||||
|
|
||||||
if response.StatusCode != 200 {
|
|
||||||
return errors.New("received non 200 response code")
|
|
||||||
}
|
|
||||||
|
|
||||||
err = WriteImage(fileName, response.Body)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
72
store.go
@@ -1,72 +0,0 @@
|
|||||||
/*
|
|
||||||
Copyright © 2023 Thomas von Dein
|
|
||||||
|
|
||||||
This program is free software: you can redistribute it and/or modify
|
|
||||||
it under the terms of the GNU General Public License as published by
|
|
||||||
the Free Software Foundation, either version 3 of the License, or
|
|
||||||
(at your option) any later version.
|
|
||||||
|
|
||||||
This program is distributed in the hope that it will be useful,
|
|
||||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
||||||
GNU General Public License for more details.
|
|
||||||
|
|
||||||
You should have received a copy of the GNU General Public License
|
|
||||||
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
*/
|
|
||||||
|
|
||||||
package main
|
|
||||||
|
|
||||||
import (
|
|
||||||
"io"
|
|
||||||
"log/slog"
|
|
||||||
"os"
|
|
||||||
"strings"
|
|
||||||
tpl "text/template"
|
|
||||||
)
|
|
||||||
|
|
||||||
func WriteAd(dir string, ad *Ad, template string) error {
|
|
||||||
// prepare output dir
|
|
||||||
dir = dir + "/" + ad.Slug
|
|
||||||
err := Mkdir(dir)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
// write ad file
|
|
||||||
listingfile := strings.Join([]string{dir, "Adlisting.txt"}, "/")
|
|
||||||
f, err := os.Create(listingfile)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
ad.Text = strings.ReplaceAll(ad.Text, "<br/>", "\n")
|
|
||||||
|
|
||||||
tmpl, err := tpl.New("adlisting").Parse(template)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
err = tmpl.Execute(f, ad)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
slog.Info("wrote ad listing", "listingfile", listingfile)
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func WriteImage(filename string, reader io.ReadCloser) error {
|
|
||||||
file, err := os.Create(filename)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
defer file.Close()
|
|
||||||
|
|
||||||
_, err = io.Copy(file, reader)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
55
util.go
@@ -1,55 +0,0 @@
|
|||||||
/*
|
|
||||||
Copyright © 2023 Thomas von Dein
|
|
||||||
|
|
||||||
This program is free software: you can redistribute it and/or modify
|
|
||||||
it under the terms of the GNU General Public License as published by
|
|
||||||
the Free Software Foundation, either version 3 of the License, or
|
|
||||||
(at your option) any later version.
|
|
||||||
|
|
||||||
This program is distributed in the hope that it will be useful,
|
|
||||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
||||||
GNU General Public License for more details.
|
|
||||||
|
|
||||||
You should have received a copy of the GNU General Public License
|
|
||||||
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
*/
|
|
||||||
|
|
||||||
package main
|
|
||||||
|
|
||||||
import (
|
|
||||||
"bytes"
|
|
||||||
"errors"
|
|
||||||
"os"
|
|
||||||
"os/exec"
|
|
||||||
)
|
|
||||||
|
|
||||||
func Mkdir(dir string) error {
|
|
||||||
if _, err := os.Stat(dir); errors.Is(err, os.ErrNotExist) {
|
|
||||||
err := os.Mkdir(dir, os.ModePerm)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func man() error {
|
|
||||||
man := exec.Command("less", "-")
|
|
||||||
|
|
||||||
var b bytes.Buffer
|
|
||||||
b.Write([]byte(manpage))
|
|
||||||
|
|
||||||
man.Stdout = os.Stdout
|
|
||||||
man.Stdin = &b
|
|
||||||
man.Stderr = os.Stderr
|
|
||||||
|
|
||||||
err := man.Run()
|
|
||||||
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||