Compare commits
192 commits
ecb1a161e2
...
c65568437f
Author | SHA1 | Date | |
---|---|---|---|
c65568437f | |||
|
ad661c4c91 | ||
|
ba8c4623ed | ||
|
ba43c87952 | ||
|
595b87946d | ||
|
99d4e1a43d | ||
|
477de4e2df | ||
|
246470da18 | ||
|
df9f7eb778 | ||
|
375831f516 | ||
|
e518936be7 | ||
|
583dfb4958 | ||
|
2de45b163e | ||
|
48b0164676 | ||
|
4b3c3c58d2 | ||
|
60768b4885 | ||
|
02dd778124 | ||
|
5b63121e92 | ||
|
49019a843f | ||
|
d65714fa47 | ||
|
8161829ad5 | ||
|
7f35fc9f6b | ||
|
3bc8c9468a | ||
|
1df3598a74 | ||
|
5f64fe2516 | ||
|
50eee7e7b3 | ||
|
c0df9815c7 | ||
|
46d5895d1d | ||
|
7c16aaf303 | ||
|
cdc1d9c9ba | ||
|
6bc83310b9 | ||
|
c8d5c85c76 | ||
|
d1e4bd7285 | ||
|
1022b5fdf9 | ||
|
e8536ac1b2 | ||
|
a0afe36d56 | ||
|
0b80f9d61c | ||
|
424075981f | ||
|
c334df91ec | ||
|
f2346fb33e | ||
|
8a21fd1476 | ||
|
2ac44172ac | ||
|
4c78721f03 | ||
|
04be85996d | ||
|
59be6bded2 | ||
|
46873e14fe | ||
|
0f01cc97a4 | ||
|
a70e00a76d | ||
|
f0260c62c3 | ||
|
fc5a1526ca | ||
|
4c0e234479 | ||
|
0eab63d728 | ||
|
b0884e9158 | ||
|
b4581418d4 | ||
|
af1566f40d | ||
|
529e0d0cca | ||
|
a3532804ac | ||
|
8c19146d29 | ||
|
2a3d5865ad | ||
|
4d36c9dc30 | ||
|
a2e47a88c3 | ||
|
b09f50853f | ||
|
9b5bf565b3 | ||
|
5cc956367f | ||
|
548e28249b | ||
|
684c69b0cd | ||
|
3dae4e0801 | ||
|
4622d9be1e | ||
|
76183dcd44 | ||
|
50b234d893 | ||
|
af48f36fd2 | ||
|
7f6ca23e8f | ||
|
1daef22a3d | ||
|
c694810d9a | ||
|
f12f6a2dba | ||
|
b1be45df6c | ||
|
b4f393a5cc | ||
|
29126ebe29 | ||
|
50c971d545 | ||
|
7aba7992aa | ||
|
48ebed7b38 | ||
|
ccef6b95ad | ||
|
dd5da99a30 | ||
|
2ff27b92ff | ||
|
b47189921f | ||
|
f1d3e8c9c9 | ||
|
53fbd2a5a0 | ||
|
3254a4d7bc | ||
|
52d2d21da5 | ||
|
abb74f056c | ||
|
25548b6757 | ||
|
cfe433e9e2 | ||
|
0dfc4ea2c5 | ||
|
38960df180 | ||
|
b440a6fdc6 | ||
|
48d0385653 | ||
|
b68c0e0df8 | ||
|
f27b267614 | ||
|
8bff63d9c6 | ||
|
2b4a030158 | ||
|
6a99904e64 | ||
|
f3c687604f | ||
|
a86a94555d | ||
|
acc0787b00 | ||
|
c8992650a1 | ||
|
f9f511a849 | ||
|
990719d614 | ||
|
b6be18d585 | ||
|
cf525c964a | ||
|
52a4f0860c | ||
|
21b27a1042 | ||
|
2eee535171 | ||
|
da51fc065f | ||
|
e032705c9a | ||
|
be27bc9250 | ||
|
75edc1b2b7 | ||
|
c9ea53806d | ||
|
2bb9480555 | ||
|
eb942bc498 | ||
|
5a0ea423c4 | ||
|
2120cc42fb | ||
|
5067501661 | ||
|
aea8484ccc | ||
|
6b9394dc78 | ||
|
4b51d42b8c | ||
|
d3fbf0d872 | ||
|
41a8eb74a1 | ||
|
7e6c58b67a | ||
|
a31e518a07 | ||
|
50162f52b6 | ||
|
c0edf6e424 | ||
|
2ea8d73ac1 | ||
|
465cd8c768 | ||
|
73f4bc078e | ||
|
1add201d3b | ||
|
09113c2594 | ||
|
c39e642877 | ||
|
e2460ead18 | ||
|
60c1339612 | ||
|
d324aa5da1 | ||
|
6f24987601 | ||
|
54fb29d443 | ||
|
ebe463dd08 | ||
|
987f42d6d4 | ||
|
fa8253c8bf | ||
|
e4444e6432 | ||
|
3769850ba3 | ||
|
89e3da0b6f | ||
|
99d4571c6b | ||
|
69acc6228a | ||
|
5e2f0fb626 | ||
|
372461b1a3 | ||
|
1591e18027 | ||
|
e2bca5bb05 | ||
|
7926ffad73 | ||
|
7ff97c0c7b | ||
|
1989252608 | ||
|
91e73b00b5 | ||
|
5c6c79baf4 | ||
|
99d1343045 | ||
|
14e6dbb645 | ||
|
fc8421ed50 | ||
|
2460b67886 | ||
|
705b9daa0b | ||
|
1ada9c26f8 | ||
|
55e1703741 | ||
|
849eaeb50e | ||
|
aeca4cfd60 | ||
|
686f21bc50 | ||
|
8dd8be9694 | ||
|
dfa9c651cd | ||
|
6d6d6037a3 | ||
|
2559dbbf49 | ||
|
de53120843 | ||
|
b1b7e4edce | ||
|
b27487ace0 | ||
|
d005acca83 | ||
|
93de8c239b | ||
|
75b0213684 | ||
|
f76a23f0a5 | ||
|
e4e04a7865 | ||
|
da339fd5cc | ||
|
ba116d9ab6 | ||
|
ea08445946 | ||
|
ade09b2aad | ||
|
28d46b6721 | ||
|
1efb7c7bce | ||
|
d34411137f | ||
|
70542686bb | ||
|
edf10be93a | ||
|
a725fdd315 | ||
|
84ba0c4a9e |
168 changed files with 21018 additions and 1877 deletions
26
.gitattributes
vendored
26
.gitattributes
vendored
|
@ -22,18 +22,24 @@
|
||||||
*.RTF diff=astextplain
|
*.RTF diff=astextplain
|
||||||
|
|
||||||
# Ignore files in git archive (i.e. GitHub release builds)
|
# Ignore files in git archive (i.e. GitHub release builds)
|
||||||
|
|
||||||
## Docker
|
## Docker
|
||||||
Dockerfile export-ignore
|
Dockerfile export-ignore
|
||||||
.dockerignore export-ignore
|
.dockerignore export-ignore
|
||||||
|
|
||||||
## Travis
|
## Travis
|
||||||
.travis.yml export-ignore
|
.travis.yml export-ignore
|
||||||
|
|
||||||
## GitHub
|
## GitHub
|
||||||
.github/ export-ignore
|
.github/ export-ignore
|
||||||
|
|
||||||
## Git
|
## Git
|
||||||
.gitattributes export-ignore
|
.gitattributes export-ignore
|
||||||
.gitignore export-ignore
|
.gitignore export-ignore
|
||||||
|
|
||||||
## Scalingo
|
## Scalingo
|
||||||
scalingo.json export-ignore
|
scalingo.json export-ignore
|
||||||
|
|
||||||
## RSS-Bridge
|
## RSS-Bridge
|
||||||
phpunit.xml export-ignore
|
phpunit.xml export-ignore
|
||||||
phpcs.xml export-ignore
|
phpcs.xml export-ignore
|
||||||
|
@ -42,8 +48,22 @@ tests/ export-ignore
|
||||||
cache/.gitkeep export-ignore
|
cache/.gitkeep export-ignore
|
||||||
bridges/DemoBridge.php export-ignore
|
bridges/DemoBridge.php export-ignore
|
||||||
bridges/FeedExpanderExampleBridge.php export-ignore
|
bridges/FeedExpanderExampleBridge.php export-ignore
|
||||||
|
|
||||||
## Composer
|
## Composer
|
||||||
composer.json export-ignore
|
#
|
||||||
composer.lock export-ignore
|
# Keep the following lines commented out. Heroku does
|
||||||
|
# not function if the composer files are ignored during
|
||||||
|
# export. For more information see
|
||||||
|
# https://github.com/rss-bridge/rss-bridge/issues/1165
|
||||||
|
#
|
||||||
|
# composer.json export-ignore
|
||||||
|
# composer.lock export-ignore
|
||||||
|
|
||||||
## Heroku
|
## Heroku
|
||||||
app.json export-ignore
|
#
|
||||||
|
# Keep the following line commented out. Heroku does
|
||||||
|
# not function if app.json is ignored during export.
|
||||||
|
# For more information see
|
||||||
|
# https://github.com/rss-bridge/rss-bridge/issues/1165
|
||||||
|
#
|
||||||
|
# app.json export-ignore
|
||||||
|
|
|
@ -6,6 +6,8 @@ RUN mv "$PHP_INI_DIR/php.ini-production" "$PHP_INI_DIR/php.ini" \
|
||||||
&& apt-get --yes update && apt-get --yes install libxml2-dev \
|
&& apt-get --yes update && apt-get --yes install libxml2-dev \
|
||||||
&& docker-php-ext-install -j$(nproc) simplexml \
|
&& docker-php-ext-install -j$(nproc) simplexml \
|
||||||
&& sed -ri -e 's!/var/www/html!${APACHE_DOCUMENT_ROOT}!g' /etc/apache2/sites-available/*.conf \
|
&& sed -ri -e 's!/var/www/html!${APACHE_DOCUMENT_ROOT}!g' /etc/apache2/sites-available/*.conf \
|
||||||
&& sed -ri -e 's!/var/www/!${APACHE_DOCUMENT_ROOT}!g' /etc/apache2/apache2.conf /etc/apache2/conf-available/*.conf
|
&& sed -ri -e 's!/var/www/!${APACHE_DOCUMENT_ROOT}!g' /etc/apache2/apache2.conf /etc/apache2/conf-available/*.conf \
|
||||||
|
&& sed -ri -e 's/(MinProtocol\s*=\s*)TLSv1\.2/\1None/' /etc/ssl/openssl.cnf \
|
||||||
|
&& sed -ri -e 's/(CipherString\s*=\s*DEFAULT)@SECLEVEL=2/\1/' /etc/ssl/openssl.cnf
|
||||||
|
|
||||||
COPY --chown=www-data:www-data ./ /app/
|
COPY --chown=www-data:www-data ./ /app/
|
27
README.md
27
README.md
|
@ -1,8 +1,8 @@
|
||||||
rss-bridge
|
![RSS-Bridge](static/logo_600px.png)
|
||||||
===
|
===
|
||||||
[![LICENSE](https://img.shields.io/badge/license-UNLICENSE-blue.svg)](UNLICENSE) [![GitHub release](https://img.shields.io/github/release/rss-bridge/rss-bridge.svg)](https://github.com/rss-bridge/rss-bridge/releases/latest) [![Debian Release](https://img.shields.io/badge/dynamic/json.svg?label=debian%20release&url=https%3A%2F%2Fsources.debian.org%2Fapi%2Fsrc%2Frss-bridge%2F&query=%24.versions%5B0%5D.version&colorB=blue)](https://tracker.debian.org/pkg/rss-bridge) [![Guix Release](https://img.shields.io/badge/guix%20release-unknown-light--gray.svg)](https://www.gnu.org/software/guix/packages/R/) [![Build Status](https://travis-ci.org/RSS-Bridge/rss-bridge.svg?branch=master)](https://travis-ci.org/RSS-Bridge/rss-bridge) [![Docker Build Status](https://img.shields.io/docker/build/rssbridge/rss-bridge.svg)](https://hub.docker.com/r/rssbridge/rss-bridge/)
|
[![LICENSE](https://img.shields.io/badge/license-UNLICENSE-blue.svg)](UNLICENSE) [![GitHub release](https://img.shields.io/github/release/rss-bridge/rss-bridge.svg?logo=github)](https://github.com/rss-bridge/rss-bridge/releases/latest) [![Debian Release](https://img.shields.io/badge/dynamic/json.svg?logo=debian&label=debian%20release&url=https%3A%2F%2Fsources.debian.org%2Fapi%2Fsrc%2Frss-bridge%2F&query=%24.versions%5B0%5D.version&colorB=blue)](https://tracker.debian.org/pkg/rss-bridge) [![Guix Release](https://img.shields.io/badge/guix%20release-unknown-blue.svg)](https://www.gnu.org/software/guix/packages/R/) [![Build Status](https://travis-ci.org/RSS-Bridge/rss-bridge.svg?branch=master)](https://travis-ci.org/RSS-Bridge/rss-bridge) [![Docker Build Status](https://img.shields.io/docker/build/rssbridge/rss-bridge.svg?logo=docker)](https://hub.docker.com/r/rssbridge/rss-bridge/)
|
||||||
|
|
||||||
RSS-Bridge is a PHP project capable of generating RSS and Atom feeds for websites which don't have one. It can be used on webservers or as stand alone application in CLI mode.
|
RSS-Bridge is a PHP project capable of generating RSS and Atom feeds for websites that don't have one. It can be used on webservers or as a stand-alone application in CLI mode.
|
||||||
|
|
||||||
**Important**: RSS-Bridge is __not__ a feed reader or feed aggregator, but a tool to generate feeds that are consumed by feed readers and feed aggregators. Find a list of feed aggregators on [Wikipedia](https://en.wikipedia.org/wiki/Comparison_of_feed_aggregators).
|
**Important**: RSS-Bridge is __not__ a feed reader or feed aggregator, but a tool to generate feeds that are consumed by feed readers and feed aggregators. Find a list of feed aggregators on [Wikipedia](https://en.wikipedia.org/wiki/Comparison_of_feed_aggregators).
|
||||||
|
|
||||||
|
@ -15,7 +15,6 @@ Supported sites/pages (examples)
|
||||||
* `DuckDuckGo`: Most recent results from [DuckDuckGo.com](https://duckduckgo.com/)
|
* `DuckDuckGo`: Most recent results from [DuckDuckGo.com](https://duckduckgo.com/)
|
||||||
* `Facebook` : Returns the latest posts on a page or profile on [Facebook](https://facebook.com/)
|
* `Facebook` : Returns the latest posts on a page or profile on [Facebook](https://facebook.com/)
|
||||||
* `FlickrExplore` : [Latest interesting images](http://www.flickr.com/explore) from Flickr
|
* `FlickrExplore` : [Latest interesting images](http://www.flickr.com/explore) from Flickr
|
||||||
* `GooglePlus` : Most recent posts of user timeline
|
|
||||||
* `GoogleSearch` : Most recent results from Google Search
|
* `GoogleSearch` : Most recent results from Google Search
|
||||||
* `Identi.ca` : Identica user timeline (Should be compatible with other Pump.io instances)
|
* `Identi.ca` : Identica user timeline (Should be compatible with other Pump.io instances)
|
||||||
* `Instagram`: Most recent photos from an Instagram user
|
* `Instagram`: Most recent photos from an Instagram user
|
||||||
|
@ -77,7 +76,7 @@ RSS-Bridge allows you to take full control over which bridges are displayed to t
|
||||||
|
|
||||||
Find more information on the [Wiki](https://github.com/RSS-Bridge/rss-bridge/wiki/Whitelisting)
|
Find more information on the [Wiki](https://github.com/RSS-Bridge/rss-bridge/wiki/Whitelisting)
|
||||||
|
|
||||||
**Notice**: By default RSS-Bridge will only show a small subset of bridges. Make sure to read up on [whitelisting](https://github.com/RSS-Bridge/rss-bridge/wiki/Whitelisting) to unlock the full potential of RSS-Bridge!
|
**Notice**: By default, RSS-Bridge will only show a small subset of bridges. Make sure to read up on [whitelisting](https://github.com/RSS-Bridge/rss-bridge/wiki/Whitelisting) to unlock the full potential of RSS-Bridge!
|
||||||
|
|
||||||
Deploy
|
Deploy
|
||||||
===
|
===
|
||||||
|
@ -119,6 +118,7 @@ https://gist.github.com/LogMANOriginal/da00cd1e5f0ca31cef8e193509b17fd8
|
||||||
* [alex73](https://github.com/alex73)
|
* [alex73](https://github.com/alex73)
|
||||||
* [alexAubin](https://github.com/alexAubin)
|
* [alexAubin](https://github.com/alexAubin)
|
||||||
* [AmauryCarrade](https://github.com/AmauryCarrade)
|
* [AmauryCarrade](https://github.com/AmauryCarrade)
|
||||||
|
* [AntoineTurmel](https://github.com/AntoineTurmel)
|
||||||
* [ArthurHoaro](https://github.com/ArthurHoaro)
|
* [ArthurHoaro](https://github.com/ArthurHoaro)
|
||||||
* [Astalaseven](https://github.com/Astalaseven)
|
* [Astalaseven](https://github.com/Astalaseven)
|
||||||
* [Astyan-42](https://github.com/Astyan-42)
|
* [Astyan-42](https://github.com/Astyan-42)
|
||||||
|
@ -132,31 +132,39 @@ https://gist.github.com/LogMANOriginal/da00cd1e5f0ca31cef8e193509b17fd8
|
||||||
* [cnlpete](https://github.com/cnlpete)
|
* [cnlpete](https://github.com/cnlpete)
|
||||||
* [corenting](https://github.com/corenting)
|
* [corenting](https://github.com/corenting)
|
||||||
* [couraudt](https://github.com/couraudt)
|
* [couraudt](https://github.com/couraudt)
|
||||||
|
* [cyberjacob](https://github.com/cyberjacob)
|
||||||
* [da2x](https://github.com/da2x)
|
* [da2x](https://github.com/da2x)
|
||||||
* [Daiyousei](https://github.com/Daiyousei)
|
* [Daiyousei](https://github.com/Daiyousei)
|
||||||
|
* [dawidsowa](https://github.com/dawidsowa)
|
||||||
* [disk0x](https://github.com/disk0x)
|
* [disk0x](https://github.com/disk0x)
|
||||||
* [DJCrashdummy](https://github.com/DJCrashdummy)
|
* [DJCrashdummy](https://github.com/DJCrashdummy)
|
||||||
* [Djuuu](https://github.com/Djuuu)
|
* [Djuuu](https://github.com/Djuuu)
|
||||||
* [DnAp](https://github.com/DnAp)
|
* [DnAp](https://github.com/DnAp)
|
||||||
|
* [dominik-th](https://github.com/dominik-th)
|
||||||
* [Draeli](https://github.com/Draeli)
|
* [Draeli](https://github.com/Draeli)
|
||||||
* [Dreckiger-Dan](https://github.com/Dreckiger-Dan)
|
* [Dreckiger-Dan](https://github.com/Dreckiger-Dan)
|
||||||
* [em92](https://github.com/em92)
|
* [em92](https://github.com/em92)
|
||||||
* [eMerzh](https://github.com/eMerzh)
|
* [eMerzh](https://github.com/eMerzh)
|
||||||
* [EtienneM](https://github.com/EtienneM)
|
* [EtienneM](https://github.com/EtienneM)
|
||||||
|
* [floviolleau](https://github.com/floviolleau)
|
||||||
* [fluffy-critter](https://github.com/fluffy-critter)
|
* [fluffy-critter](https://github.com/fluffy-critter)
|
||||||
* [Frenzie](https://github.com/Frenzie)
|
* [Frenzie](https://github.com/Frenzie)
|
||||||
* [fulmeek](https://github.com/fulmeek)
|
* [fulmeek](https://github.com/fulmeek)
|
||||||
* [Ginko-Aloe](https://github.com/Ginko-Aloe)
|
* [Ginko-Aloe](https://github.com/Ginko-Aloe)
|
||||||
* [Glandos](https://github.com/Glandos)
|
* [Glandos](https://github.com/Glandos)
|
||||||
|
* [gloony](https://github.com/gloony)
|
||||||
* [GregThib](https://github.com/GregThib)
|
* [GregThib](https://github.com/GregThib)
|
||||||
* [griffaurel](https://github.com/griffaurel)
|
* [griffaurel](https://github.com/griffaurel)
|
||||||
* [Grummfy](https://github.com/Grummfy)
|
* [Grummfy](https://github.com/Grummfy)
|
||||||
* [hunhejj](https://github.com/hunhejj)
|
* [hunhejj](https://github.com/hunhejj)
|
||||||
|
* [husim0](https://github.com/husim0)
|
||||||
|
* [IceWreck](https://github.com/IceWreck)
|
||||||
* [j0k3r](https://github.com/j0k3r)
|
* [j0k3r](https://github.com/j0k3r)
|
||||||
* [JackNUMBER](https://github.com/JackNUMBER)
|
* [JackNUMBER](https://github.com/JackNUMBER)
|
||||||
* [jdigilio](https://github.com/jdigilio)
|
* [jdigilio](https://github.com/jdigilio)
|
||||||
* [JeremyRand](https://github.com/JeremyRand)
|
* [JeremyRand](https://github.com/JeremyRand)
|
||||||
* [Jocker666z](https://github.com/Jocker666z)
|
* [Jocker666z](https://github.com/Jocker666z)
|
||||||
|
* [johnnygroovy](https://github.com/johnnygroovy)
|
||||||
* [killruana](https://github.com/killruana)
|
* [killruana](https://github.com/killruana)
|
||||||
* [klimplant](https://github.com/klimplant)
|
* [klimplant](https://github.com/klimplant)
|
||||||
* [kranack](https://github.com/kranack)
|
* [kranack](https://github.com/kranack)
|
||||||
|
@ -166,9 +174,11 @@ https://gist.github.com/LogMANOriginal/da00cd1e5f0ca31cef8e193509b17fd8
|
||||||
* [lagaisse](https://github.com/lagaisse)
|
* [lagaisse](https://github.com/lagaisse)
|
||||||
* [lalannev](https://github.com/lalannev)
|
* [lalannev](https://github.com/lalannev)
|
||||||
* [ldidry](https://github.com/ldidry)
|
* [ldidry](https://github.com/ldidry)
|
||||||
|
* [Leomaradan](https://github.com/Leomaradan)
|
||||||
* [Limero](https://github.com/Limero)
|
* [Limero](https://github.com/Limero)
|
||||||
* [LogMANOriginal](https://github.com/LogMANOriginal)
|
* [LogMANOriginal](https://github.com/LogMANOriginal)
|
||||||
* [lorenzos](https://github.com/lorenzos)
|
* [lorenzos](https://github.com/lorenzos)
|
||||||
|
* [lukasklinger](https://github.com/lukasklinger)
|
||||||
* [m0zes](https://github.com/m0zes)
|
* [m0zes](https://github.com/m0zes)
|
||||||
* [matthewseal](https://github.com/matthewseal)
|
* [matthewseal](https://github.com/matthewseal)
|
||||||
* [mcbyte-it](https://github.com/mcbyte-it)
|
* [mcbyte-it](https://github.com/mcbyte-it)
|
||||||
|
@ -184,6 +194,8 @@ https://gist.github.com/LogMANOriginal/da00cd1e5f0ca31cef8e193509b17fd8
|
||||||
* [niawag](https://github.com/niawag)
|
* [niawag](https://github.com/niawag)
|
||||||
* [Nono-m0le](https://github.com/Nono-m0le)
|
* [Nono-m0le](https://github.com/Nono-m0le)
|
||||||
* [ObsidianWitch](https://github.com/ObsidianWitch)
|
* [ObsidianWitch](https://github.com/ObsidianWitch)
|
||||||
|
* [OliverParoczai](https://github.com/OliverParoczai)
|
||||||
|
* [oratosquilla-oratoria](https://github.com/oratosquilla-oratoria)
|
||||||
* [ORelio](https://github.com/ORelio)
|
* [ORelio](https://github.com/ORelio)
|
||||||
* [PaulVayssiere](https://github.com/PaulVayssiere)
|
* [PaulVayssiere](https://github.com/PaulVayssiere)
|
||||||
* [pellaeon](https://github.com/pellaeon)
|
* [pellaeon](https://github.com/pellaeon)
|
||||||
|
@ -199,15 +211,20 @@ https://gist.github.com/LogMANOriginal/da00cd1e5f0ca31cef8e193509b17fd8
|
||||||
* [rogerdc](https://github.com/rogerdc)
|
* [rogerdc](https://github.com/rogerdc)
|
||||||
* [Roliga](https://github.com/Roliga)
|
* [Roliga](https://github.com/Roliga)
|
||||||
* [sebsauvage](https://github.com/sebsauvage)
|
* [sebsauvage](https://github.com/sebsauvage)
|
||||||
|
* [shutosg](https://github.com/shutosg)
|
||||||
* [somini](https://github.com/somini)
|
* [somini](https://github.com/somini)
|
||||||
* [squeek502](https://github.com/squeek502)
|
* [squeek502](https://github.com/squeek502)
|
||||||
|
* [stjohnjohnson](https://github.com/stjohnjohnson)
|
||||||
* [Strubbl](https://github.com/Strubbl)
|
* [Strubbl](https://github.com/Strubbl)
|
||||||
* [sublimz](https://github.com/sublimz)
|
* [sublimz](https://github.com/sublimz)
|
||||||
|
* [sunchaserinfo](https://github.com/sunchaserinfo)
|
||||||
* [sysadminstory](https://github.com/sysadminstory)
|
* [sysadminstory](https://github.com/sysadminstory)
|
||||||
* [tameroski](https://github.com/tameroski)
|
* [tameroski](https://github.com/tameroski)
|
||||||
* [teromene](https://github.com/teromene)
|
* [teromene](https://github.com/teromene)
|
||||||
* [thefranke](https://github.com/thefranke)
|
* [thefranke](https://github.com/thefranke)
|
||||||
|
* [ThePadawan](https://github.com/ThePadawan)
|
||||||
* [TheRadialActive](https://github.com/TheRadialActive)
|
* [TheRadialActive](https://github.com/TheRadialActive)
|
||||||
|
* [TitiTestScalingo](https://github.com/TitiTestScalingo)
|
||||||
* [triatic](https://github.com/triatic)
|
* [triatic](https://github.com/triatic)
|
||||||
* [VerifiedJoseph](https://github.com/VerifiedJoseph)
|
* [VerifiedJoseph](https://github.com/VerifiedJoseph)
|
||||||
* [WalterBarrett](https://github.com/WalterBarrett)
|
* [WalterBarrett](https://github.com/WalterBarrett)
|
||||||
|
|
136
actions/ConnectivityAction.php
Normal file
136
actions/ConnectivityAction.php
Normal file
|
@ -0,0 +1,136 @@
|
||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* This file is part of RSS-Bridge, a PHP project capable of generating RSS and
|
||||||
|
* Atom feeds for websites that don't have one.
|
||||||
|
*
|
||||||
|
* For the full license information, please view the UNLICENSE file distributed
|
||||||
|
* with this source code.
|
||||||
|
*
|
||||||
|
* @package Core
|
||||||
|
* @license http://unlicense.org/ UNLICENSE
|
||||||
|
* @link https://github.com/rss-bridge/rss-bridge
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Checks if the website for a given bridge is reachable.
|
||||||
|
*
|
||||||
|
* **Remarks**
|
||||||
|
* - This action is only available in debug mode.
|
||||||
|
* - Returns the bridge status as Json-formatted string.
|
||||||
|
* - Returns an error if the bridge is not whitelisted.
|
||||||
|
* - Returns a responsive web page that automatically checks all whitelisted
|
||||||
|
* bridges (using JavaScript) if no bridge is specified.
|
||||||
|
*/
|
||||||
|
class ConnectivityAction extends ActionAbstract {
|
||||||
|
public function execute() {
|
||||||
|
|
||||||
|
if(!Debug::isEnabled()) {
|
||||||
|
returnError('This action is only available in debug mode!');
|
||||||
|
}
|
||||||
|
|
||||||
|
if(!isset($this->userData['bridge'])) {
|
||||||
|
$this->returnEntryPage();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$bridgeName = $this->userData['bridge'];
|
||||||
|
|
||||||
|
$this->reportBridgeConnectivity($bridgeName);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Generates a report about the bridge connectivity status and sends it back
|
||||||
|
* to the user.
|
||||||
|
*
|
||||||
|
* The report is generated as Json-formatted string in the format
|
||||||
|
* {
|
||||||
|
* "bridge": "<bridge-name>",
|
||||||
|
* "successful": true/false
|
||||||
|
* }
|
||||||
|
*
|
||||||
|
* @param string $bridgeName Name of the bridge to generate the report for
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
private function reportBridgeConnectivity($bridgeName) {
|
||||||
|
|
||||||
|
$bridgeFac = new \BridgeFactory();
|
||||||
|
$bridgeFac->setWorkingDir(PATH_LIB_BRIDGES);
|
||||||
|
|
||||||
|
if(!$bridgeFac->isWhitelisted($bridgeName)) {
|
||||||
|
header('Content-Type: text/html');
|
||||||
|
returnServerError('Bridge is not whitelisted!');
|
||||||
|
}
|
||||||
|
|
||||||
|
header('Content-Type: text/json');
|
||||||
|
|
||||||
|
$retVal = array(
|
||||||
|
'bridge' => $bridgeName,
|
||||||
|
'successful' => false,
|
||||||
|
'http_code' => 200,
|
||||||
|
);
|
||||||
|
|
||||||
|
$bridge = $bridgeFac->create($bridgeName);
|
||||||
|
|
||||||
|
if($bridge === false) {
|
||||||
|
echo json_encode($retVal);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$curl_opts = array(
|
||||||
|
CURLOPT_CONNECTTIMEOUT => 5
|
||||||
|
);
|
||||||
|
|
||||||
|
try {
|
||||||
|
$reply = getContents($bridge::URI, array(), $curl_opts, true);
|
||||||
|
|
||||||
|
if($reply) {
|
||||||
|
$retVal['successful'] = true;
|
||||||
|
if (isset($reply['header'])) {
|
||||||
|
if (strpos($reply['header'], 'HTTP/1.1 301 Moved Permanently') !== false) {
|
||||||
|
$retVal['http_code'] = 301;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch(Exception $e) {
|
||||||
|
$retVal['successful'] = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
echo json_encode($retVal);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
private function returnEntryPage() {
|
||||||
|
echo <<<EOD
|
||||||
|
<!DOCTYPE html>
|
||||||
|
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<link rel="stylesheet" href="static/bootstrap.min.css">
|
||||||
|
<link
|
||||||
|
rel="stylesheet"
|
||||||
|
href="https://use.fontawesome.com/releases/v5.6.3/css/all.css"
|
||||||
|
integrity="sha384-UHRtZLI+pbxtHCWp1t77Bi1L4ZtiqrqD80Kn4Z8NTSRyMA2Fd33n5dQ8lWUE00s/"
|
||||||
|
crossorigin="anonymous">
|
||||||
|
<link rel="stylesheet" href="static/connectivity.css">
|
||||||
|
<script src="static/connectivity.js" type="text/javascript"></script>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<div id="main-content" class="container">
|
||||||
|
<div class="progress">
|
||||||
|
<div class="progress-bar" role="progressbar" aria-valuenow="75" aria-valuemin="0" aria-valuemax="100"></div>
|
||||||
|
</div>
|
||||||
|
<div id="status-message" class="sticky-top alert alert-primary alert-dismissible fade show" role="alert">
|
||||||
|
<i id="status-icon" class="fas fa-sync"></i>
|
||||||
|
<span>...</span>
|
||||||
|
<button type="button" class="close" data-dismiss="alert" aria-label="Close" onclick="stopConnectivityChecks()">
|
||||||
|
<span aria-hidden="true">×</span>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
<input type="text" class="form-control" id="search" onkeyup="search()" placeholder="Search for bridge..">
|
||||||
|
</div>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
|
EOD;
|
||||||
|
}
|
||||||
|
}
|
|
@ -19,13 +19,16 @@ class DetectAction extends ActionAbstract {
|
||||||
$format = $this->userData['format']
|
$format = $this->userData['format']
|
||||||
or returnClientError('You must specify a format!');
|
or returnClientError('You must specify a format!');
|
||||||
|
|
||||||
foreach(Bridge::getBridgeNames() as $bridgeName) {
|
$bridgeFac = new \BridgeFactory();
|
||||||
|
$bridgeFac->setWorkingDir(PATH_LIB_BRIDGES);
|
||||||
|
|
||||||
if(!Bridge::isWhitelisted($bridgeName)) {
|
foreach($bridgeFac->getBridgeNames() as $bridgeName) {
|
||||||
|
|
||||||
|
if(!$bridgeFac->isWhitelisted($bridgeName)) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
$bridge = Bridge::create($bridgeName);
|
$bridge = $bridgeFac->create($bridgeName);
|
||||||
|
|
||||||
if($bridge === false) {
|
if($bridge === false) {
|
||||||
continue;
|
continue;
|
||||||
|
|
|
@ -12,26 +12,32 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
class DisplayAction extends ActionAbstract {
|
class DisplayAction extends ActionAbstract {
|
||||||
|
private function get_return_code($error) {
|
||||||
|
$returnCode = $error->getCode();
|
||||||
|
if ($returnCode === 301 || $returnCode === 302) {
|
||||||
|
# Don't pass redirect codes to the exterior
|
||||||
|
$returnCode = 508;
|
||||||
|
}
|
||||||
|
return $returnCode;
|
||||||
|
}
|
||||||
|
|
||||||
public function execute() {
|
public function execute() {
|
||||||
$bridge = array_key_exists('bridge', $this->userData) ? $this->userData['bridge'] : null;
|
$bridge = array_key_exists('bridge', $this->userData) ? $this->userData['bridge'] : null;
|
||||||
|
|
||||||
$format = $this->userData['format']
|
$format = $this->userData['format']
|
||||||
or returnClientError('You must specify a format!');
|
or returnClientError('You must specify a format!');
|
||||||
|
|
||||||
// DEPRECATED: 'nameFormat' scheme is replaced by 'name' in format parameter values
|
$bridgeFac = new \BridgeFactory();
|
||||||
// this is to keep compatibility until futher complete removal
|
$bridgeFac->setWorkingDir(PATH_LIB_BRIDGES);
|
||||||
if(($pos = strpos($format, 'Format')) === (strlen($format) - strlen('Format'))) {
|
|
||||||
$format = substr($format, 0, $pos);
|
|
||||||
}
|
|
||||||
|
|
||||||
// whitelist control
|
// whitelist control
|
||||||
if(!Bridge::isWhitelisted($bridge)) {
|
if(!$bridgeFac->isWhitelisted($bridge)) {
|
||||||
throw new \Exception('This bridge is not whitelisted', 401);
|
throw new \Exception('This bridge is not whitelisted', 401);
|
||||||
die;
|
die;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Data retrieval
|
// Data retrieval
|
||||||
$bridge = Bridge::create($bridge);
|
$bridge = $bridgeFac->create($bridge);
|
||||||
|
|
||||||
$noproxy = array_key_exists('_noproxy', $this->userData)
|
$noproxy = array_key_exists('_noproxy', $this->userData)
|
||||||
&& filter_var($this->userData['_noproxy'], FILTER_VALIDATE_BOOLEAN);
|
&& filter_var($this->userData['_noproxy'], FILTER_VALIDATE_BOOLEAN);
|
||||||
|
@ -85,7 +91,9 @@ class DisplayAction extends ActionAbstract {
|
||||||
);
|
);
|
||||||
|
|
||||||
// Initialize cache
|
// Initialize cache
|
||||||
$cache = Cache::create(Configuration::getConfig('cache', 'type'));
|
$cacheFac = new CacheFactory();
|
||||||
|
$cacheFac->setWorkingDir(PATH_LIB_CACHES);
|
||||||
|
$cache = $cacheFac->create(Configuration::getConfig('cache', 'type'));
|
||||||
$cache->setScope('');
|
$cache->setScope('');
|
||||||
$cache->purgeCache(86400); // 24 hours
|
$cache->purgeCache(86400); // 24 hours
|
||||||
$cache->setKey($cache_params);
|
$cache->setKey($cache_params);
|
||||||
|
@ -147,6 +155,8 @@ class DisplayAction extends ActionAbstract {
|
||||||
} catch(Error $e) {
|
} catch(Error $e) {
|
||||||
error_log($e);
|
error_log($e);
|
||||||
|
|
||||||
|
if(logBridgeError($bridge::NAME, $e->getCode()) >= Configuration::getConfig('error', 'report_limit')) {
|
||||||
|
if(Configuration::getConfig('error', 'output') === 'feed') {
|
||||||
$item = new \FeedItem();
|
$item = new \FeedItem();
|
||||||
|
|
||||||
// Create "new" error message every 24 hours
|
// Create "new" error message every 24 hours
|
||||||
|
@ -179,9 +189,16 @@ class DisplayAction extends ActionAbstract {
|
||||||
$item->setContent(buildBridgeException($e, $bridge));
|
$item->setContent(buildBridgeException($e, $bridge));
|
||||||
|
|
||||||
$items[] = $item;
|
$items[] = $item;
|
||||||
|
} elseif(Configuration::getConfig('error', 'output') === 'http') {
|
||||||
|
header('Content-Type: text/html', true, $this->get_return_code($e));
|
||||||
|
die(buildTransformException($e, $bridge));
|
||||||
|
}
|
||||||
|
}
|
||||||
} catch(Exception $e) {
|
} catch(Exception $e) {
|
||||||
error_log($e);
|
error_log($e);
|
||||||
|
|
||||||
|
if(logBridgeError($bridge::NAME, $e->getCode()) >= Configuration::getConfig('error', 'report_limit')) {
|
||||||
|
if(Configuration::getConfig('error', 'output') === 'feed') {
|
||||||
$item = new \FeedItem();
|
$item = new \FeedItem();
|
||||||
|
|
||||||
// Create "new" error message every 24 hours
|
// Create "new" error message every 24 hours
|
||||||
|
@ -204,6 +221,11 @@ class DisplayAction extends ActionAbstract {
|
||||||
$item->setContent(buildBridgeException($e, $bridge));
|
$item->setContent(buildBridgeException($e, $bridge));
|
||||||
|
|
||||||
$items[] = $item;
|
$items[] = $item;
|
||||||
|
} elseif(Configuration::getConfig('error', 'output') === 'http') {
|
||||||
|
header('Content-Type: text/html', true, $this->get_return_code($e));
|
||||||
|
die(buildTransformException($e, $bridge));
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Store data in cache
|
// Store data in cache
|
||||||
|
@ -216,7 +238,9 @@ class DisplayAction extends ActionAbstract {
|
||||||
|
|
||||||
// Data transformation
|
// Data transformation
|
||||||
try {
|
try {
|
||||||
$format = Format::create($format);
|
$formatFac = new FormatFactory();
|
||||||
|
$formatFac->setWorkingDir(PATH_LIB_FORMATS);
|
||||||
|
$format = $formatFac->create($format);
|
||||||
$format->setItems($items);
|
$format->setItems($items);
|
||||||
$format->setExtraInfos($infos);
|
$format->setExtraInfos($infos);
|
||||||
$format->setLastModified($cache->getTime());
|
$format->setLastModified($cache->getTime());
|
||||||
|
|
|
@ -17,9 +17,12 @@ class ListAction extends ActionAbstract {
|
||||||
$list->bridges = array();
|
$list->bridges = array();
|
||||||
$list->total = 0;
|
$list->total = 0;
|
||||||
|
|
||||||
foreach(Bridge::getBridgeNames() as $bridgeName) {
|
$bridgeFac = new \BridgeFactory();
|
||||||
|
$bridgeFac->setWorkingDir(PATH_LIB_BRIDGES);
|
||||||
|
|
||||||
$bridge = Bridge::create($bridgeName);
|
foreach($bridgeFac->getBridgeNames() as $bridgeName) {
|
||||||
|
|
||||||
|
$bridge = $bridgeFac->create($bridgeName);
|
||||||
|
|
||||||
if($bridge === false) { // Broken bridge, show as inactive
|
if($bridge === false) { // Broken bridge, show as inactive
|
||||||
|
|
||||||
|
@ -31,7 +34,7 @@ class ListAction extends ActionAbstract {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
$status = Bridge::isWhitelisted($bridgeName) ? 'active' : 'inactive';
|
$status = $bridgeFac->isWhitelisted($bridgeName) ? 'active' : 'inactive';
|
||||||
|
|
||||||
$list->bridges[$bridgeName] = array(
|
$list->bridges[$bridgeName] = array(
|
||||||
'status' => $status,
|
'status' => $status,
|
||||||
|
|
|
@ -134,11 +134,11 @@ EOT;
|
||||||
// data-asin="B00WTHJ5SU" data-asin-price="14.99" data-asin-shipping="0"
|
// data-asin="B00WTHJ5SU" data-asin-price="14.99" data-asin-shipping="0"
|
||||||
// data-asin-currency-code="USD" data-substitute-count="-1" ... />
|
// data-asin-currency-code="USD" data-substitute-count="-1" ... />
|
||||||
if ($asinData) {
|
if ($asinData) {
|
||||||
return [
|
return array(
|
||||||
'price' => $asinData->getAttribute('data-asin-price'),
|
'price' => $asinData->getAttribute('data-asin-price'),
|
||||||
'currency' => $asinData->getAttribute('data-asin-currency-code'),
|
'currency' => $asinData->getAttribute('data-asin-currency-code'),
|
||||||
'shipping' => $asinData->getAttribute('data-asin-shipping')
|
'shipping' => $asinData->getAttribute('data-asin-shipping')
|
||||||
];
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
return false;
|
return false;
|
||||||
|
@ -150,11 +150,11 @@ EOT;
|
||||||
preg_match('/^\s*([A-Z]{3}|£|\$)\s?([\d.,]+)\s*$/', $priceDiv->plaintext, $matches);
|
preg_match('/^\s*([A-Z]{3}|£|\$)\s?([\d.,]+)\s*$/', $priceDiv->plaintext, $matches);
|
||||||
|
|
||||||
if (count($matches) === 3) {
|
if (count($matches) === 3) {
|
||||||
return [
|
return array(
|
||||||
'price' => $matches[2],
|
'price' => $matches[2],
|
||||||
'currency' => $matches[1],
|
'currency' => $matches[1],
|
||||||
'shipping' => '0'
|
'shipping' => '0'
|
||||||
];
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
return false;
|
return false;
|
||||||
|
|
149
bridges/AppleAppStoreBridge.php
Normal file
149
bridges/AppleAppStoreBridge.php
Normal file
|
@ -0,0 +1,149 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
class AppleAppStoreBridge extends BridgeAbstract {
|
||||||
|
|
||||||
|
const MAINTAINER = 'captn3m0';
|
||||||
|
const NAME = 'Apple App Store';
|
||||||
|
const URI = 'https://apps.apple.com/';
|
||||||
|
const CACHE_TIMEOUT = 3600; // 1h
|
||||||
|
const DESCRIPTION = 'Returns version updates for a specific application';
|
||||||
|
|
||||||
|
const PARAMETERS = array(array(
|
||||||
|
'id' => array(
|
||||||
|
'name' => 'Application ID',
|
||||||
|
'required' => true,
|
||||||
|
'exampleValue' => '310633997'
|
||||||
|
),
|
||||||
|
'p' => array(
|
||||||
|
'name' => 'Platform',
|
||||||
|
'type' => 'list',
|
||||||
|
'values' => array(
|
||||||
|
'iPad' => 'ipad',
|
||||||
|
'iPhone' => 'iphone',
|
||||||
|
'Mac' => 'mac',
|
||||||
|
|
||||||
|
// The following 2 are present in responses
|
||||||
|
// but not yet tested
|
||||||
|
'Web' => 'web',
|
||||||
|
'Apple TV' => 'appletv',
|
||||||
|
),
|
||||||
|
'defaultValue' => 'iphone',
|
||||||
|
),
|
||||||
|
'country' => array(
|
||||||
|
'name' => 'Store Country',
|
||||||
|
'type' => 'list',
|
||||||
|
'values' => array(
|
||||||
|
'US' => 'US',
|
||||||
|
'India' => 'IN',
|
||||||
|
'Canada' => 'CA'
|
||||||
|
),
|
||||||
|
'defaultValue' => 'US',
|
||||||
|
),
|
||||||
|
));
|
||||||
|
|
||||||
|
const PLATFORM_MAPPING = array(
|
||||||
|
'iphone' => 'ios',
|
||||||
|
'ipad' => 'ios',
|
||||||
|
);
|
||||||
|
|
||||||
|
private function makeHtmlUrl($id, $country){
|
||||||
|
return 'https://apps.apple.com/' . $country . '/app/id' . $id;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function makeJsonUrl($id, $platform, $country){
|
||||||
|
return "https://amp-api.apps.apple.com/v1/catalog/$country/apps/$id?platform=$platform&extend=versionHistory";
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getName(){
|
||||||
|
if (isset($this->name)) {
|
||||||
|
return $this->name . ' - AppStore Updates';
|
||||||
|
}
|
||||||
|
|
||||||
|
return parent::getName();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* In case of some platforms, the data is present in the initial response
|
||||||
|
*/
|
||||||
|
private function getDataFromShoebox($id, $platform, $country){
|
||||||
|
$uri = $this->makeHtmlUrl($id, $country);
|
||||||
|
$html = getSimpleHTMLDOMCached($uri, 3600);
|
||||||
|
$script = $html->find('script[id="shoebox-ember-data-store"]', 0);
|
||||||
|
|
||||||
|
$json = json_decode($script->innertext, true);
|
||||||
|
return $json['data'];
|
||||||
|
}
|
||||||
|
|
||||||
|
private function getJWTToken($id, $platform, $country){
|
||||||
|
$uri = $this->makeHtmlUrl($id, $country);
|
||||||
|
|
||||||
|
$html = getSimpleHTMLDOMCached($uri, 3600);
|
||||||
|
|
||||||
|
$meta = $html->find('meta[name="web-experience-app/config/environment"]', 0);
|
||||||
|
|
||||||
|
$json = urldecode($meta->content);
|
||||||
|
|
||||||
|
$json = json_decode($json);
|
||||||
|
|
||||||
|
return $json->MEDIA_API->token;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function getAppData($id, $platform, $country, $token){
|
||||||
|
$uri = $this->makeJsonUrl($id, $platform, $country);
|
||||||
|
|
||||||
|
$headers = array(
|
||||||
|
"Authorization: Bearer $token",
|
||||||
|
);
|
||||||
|
|
||||||
|
$json = json_decode(getContents($uri, $headers), true);
|
||||||
|
|
||||||
|
return $json['data'][0];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Parses the version history from the data received
|
||||||
|
* @return array list of versions with details on each element
|
||||||
|
*/
|
||||||
|
private function getVersionHistory($data, $platform){
|
||||||
|
switch($platform) {
|
||||||
|
case 'mac':
|
||||||
|
return $data['relationships']['platforms']['data'][0]['attributes']['versionHistory'];
|
||||||
|
default:
|
||||||
|
$os = self::PLATFORM_MAPPING[$platform];
|
||||||
|
return $data['attributes']['platformAttributes'][$os]['versionHistory'];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public function collectData() {
|
||||||
|
$id = $this->getInput('id');
|
||||||
|
$country = $this->getInput('country');
|
||||||
|
$platform = $this->getInput('p');
|
||||||
|
|
||||||
|
switch ($platform) {
|
||||||
|
case 'mac':
|
||||||
|
$data = $this->getDataFromShoebox($id, $platform, $country);
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
$token = $this->getJWTToken($id, $platform, $country);
|
||||||
|
$data = $this->getAppData($id, $platform, $country, $token);
|
||||||
|
}
|
||||||
|
|
||||||
|
$versionHistory = $this->getVersionHistory($data, $platform);
|
||||||
|
$name = $this->name = $data['attributes']['name'];
|
||||||
|
$author = $data['attributes']['artistName'];
|
||||||
|
|
||||||
|
foreach ($versionHistory as $row) {
|
||||||
|
$item = array();
|
||||||
|
|
||||||
|
$item['content'] = nl2br($row['releaseNotes']);
|
||||||
|
$item['title'] = $name . ' - ' . $row['versionDisplay'];
|
||||||
|
$item['timestamp'] = $row['releaseDate'];
|
||||||
|
$item['author'] = $author;
|
||||||
|
|
||||||
|
$item['uri'] = $this->makeHtmlUrl($id, $country);
|
||||||
|
|
||||||
|
$this->items[] = $item;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -5,19 +5,19 @@ class AppleMusicBridge extends BridgeAbstract {
|
||||||
const URI = 'https://www.apple.com';
|
const URI = 'https://www.apple.com';
|
||||||
const DESCRIPTION = 'Fetches the latest releases from an artist';
|
const DESCRIPTION = 'Fetches the latest releases from an artist';
|
||||||
const MAINTAINER = 'Limero';
|
const MAINTAINER = 'Limero';
|
||||||
const PARAMETERS = [[
|
const PARAMETERS = array(array(
|
||||||
'url' => [
|
'url' => array(
|
||||||
'name' => 'Artist URL',
|
'name' => 'Artist URL',
|
||||||
'exampleValue' => 'https://itunes.apple.com/us/artist/dunderpatrullen/329796274',
|
'exampleValue' => 'https://itunes.apple.com/us/artist/dunderpatrullen/329796274',
|
||||||
'required' => true,
|
'required' => true,
|
||||||
],
|
),
|
||||||
'imgSize' => [
|
'imgSize' => array(
|
||||||
'name' => 'Image size for thumbnails (in px)',
|
'name' => 'Image size for thumbnails (in px)',
|
||||||
'type' => 'number',
|
'type' => 'number',
|
||||||
'defaultValue' => 512,
|
'defaultValue' => 512,
|
||||||
'required' => true,
|
'required' => true,
|
||||||
]
|
)
|
||||||
]];
|
));
|
||||||
const CACHE_TIMEOUT = 21600; // 6 hours
|
const CACHE_TIMEOUT = 21600; // 6 hours
|
||||||
|
|
||||||
public function collectData() {
|
public function collectData() {
|
||||||
|
@ -36,12 +36,12 @@ class AppleMusicBridge extends BridgeAbstract {
|
||||||
// Loop through each object
|
// Loop through each object
|
||||||
foreach ($json->included as $obj) {
|
foreach ($json->included as $obj) {
|
||||||
if ($obj->type === 'lockup/album') {
|
if ($obj->type === 'lockup/album') {
|
||||||
$this->items[] = [
|
$this->items[] = array(
|
||||||
'title' => $obj->attributes->artistName . ' - ' . $obj->attributes->name,
|
'title' => $obj->attributes->artistName . ' - ' . $obj->attributes->name,
|
||||||
'uri' => $obj->attributes->url,
|
'uri' => $obj->attributes->url,
|
||||||
'timestamp' => $obj->attributes->releaseDate,
|
'timestamp' => $obj->attributes->releaseDate,
|
||||||
'enclosures' => $obj->relationships->artwork->data->id,
|
'enclosures' => $obj->relationships->artwork->data->id,
|
||||||
];
|
);
|
||||||
} elseif ($obj->type === 'image') {
|
} elseif ($obj->type === 'image') {
|
||||||
$images[$obj->id] = $obj->attributes->url;
|
$images[$obj->id] = $obj->attributes->url;
|
||||||
}
|
}
|
||||||
|
@ -49,9 +49,9 @@ class AppleMusicBridge extends BridgeAbstract {
|
||||||
|
|
||||||
// Add the images to each item
|
// Add the images to each item
|
||||||
foreach ($this->items as &$item) {
|
foreach ($this->items as &$item) {
|
||||||
$item['enclosures'] = [
|
$item['enclosures'] = array(
|
||||||
str_replace('{w}x{h}bb.{f}', $imgSize . 'x0w.jpg', $images[$item['enclosures']]),
|
str_replace('{w}x{h}bb.{f}', $imgSize . 'x0w.jpg', $images[$item['enclosures']]),
|
||||||
];
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Sort the order to put the latest albums first
|
// Sort the order to put the latest albums first
|
||||||
|
|
4638
bridges/AtmoNouvelleAquitaineBridge.php
Normal file
4638
bridges/AtmoNouvelleAquitaineBridge.php
Normal file
File diff suppressed because it is too large
Load diff
|
@ -15,16 +15,6 @@ class AutoJMBridge extends BridgeAbstract {
|
||||||
'title' => 'URL d\'une recherche avec filtre de véhicules sans le http://www.autojm.fr/',
|
'title' => 'URL d\'une recherche avec filtre de véhicules sans le http://www.autojm.fr/',
|
||||||
'exampleValue' => 'achat-voitures-neuves-peugeot-nouvelle-308-5p'
|
'exampleValue' => 'achat-voitures-neuves-peugeot-nouvelle-308-5p'
|
||||||
),
|
),
|
||||||
'isDispo' => array(
|
|
||||||
'name' => 'Disponibilité',
|
|
||||||
'type' => 'list',
|
|
||||||
'values' => array(
|
|
||||||
'-' => '',
|
|
||||||
'En stock' => 1,
|
|
||||||
'Sur commande' => 0
|
|
||||||
),
|
|
||||||
'title' => 'Critère de disponibilité'
|
|
||||||
),
|
|
||||||
'energy' => array(
|
'energy' => array(
|
||||||
'name' => 'Carburant',
|
'name' => 'Carburant',
|
||||||
'type' => 'list',
|
'type' => 'list',
|
||||||
|
@ -92,7 +82,6 @@ class AutoJMBridge extends BridgeAbstract {
|
||||||
|
|
||||||
// Build the form
|
// Build the form
|
||||||
$post_data = array(
|
$post_data = array(
|
||||||
'form[isDispo]' => $this->getInput('isDispo'),
|
|
||||||
'form[energy]' => $this->getInput('energy'),
|
'form[energy]' => $this->getInput('energy'),
|
||||||
'form[transmission]' => $this->getInput('transmission'),
|
'form[transmission]' => $this->getInput('transmission'),
|
||||||
'form[priceMin]' => $this->getInput('priceMin'),
|
'form[priceMin]' => $this->getInput('priceMin'),
|
||||||
|
@ -121,7 +110,7 @@ class AutoJMBridge extends BridgeAbstract {
|
||||||
$html = str_get_html($data->content);
|
$html = str_get_html($data->content);
|
||||||
|
|
||||||
// Go through every finisha of the model
|
// Go through every finisha of the model
|
||||||
$list = $html->find('h2');
|
$list = $html->find('h3');
|
||||||
foreach ($list as $finish) {
|
foreach ($list as $finish) {
|
||||||
$finish_name = $finish->plaintext;
|
$finish_name = $finish->plaintext;
|
||||||
$motorizations = $finish->next_sibling()->find('li');
|
$motorizations = $finish->next_sibling()->find('li');
|
||||||
|
|
|
@ -55,9 +55,7 @@ class BAEBridge extends BridgeAbstract {
|
||||||
|
|
||||||
$content .= '<hr>';
|
$content .= '<hr>';
|
||||||
$content .= $htmlDetail->find('section', 0)->innertext;
|
$content .= $htmlDetail->find('section', 0)->innertext;
|
||||||
$content = str_replace('src="/', 'src="' . parent::getURI() . '/', $content);
|
$item['content'] = defaultLinkTo($content, parent::getURI());
|
||||||
$content = str_replace('href="/', 'href="' . parent::getURI() . '/', $content);
|
|
||||||
$item['content'] = $content;
|
|
||||||
$image = $htmlDetail->find('#zoom', 0);
|
$image = $htmlDetail->find('#zoom', 0);
|
||||||
if ($image) {
|
if ($image) {
|
||||||
$item['enclosures'] = array(parent::getURI() . $image->getAttribute('src'));
|
$item['enclosures'] = array(parent::getURI() . $image->getAttribute('src'));
|
||||||
|
|
|
@ -1,27 +1,81 @@
|
||||||
<?php
|
<?php
|
||||||
class BandcampBridge extends BridgeAbstract {
|
class BandcampBridge extends BridgeAbstract {
|
||||||
|
|
||||||
const MAINTAINER = 'sebsauvage';
|
const MAINTAINER = 'sebsauvage, Roliga';
|
||||||
const NAME = 'Bandcamp Tag';
|
const NAME = 'Bandcamp Bridge';
|
||||||
const URI = 'https://bandcamp.com/';
|
const URI = 'https://bandcamp.com/';
|
||||||
const CACHE_TIMEOUT = 600; // 10min
|
const CACHE_TIMEOUT = 600; // 10min
|
||||||
const DESCRIPTION = 'New bandcamp release by tag';
|
const DESCRIPTION = 'New bandcamp releases by tag, band or album';
|
||||||
const PARAMETERS = array( array(
|
const PARAMETERS = array(
|
||||||
|
'By tag' => array(
|
||||||
'tag' => array(
|
'tag' => array(
|
||||||
'name' => 'tag',
|
'name' => 'tag',
|
||||||
'type' => 'text',
|
'type' => 'text',
|
||||||
'required' => true
|
'required' => true
|
||||||
)
|
)
|
||||||
));
|
),
|
||||||
|
'By band' => array(
|
||||||
|
'band' => array(
|
||||||
|
'name' => 'band',
|
||||||
|
'type' => 'text',
|
||||||
|
'title' => 'Band name as seen in the band page URL',
|
||||||
|
'required' => true
|
||||||
|
),
|
||||||
|
'type' => array(
|
||||||
|
'name' => 'Articles are',
|
||||||
|
'type' => 'list',
|
||||||
|
'values' => array(
|
||||||
|
'Releases' => 'releases',
|
||||||
|
'Releases, new one when track list changes' => 'changes',
|
||||||
|
'Individual tracks' => 'tracks'
|
||||||
|
),
|
||||||
|
'defaultValue' => 'changes'
|
||||||
|
),
|
||||||
|
'limit' => array(
|
||||||
|
'name' => 'limit',
|
||||||
|
'type' => 'number',
|
||||||
|
'title' => 'Number of releases to return',
|
||||||
|
'defaultValue' => 5
|
||||||
|
)
|
||||||
|
),
|
||||||
|
'By album' => array(
|
||||||
|
'band' => array(
|
||||||
|
'name' => 'band',
|
||||||
|
'type' => 'text',
|
||||||
|
'title' => 'Band name as seen in the album page URL',
|
||||||
|
'required' => true
|
||||||
|
),
|
||||||
|
'album' => array(
|
||||||
|
'name' => 'album',
|
||||||
|
'type' => 'text',
|
||||||
|
'title' => 'Album name as seen in the album page URL',
|
||||||
|
'required' => true
|
||||||
|
),
|
||||||
|
'type' => array(
|
||||||
|
'name' => 'Articles are',
|
||||||
|
'type' => 'list',
|
||||||
|
'values' => array(
|
||||||
|
'Releases' => 'releases',
|
||||||
|
'Releases, new one when track list changes' => 'changes',
|
||||||
|
'Individual tracks' => 'tracks'
|
||||||
|
),
|
||||||
|
'defaultValue' => 'tracks'
|
||||||
|
)
|
||||||
|
)
|
||||||
|
);
|
||||||
const IMGURI = 'https://f4.bcbits.com/';
|
const IMGURI = 'https://f4.bcbits.com/';
|
||||||
const IMGSIZE_300PX = 23;
|
const IMGSIZE_300PX = 23;
|
||||||
const IMGSIZE_700PX = 16;
|
const IMGSIZE_700PX = 16;
|
||||||
|
|
||||||
|
private $feedName;
|
||||||
|
|
||||||
public function getIcon() {
|
public function getIcon() {
|
||||||
return 'https://s4.bcbits.com/img/bc_favicon.ico';
|
return 'https://s4.bcbits.com/img/bc_favicon.ico';
|
||||||
}
|
}
|
||||||
|
|
||||||
public function collectData(){
|
public function collectData(){
|
||||||
|
switch($this->queriedContext) {
|
||||||
|
case 'By tag':
|
||||||
$url = self::URI . 'api/hub/1/dig_deeper';
|
$url = self::URI . 'api/hub/1/dig_deeper';
|
||||||
$data = $this->buildRequestJson();
|
$data = $this->buildRequestJson();
|
||||||
$header = array(
|
$header = array(
|
||||||
|
@ -66,6 +120,141 @@ class BandcampBridge extends BridgeAbstract {
|
||||||
$item['enclosures'] = array($img);
|
$item['enclosures'] = array($img);
|
||||||
$this->items[] = $item;
|
$this->items[] = $item;
|
||||||
}
|
}
|
||||||
|
break;
|
||||||
|
case 'By band':
|
||||||
|
case 'By album':
|
||||||
|
$html = getSimpleHTMLDOMCached($this->getURI(), 86400);
|
||||||
|
|
||||||
|
$titleElement = $html->find('head meta[name=title]', 0)
|
||||||
|
or returnServerError('Unable to find title on: ' . $this->getURI());
|
||||||
|
$this->feedName = $titleElement->content;
|
||||||
|
|
||||||
|
$regex = '/band_id=(\d+)/';
|
||||||
|
if(preg_match($regex, $html, $matches) == false)
|
||||||
|
returnServerError('Unable to find band ID on: ' . $this->getURI());
|
||||||
|
$band_id = $matches[1];
|
||||||
|
|
||||||
|
$tralbums = array();
|
||||||
|
switch($this->queriedContext) {
|
||||||
|
case 'By band':
|
||||||
|
$query_data = array(
|
||||||
|
'band_id' => $band_id
|
||||||
|
);
|
||||||
|
$band_data = $this->apiGet('mobile/22/band_details', $query_data);
|
||||||
|
|
||||||
|
$num_albums = min(count($band_data->discography), $this->getInput('limit'));
|
||||||
|
for($i = 0; $i < $num_albums; $i++) {
|
||||||
|
$album_basic_data = $band_data->discography[$i];
|
||||||
|
|
||||||
|
// 'a' or 't' for albums and individual tracks respectively
|
||||||
|
$tralbum_type = substr($album_basic_data->item_type, 0, 1);
|
||||||
|
|
||||||
|
$query_data = array(
|
||||||
|
'band_id' => $band_id,
|
||||||
|
'tralbum_type' => $tralbum_type,
|
||||||
|
'tralbum_id' => $album_basic_data->item_id
|
||||||
|
);
|
||||||
|
$tralbums[] = $this->apiGet('mobile/22/tralbum_details', $query_data);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 'By album':
|
||||||
|
$regex = '/album=(\d+)/';
|
||||||
|
if(preg_match($regex, $html, $matches) == false)
|
||||||
|
returnServerError('Unable to find album ID on: ' . $this->getURI());
|
||||||
|
$album_id = $matches[1];
|
||||||
|
|
||||||
|
$query_data = array(
|
||||||
|
'band_id' => $band_id,
|
||||||
|
'tralbum_type' => 'a',
|
||||||
|
'tralbum_id' => $album_id
|
||||||
|
);
|
||||||
|
$tralbums[] = $this->apiGet('mobile/22/tralbum_details', $query_data);
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach ($tralbums as $tralbum_data) {
|
||||||
|
if ($tralbum_data->type === 'a' && $this->getInput('type') === 'tracks') {
|
||||||
|
foreach ($tralbum_data->tracks as $track) {
|
||||||
|
$query_data = array(
|
||||||
|
'band_id' => $band_id,
|
||||||
|
'tralbum_type' => 't',
|
||||||
|
'tralbum_id' => $track->track_id
|
||||||
|
);
|
||||||
|
$track_data = $this->apiGet('mobile/22/tralbum_details', $query_data);
|
||||||
|
|
||||||
|
$this->items[] = $this->buildTralbumItem($track_data);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
$this->items[] = $this->buildTralbumItem($tralbum_data);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private function buildTralbumItem($tralbum_data){
|
||||||
|
$band_data = $tralbum_data->band;
|
||||||
|
|
||||||
|
// Format title like: ARTIST - ALBUM/TRACK (OPTIONAL RELEASER)
|
||||||
|
// Format artist/author like: ARTIST (OPTIONAL RELEASER)
|
||||||
|
//
|
||||||
|
// If the album/track is released under a label/a band other than the artist
|
||||||
|
// themselves, append that releaser name to the title and artist/author.
|
||||||
|
//
|
||||||
|
// This sadly doesn't always work right for individual tracks as the artist
|
||||||
|
// of the track is always set to the releaser.
|
||||||
|
$artist = $tralbum_data->tralbum_artist;
|
||||||
|
$full_title = $artist . ' - ' . $tralbum_data->title;
|
||||||
|
$full_artist = $artist;
|
||||||
|
if (isset($tralbum_data->label)) {
|
||||||
|
$full_title .= ' (' . $tralbum_data->label . ')';
|
||||||
|
$full_artist .= ' (' . $tralbum_data->label . ')';
|
||||||
|
} elseif ($band_data->name !== $artist) {
|
||||||
|
$full_title .= ' (' . $band_data->name . ')';
|
||||||
|
$full_artist .= ' (' . $band_data->name . ')';
|
||||||
|
}
|
||||||
|
|
||||||
|
$small_img = $this->getImageUrl($tralbum_data->art_id, self::IMGSIZE_300PX);
|
||||||
|
$img = $this->getImageUrl($tralbum_data->art_id, self::IMGSIZE_700PX);
|
||||||
|
|
||||||
|
$item = array(
|
||||||
|
'uri' => $tralbum_data->bandcamp_url,
|
||||||
|
'author' => $full_artist,
|
||||||
|
'title' => $full_title,
|
||||||
|
'enclosures' => array($img),
|
||||||
|
'timestamp' => $tralbum_data->release_date
|
||||||
|
);
|
||||||
|
|
||||||
|
$item['categories'] = array();
|
||||||
|
foreach ($tralbum_data->tags as $tag) {
|
||||||
|
$item['categories'][] = $tag->norm_name;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Give articles a unique UID depending on its track list
|
||||||
|
// Releases should then show up as new articles when tracks are added
|
||||||
|
if ($this->getInput('type') === 'changes') {
|
||||||
|
$item['uid'] = "bandcamp/$band_data->band_id/$tralbum_data->id/";
|
||||||
|
foreach ($tralbum_data->tracks as $track) {
|
||||||
|
$item['uid'] .= $track->track_id;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$item['content'] = "<img src='$small_img' /><br/>$full_title<br/>";
|
||||||
|
if ($tralbum_data->type === 'a') {
|
||||||
|
$item['content'] .= '<ol>';
|
||||||
|
foreach ($tralbum_data->tracks as $track) {
|
||||||
|
$item['content'] .= "<li>$track->title</li>";
|
||||||
|
}
|
||||||
|
$item['content'] .= '</ol>';
|
||||||
|
}
|
||||||
|
if (!empty($tralbum_data->about)) {
|
||||||
|
$item['content'] .= '<p>'
|
||||||
|
. nl2br($tralbum_data->about)
|
||||||
|
. '</p>';
|
||||||
|
}
|
||||||
|
|
||||||
|
return $item;
|
||||||
}
|
}
|
||||||
|
|
||||||
private function buildRequestJson(){
|
private function buildRequestJson(){
|
||||||
|
@ -81,11 +270,94 @@ class BandcampBridge extends BridgeAbstract {
|
||||||
return self::IMGURI . 'img/a' . $id . '_' . $size . '.jpg';
|
return self::IMGURI . 'img/a' . $id . '_' . $size . '.jpg';
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private function apiGet($endpoint, $query_data) {
|
||||||
|
$url = self::URI . 'api/' . $endpoint . '?' . http_build_query($query_data);
|
||||||
|
$data = json_decode(getContents($url))
|
||||||
|
or returnServerError('API request to "' . $url . '" failed.');
|
||||||
|
return $data;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getURI(){
|
||||||
|
switch($this->queriedContext) {
|
||||||
|
case 'By tag':
|
||||||
|
if(!is_null($this->getInput('tag'))) {
|
||||||
|
return self::URI
|
||||||
|
. 'tag/'
|
||||||
|
. urlencode($this->getInput('tag'))
|
||||||
|
. '?sort_field=date';
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 'By band':
|
||||||
|
if(!is_null($this->getInput('band'))) {
|
||||||
|
return 'https://'
|
||||||
|
. $this->getInput('band')
|
||||||
|
. '.bandcamp.com/music';
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 'By album':
|
||||||
|
if(!is_null($this->getInput('band')) && !is_null($this->getInput('album'))) {
|
||||||
|
return 'https://'
|
||||||
|
. $this->getInput('band')
|
||||||
|
. '.bandcamp.com/album/'
|
||||||
|
. $this->getInput('album');
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return parent::getURI();
|
||||||
|
}
|
||||||
|
|
||||||
public function getName(){
|
public function getName(){
|
||||||
|
switch($this->queriedContext) {
|
||||||
|
case 'By tag':
|
||||||
if(!is_null($this->getInput('tag'))) {
|
if(!is_null($this->getInput('tag'))) {
|
||||||
return $this->getInput('tag') . ' - Bandcamp Tag';
|
return $this->getInput('tag') . ' - Bandcamp Tag';
|
||||||
}
|
}
|
||||||
|
break;
|
||||||
|
case 'By band':
|
||||||
|
if(isset($this->feedName)) {
|
||||||
|
return $this->feedName . ' - Bandcamp Band';
|
||||||
|
} elseif(!is_null($this->getInput('band'))) {
|
||||||
|
return $this->getInput('band') . ' - Bandcamp Band';
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 'By album':
|
||||||
|
if(isset($this->feedName)) {
|
||||||
|
return $this->feedName . ' - Bandcamp Album';
|
||||||
|
} elseif(!is_null($this->getInput('album'))) {
|
||||||
|
return $this->getInput('album') . ' - Bandcamp Album';
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
return parent::getName();
|
return parent::getName();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function detectParameters($url) {
|
||||||
|
$params = array();
|
||||||
|
|
||||||
|
// By tag
|
||||||
|
$regex = '/^(https?:\/\/)?bandcamp\.com\/tag\/([^\/.&?\n]+)/';
|
||||||
|
if(preg_match($regex, $url, $matches) > 0) {
|
||||||
|
$params['tag'] = urldecode($matches[2]);
|
||||||
|
return $params;
|
||||||
|
}
|
||||||
|
|
||||||
|
// By band
|
||||||
|
$regex = '/^(https?:\/\/)?([^\/.&?\n]+?)\.bandcamp\.com/';
|
||||||
|
if(preg_match($regex, $url, $matches) > 0) {
|
||||||
|
$params['band'] = urldecode($matches[2]);
|
||||||
|
return $params;
|
||||||
|
}
|
||||||
|
|
||||||
|
// By album
|
||||||
|
$regex = '/^(https?:\/\/)?([^\/.&?\n]+?)\.bandcamp\.com\/album\/([^\/.&?\n]+)/';
|
||||||
|
if(preg_match($regex, $url, $matches) > 0) {
|
||||||
|
$params['band'] = urldecode($matches[2]);
|
||||||
|
$params['album'] = urldecode($matches[3]);
|
||||||
|
return $params;
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,17 +3,11 @@ class BastaBridge extends BridgeAbstract {
|
||||||
|
|
||||||
const MAINTAINER = 'qwertygc';
|
const MAINTAINER = 'qwertygc';
|
||||||
const NAME = 'Bastamag Bridge';
|
const NAME = 'Bastamag Bridge';
|
||||||
const URI = 'http://www.bastamag.net/';
|
const URI = 'https://www.bastamag.net/';
|
||||||
const CACHE_TIMEOUT = 7200; // 2h
|
const CACHE_TIMEOUT = 7200; // 2h
|
||||||
const DESCRIPTION = 'Returns the newest articles.';
|
const DESCRIPTION = 'Returns the newest articles.';
|
||||||
|
|
||||||
public function collectData(){
|
public function collectData(){
|
||||||
// Replaces all relative image URLs by absolute URLs.
|
|
||||||
// Relative URLs always start with 'local/'!
|
|
||||||
function replaceImageUrl($content){
|
|
||||||
return preg_replace('/src=["\']{1}([^"\']+)/ims', 'src=\'' . self::URI . '$1\'', $content);
|
|
||||||
}
|
|
||||||
|
|
||||||
$html = getSimpleHTMLDOM(self::URI . 'spip.php?page=backend')
|
$html = getSimpleHTMLDOM(self::URI . 'spip.php?page=backend')
|
||||||
or returnServerError('Could not request Bastamag.');
|
or returnServerError('Could not request Bastamag.');
|
||||||
|
|
||||||
|
@ -25,7 +19,13 @@ class BastaBridge extends BridgeAbstract {
|
||||||
$item['title'] = $element->find('title', 0)->innertext;
|
$item['title'] = $element->find('title', 0)->innertext;
|
||||||
$item['uri'] = $element->find('guid', 0)->plaintext;
|
$item['uri'] = $element->find('guid', 0)->plaintext;
|
||||||
$item['timestamp'] = strtotime($element->find('dc:date', 0)->plaintext);
|
$item['timestamp'] = strtotime($element->find('dc:date', 0)->plaintext);
|
||||||
$item['content'] = replaceImageUrl(getSimpleHTMLDOM($item['uri'])->find('div.texte', 0)->innertext);
|
// Replaces all relative image URLs by absolute URLs.
|
||||||
|
// Relative URLs always start with 'local/'!
|
||||||
|
$item['content'] = preg_replace(
|
||||||
|
'/src=["\']{1}([^"\']+)/ims',
|
||||||
|
'src=\'' . self::URI . '$1\'',
|
||||||
|
getSimpleHTMLDOM($item['uri'])->find('div.texte', 0)->innertext
|
||||||
|
);
|
||||||
$this->items[] = $item;
|
$this->items[] = $item;
|
||||||
$limit++;
|
$limit++;
|
||||||
}
|
}
|
||||||
|
|
|
@ -92,7 +92,7 @@ class BingSearchBridge extends BridgeAbstract
|
||||||
or returnServerError('Could not request ' . self::NAME);
|
or returnServerError('Could not request ' . self::NAME);
|
||||||
$sizeKey = $this->getInput('image_size');
|
$sizeKey = $this->getInput('image_size');
|
||||||
|
|
||||||
$items = [];
|
$items = array();
|
||||||
foreach ($html->find('a.iusc') as $element) {
|
foreach ($html->find('a.iusc') as $element) {
|
||||||
$data = json_decode(htmlspecialchars_decode($element->getAttribute('m')), true);
|
$data = json_decode(htmlspecialchars_decode($element->getAttribute('m')), true);
|
||||||
|
|
||||||
|
|
|
@ -1,69 +0,0 @@
|
||||||
<?php
|
|
||||||
class BloombergBridge extends BridgeAbstract
|
|
||||||
{
|
|
||||||
const NAME = 'Bloomberg';
|
|
||||||
const URI = 'https://www.bloomberg.com/';
|
|
||||||
const DESCRIPTION = 'Trending stories from Bloomberg';
|
|
||||||
const MAINTAINER = 'mdemoss';
|
|
||||||
|
|
||||||
const PARAMETERS = array(
|
|
||||||
'Trending Stories' => array(),
|
|
||||||
'From Search' => array(
|
|
||||||
'q' => array(
|
|
||||||
'name' => 'Keyword',
|
|
||||||
'required' => true
|
|
||||||
)
|
|
||||||
)
|
|
||||||
);
|
|
||||||
|
|
||||||
public function getName()
|
|
||||||
{
|
|
||||||
switch($this->queriedContext) {
|
|
||||||
case 'Trending Stories':
|
|
||||||
return self::NAME . ' Trending Stories';
|
|
||||||
case 'From Search':
|
|
||||||
if (!is_null($this->getInput('q'))) {
|
|
||||||
return self::NAME . ' Search : ' . $this->getInput('q');
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
return parent::getName();
|
|
||||||
}
|
|
||||||
|
|
||||||
public function getIcon() {
|
|
||||||
return 'https://assets.bwbx.io/s3/javelin/public/hub/images/favicon-black-63fe5249d3.png';
|
|
||||||
}
|
|
||||||
|
|
||||||
public function collectData()
|
|
||||||
{
|
|
||||||
switch($this->queriedContext) {
|
|
||||||
case 'Trending Stories': // Get list of top new <article>s from the front page.
|
|
||||||
$html = getSimpleHTMLDOMCached($this->getURI(), 300);
|
|
||||||
$stories = $html->find('ul.top-news-v3__stories article.top-news-v3-story');
|
|
||||||
break;
|
|
||||||
case 'From Search': // Get list of <article> elements from search.
|
|
||||||
$html = getSimpleHTMLDOMCached(
|
|
||||||
$this->getURI() .
|
|
||||||
'search?sort=time:desc&page=1&query=' .
|
|
||||||
urlencode($this->getInput('q')), 300
|
|
||||||
);
|
|
||||||
$stories = $html->find('div.search-result-items article.search-result-story');
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
foreach ($stories as $element) {
|
|
||||||
$item['uri'] = $element->find('h1 a', 0)->href;
|
|
||||||
if (preg_match('#^https://#i', $item['uri']) !== 1) {
|
|
||||||
$item['uri'] = $this->getURI() . $item['uri'];
|
|
||||||
}
|
|
||||||
$articleHtml = getSimpleHTMLDOMCached($item['uri']);
|
|
||||||
if (!$articleHtml) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
$item['title'] = $element->find('h1 a', 0)->plaintext;
|
|
||||||
$item['timestamp'] = strtotime($articleHtml->find('meta[name=iso-8601-publish-date],meta[name=date]', 0)->content);
|
|
||||||
$item['content'] = $articleHtml->find('meta[name=description]', 0)->content;
|
|
||||||
$this->items[] = $item;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -92,6 +92,21 @@ class BrutBridge extends BridgeAbstract {
|
||||||
return parent::getURI();
|
return parent::getURI();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function getName() {
|
||||||
|
|
||||||
|
if (!is_null($this->getInput('edition')) && !is_null($this->getInput('category'))) {
|
||||||
|
$parameters = $this->getParameters();
|
||||||
|
|
||||||
|
$editionValues = array_flip($parameters[0]['edition']['values']);
|
||||||
|
$categoryValues = array_flip($parameters[0]['category']['values']);
|
||||||
|
|
||||||
|
return $categoryValues[$this->getInput('category')] . ' - ' .
|
||||||
|
$editionValues[$this->getInput('edition')] . ' - Brut.';
|
||||||
|
}
|
||||||
|
|
||||||
|
return parent::getName();
|
||||||
|
}
|
||||||
|
|
||||||
private function processDate($description) {
|
private function processDate($description) {
|
||||||
|
|
||||||
if ($this->getInput('edition') === 'uk') {
|
if ($this->getInput('edition') === 'uk') {
|
||||||
|
|
63
bridges/CNETFranceBridge.php
Normal file
63
bridges/CNETFranceBridge.php
Normal file
|
@ -0,0 +1,63 @@
|
||||||
|
<?php
|
||||||
|
class CNETFranceBridge extends FeedExpander
|
||||||
|
{
|
||||||
|
const MAINTAINER = 'leomaradan';
|
||||||
|
const NAME = 'CNET France';
|
||||||
|
const URI = 'https://www.cnetfrance.fr/';
|
||||||
|
const CACHE_TIMEOUT = 3600; // 1h
|
||||||
|
const DESCRIPTION = 'CNET France RSS with filters';
|
||||||
|
const PARAMETERS = array(
|
||||||
|
'filters' => array(
|
||||||
|
'title' => array(
|
||||||
|
'name' => 'Exclude by title',
|
||||||
|
'required' => false,
|
||||||
|
'title' => 'Title term, separated by semicolon (;)',
|
||||||
|
'defaultValue' => 'bon plan;bons plans;au meilleur prix;des meilleures offres;Amazon Prime Day;RED by SFR ou B&You'
|
||||||
|
),
|
||||||
|
'url' => array(
|
||||||
|
'name' => 'Exclude by url',
|
||||||
|
'required' => false,
|
||||||
|
'title' => 'URL term, separated by semicolon (;)',
|
||||||
|
'defaultValue' => 'bon-plan;bons-plans'
|
||||||
|
)
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
private $bannedTitle = array();
|
||||||
|
private $bannedURL = array();
|
||||||
|
|
||||||
|
public function collectData()
|
||||||
|
{
|
||||||
|
$title = $this->getInput('title');
|
||||||
|
$url = $this->getInput('url');
|
||||||
|
|
||||||
|
if ($title !== null) {
|
||||||
|
$this->bannedTitle = explode(';', $title);
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($url !== null) {
|
||||||
|
$this->bannedURL = explode(';', $url);
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->collectExpandableDatas('https://www.cnetfrance.fr/feeds/rss/news/');
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function parseItem($feedItem)
|
||||||
|
{
|
||||||
|
$item = parent::parseItem($feedItem);
|
||||||
|
|
||||||
|
foreach ($this->bannedTitle as $term) {
|
||||||
|
if (preg_match('/' . $term . '/mi', $item['title']) === 1) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach ($this->bannedURL as $term) {
|
||||||
|
if (preg_match('/' . $term . '/mi', $item['uri']) === 1) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $item;
|
||||||
|
}
|
||||||
|
}
|
|
@ -22,7 +22,7 @@ class CachetBridge extends BridgeAbstract {
|
||||||
);
|
);
|
||||||
const CACHE_TIMEOUT = 300;
|
const CACHE_TIMEOUT = 300;
|
||||||
|
|
||||||
private $componentCache = [];
|
private $componentCache = array();
|
||||||
|
|
||||||
public function getURI() {
|
public function getURI() {
|
||||||
return $this->getInput('host') === null ? 'https://cachethq.io/' : $this->getInput('host');
|
return $this->getInput('host') === null ? 'https://cachethq.io/' : $this->getInput('host');
|
||||||
|
@ -114,13 +114,13 @@ class CachetBridge extends BridgeAbstract {
|
||||||
$uidOrig = $permalink . $incident->created_at;
|
$uidOrig = $permalink . $incident->created_at;
|
||||||
$uid = hash('sha512', $uidOrig);
|
$uid = hash('sha512', $uidOrig);
|
||||||
$timestamp = strtotime($incident->created_at);
|
$timestamp = strtotime($incident->created_at);
|
||||||
$categories = [];
|
$categories = array();
|
||||||
$categories[] = $incident->human_status;
|
$categories[] = $incident->human_status;
|
||||||
if ($componentName !== '') {
|
if ($componentName !== '') {
|
||||||
$categories[] = $componentName;
|
$categories[] = $componentName;
|
||||||
}
|
}
|
||||||
|
|
||||||
$item = [];
|
$item = array();
|
||||||
$item['uri'] = $permalink;
|
$item['uri'] = $permalink;
|
||||||
$item['title'] = $title;
|
$item['title'] = $title;
|
||||||
$item['timestamp'] = $timestamp;
|
$item['timestamp'] = $timestamp;
|
||||||
|
|
|
@ -2,7 +2,7 @@
|
||||||
class CastorusBridge extends BridgeAbstract {
|
class CastorusBridge extends BridgeAbstract {
|
||||||
const MAINTAINER = 'logmanoriginal';
|
const MAINTAINER = 'logmanoriginal';
|
||||||
const NAME = 'Castorus Bridge';
|
const NAME = 'Castorus Bridge';
|
||||||
const URI = 'http://www.castorus.com';
|
const URI = 'https://www.castorus.com';
|
||||||
const CACHE_TIMEOUT = 600; // 10min
|
const CACHE_TIMEOUT = 600; // 10min
|
||||||
const DESCRIPTION = 'Returns the latest changes';
|
const DESCRIPTION = 'Returns the latest changes';
|
||||||
|
|
||||||
|
@ -83,7 +83,7 @@ class CastorusBridge extends BridgeAbstract {
|
||||||
if(!$html)
|
if(!$html)
|
||||||
returnServerError('Could not load data from ' . self::URI . '!');
|
returnServerError('Could not load data from ' . self::URI . '!');
|
||||||
|
|
||||||
$activities = $html->find('div#activite/li');
|
$activities = $html->find('div#activite > li');
|
||||||
|
|
||||||
if(!$activities)
|
if(!$activities)
|
||||||
returnServerError('Failed to find activities!');
|
returnServerError('Failed to find activities!');
|
||||||
|
|
|
@ -3,7 +3,7 @@ class CollegeDeFranceBridge extends BridgeAbstract {
|
||||||
|
|
||||||
const MAINTAINER = 'pit-fgfjiudghdf';
|
const MAINTAINER = 'pit-fgfjiudghdf';
|
||||||
const NAME = 'CollegeDeFrance';
|
const NAME = 'CollegeDeFrance';
|
||||||
const URI = 'http://www.college-de-france.fr/';
|
const URI = 'https://www.college-de-france.fr/';
|
||||||
const CACHE_TIMEOUT = 10800; // 3h
|
const CACHE_TIMEOUT = 10800; // 3h
|
||||||
const DESCRIPTION = 'Returns the latest audio and video from CollegeDeFrance';
|
const DESCRIPTION = 'Returns the latest audio and video from CollegeDeFrance';
|
||||||
|
|
||||||
|
|
65
bridges/ComicsKingdomBridge.php
Normal file
65
bridges/ComicsKingdomBridge.php
Normal file
|
@ -0,0 +1,65 @@
|
||||||
|
<?php
|
||||||
|
class ComicsKingdomBridge extends BridgeAbstract {
|
||||||
|
|
||||||
|
const MAINTAINER = 'stjohnjohnson';
|
||||||
|
const NAME = 'Comics Kingdom Unofficial RSS';
|
||||||
|
const URI = 'https://www.comicskingdom.com/';
|
||||||
|
const CACHE_TIMEOUT = 21600; // 6h
|
||||||
|
const DESCRIPTION = 'Comics Kingdom Unofficial RSS';
|
||||||
|
const PARAMETERS = array( array(
|
||||||
|
'comicname' => array(
|
||||||
|
'name' => 'comicname',
|
||||||
|
'type' => 'text',
|
||||||
|
'required' => true
|
||||||
|
)
|
||||||
|
));
|
||||||
|
|
||||||
|
public function collectData(){
|
||||||
|
$html = getSimpleHTMLDOM($this->getURI(), array(), array(), true, false)
|
||||||
|
or returnServerError('Could not request Comics Kingdom: ' . $this->getURI());
|
||||||
|
|
||||||
|
// Get author from first page
|
||||||
|
$author = $html->find('div.author p', 0)->plaintext
|
||||||
|
or returnServerError('Comics Kingdom comic does not exist: ' . $this->getURI());;
|
||||||
|
|
||||||
|
// Get current date/link
|
||||||
|
$link = $html->find('meta[property=og:url]', 0)->content;
|
||||||
|
for($i = 0; $i < 5; $i++) {
|
||||||
|
$item = array();
|
||||||
|
|
||||||
|
$page = getSimpleHTMLDOM($link)
|
||||||
|
or returnServerError('Could not request Comics Kingdom: ' . $link);
|
||||||
|
|
||||||
|
$imagelink = $page->find('meta[property=og:image]', 0)->content;
|
||||||
|
$prevSlug = $page->find('slider-arrow[:is-left-arrow=true]', 0);
|
||||||
|
$link = $this->getURI() . '/' . $prevSlug->getAttribute('date-slug');
|
||||||
|
|
||||||
|
$date = explode('/', $link);
|
||||||
|
|
||||||
|
$item['id'] = $imagelink;
|
||||||
|
$item['uri'] = $link;
|
||||||
|
$item['author'] = $author;
|
||||||
|
$item['title'] = 'Comics Kingdom ' . $this->getInput('comicname');
|
||||||
|
$item['timestamp'] = DateTime::createFromFormat('Y-m-d', $date[count($date) - 1])->getTimestamp();
|
||||||
|
$item['content'] = '<img src="' . $imagelink . '" />';
|
||||||
|
|
||||||
|
$this->items[] = $item;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getURI(){
|
||||||
|
if(!is_null($this->getInput('comicname'))) {
|
||||||
|
return self::URI . urlencode($this->getInput('comicname'));
|
||||||
|
}
|
||||||
|
|
||||||
|
return parent::getURI();
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getName(){
|
||||||
|
if(!is_null($this->getInput('comicname'))) {
|
||||||
|
return $this->getInput('comicname') . ' - Comics Kingdom';
|
||||||
|
}
|
||||||
|
|
||||||
|
return parent::getName();
|
||||||
|
}
|
||||||
|
}
|
|
@ -10,20 +10,20 @@ class ContainerLinuxReleasesBridge extends BridgeAbstract {
|
||||||
const BETA = 'beta';
|
const BETA = 'beta';
|
||||||
const ALPHA = 'alpha';
|
const ALPHA = 'alpha';
|
||||||
|
|
||||||
const PARAMETERS = [
|
const PARAMETERS = array(
|
||||||
[
|
array(
|
||||||
'channel' => [
|
'channel' => array(
|
||||||
'name' => 'Release Channel',
|
'name' => 'Release Channel',
|
||||||
'type' => 'list',
|
'type' => 'list',
|
||||||
'defaultValue' => self::STABLE,
|
'defaultValue' => self::STABLE,
|
||||||
'values' => [
|
'values' => array(
|
||||||
'Stable' => self::STABLE,
|
'Stable' => self::STABLE,
|
||||||
'Beta' => self::BETA,
|
'Beta' => self::BETA,
|
||||||
'Alpha' => self::ALPHA,
|
'Alpha' => self::ALPHA,
|
||||||
],
|
),
|
||||||
]
|
)
|
||||||
]
|
)
|
||||||
];
|
);
|
||||||
|
|
||||||
private function getReleaseFeed($jsonUrl) {
|
private function getReleaseFeed($jsonUrl) {
|
||||||
$json = getContents($jsonUrl)
|
$json = getContents($jsonUrl)
|
||||||
|
@ -39,7 +39,7 @@ class ContainerLinuxReleasesBridge extends BridgeAbstract {
|
||||||
$data = $this->getReleaseFeed($this->getJsonUri());
|
$data = $this->getReleaseFeed($this->getJsonUri());
|
||||||
|
|
||||||
foreach ($data as $releaseVersion => $release) {
|
foreach ($data as $releaseVersion => $release) {
|
||||||
$item = [];
|
$item = array();
|
||||||
|
|
||||||
$item['uri'] = "https://coreos.com/releases/#$releaseVersion";
|
$item['uri'] = "https://coreos.com/releases/#$releaseVersion";
|
||||||
$item['title'] = $releaseVersion;
|
$item['title'] = $releaseVersion;
|
||||||
|
|
109
bridges/CuriousCatBridge.php
Normal file
109
bridges/CuriousCatBridge.php
Normal file
|
@ -0,0 +1,109 @@
|
||||||
|
<?php
|
||||||
|
class CuriousCatBridge extends BridgeAbstract {
|
||||||
|
const NAME = 'Curious Cat Bridge';
|
||||||
|
const URI = 'https://curiouscat.me';
|
||||||
|
const DESCRIPTION = 'Returns list of newest questions and answers for a user profile';
|
||||||
|
const MAINTAINER = 'VerifiedJoseph';
|
||||||
|
const PARAMETERS = array(array(
|
||||||
|
'username' => array(
|
||||||
|
'name' => 'Username',
|
||||||
|
'type' => 'text',
|
||||||
|
'required' => true,
|
||||||
|
'exampleValue' => 'koethekoethe',
|
||||||
|
)
|
||||||
|
));
|
||||||
|
|
||||||
|
const CACHE_TIMEOUT = 3600;
|
||||||
|
|
||||||
|
public function collectData() {
|
||||||
|
|
||||||
|
$url = self::URI . '/api/v2/profile?username=' . urlencode($this->getInput('username'));
|
||||||
|
|
||||||
|
$apiJson = getContents($url)
|
||||||
|
or returnServerError('Could not request: ' . $url);
|
||||||
|
|
||||||
|
$apiData = json_decode($apiJson, true);
|
||||||
|
|
||||||
|
foreach($apiData['posts'] as $post) {
|
||||||
|
$item = array();
|
||||||
|
|
||||||
|
$item['author'] = 'Anonymous';
|
||||||
|
|
||||||
|
if ($post['senderData']['id'] !== false) {
|
||||||
|
$item['author'] = $post['senderData']['username'];
|
||||||
|
}
|
||||||
|
|
||||||
|
$item['uri'] = $this->getURI() . '/post/' . $post['id'];
|
||||||
|
$item['title'] = $this->ellipsisTitle($post['comment']);
|
||||||
|
|
||||||
|
$item['content'] = $this->processContent($post);
|
||||||
|
$item['timestamp'] = $post['timestamp'];
|
||||||
|
|
||||||
|
$this->items[] = $item;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getURI() {
|
||||||
|
|
||||||
|
if (!is_null($this->getInput('username'))) {
|
||||||
|
return self::URI . '/' . $this->getInput('username');
|
||||||
|
}
|
||||||
|
|
||||||
|
return parent::getURI();
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getName() {
|
||||||
|
|
||||||
|
if (!is_null($this->getInput('username'))) {
|
||||||
|
return $this->getInput('username') . ' - Curious Cat';
|
||||||
|
}
|
||||||
|
|
||||||
|
return parent::getName();
|
||||||
|
}
|
||||||
|
|
||||||
|
private function processContent($post) {
|
||||||
|
|
||||||
|
$author = 'Anonymous';
|
||||||
|
|
||||||
|
if ($post['senderData']['id'] !== false) {
|
||||||
|
$authorUrl = self::URI . '/' . $post['senderData']['username'];
|
||||||
|
|
||||||
|
$author = <<<EOD
|
||||||
|
<a href="{$authorUrl}">{$post['senderData']['username']}</a>
|
||||||
|
EOD;
|
||||||
|
}
|
||||||
|
|
||||||
|
$question = $this->formatUrls($post['comment']);
|
||||||
|
$answer = $this->formatUrls($post['reply']);
|
||||||
|
|
||||||
|
$content = <<<EOD
|
||||||
|
<p>{$author} asked:</p>
|
||||||
|
<blockquote>{$question}</blockquote><br/>
|
||||||
|
<p>{$post['addresseeData']['username']} answered:</p>
|
||||||
|
<blockquote>{$answer}</blockquote>
|
||||||
|
EOD;
|
||||||
|
|
||||||
|
return $content;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function ellipsisTitle($text) {
|
||||||
|
$length = 150;
|
||||||
|
|
||||||
|
if (strlen($text) > $length) {
|
||||||
|
$text = explode('<br>', wordwrap($text, $length, '<br>'));
|
||||||
|
return $text[0] . '...';
|
||||||
|
}
|
||||||
|
|
||||||
|
return $text;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function formatUrls($content) {
|
||||||
|
|
||||||
|
return preg_replace(
|
||||||
|
'/(http[s]{0,1}\:\/\/[a-zA-Z0-9.\/\?\&=\-_]{4,})/ims',
|
||||||
|
'<a target="_blank" href="$1" target="_blank">$1</a> ',
|
||||||
|
$content
|
||||||
|
);
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
|
@ -4,7 +4,7 @@ class DailymotionBridge extends BridgeAbstract {
|
||||||
const MAINTAINER = 'mitsukarenai';
|
const MAINTAINER = 'mitsukarenai';
|
||||||
const NAME = 'Dailymotion Bridge';
|
const NAME = 'Dailymotion Bridge';
|
||||||
const URI = 'https://www.dailymotion.com/';
|
const URI = 'https://www.dailymotion.com/';
|
||||||
const CACHE_TIMEOUT = 10800; // 3h
|
const CACHE_TIMEOUT = 3600; // 1h
|
||||||
const DESCRIPTION = 'Returns the 5 newest videos by username/playlist or search';
|
const DESCRIPTION = 'Returns the 5 newest videos by username/playlist or search';
|
||||||
|
|
||||||
const PARAMETERS = array (
|
const PARAMETERS = array (
|
||||||
|
@ -27,47 +27,64 @@ class DailymotionBridge extends BridgeAbstract {
|
||||||
),
|
),
|
||||||
'pa' => array(
|
'pa' => array(
|
||||||
'name' => 'Page',
|
'name' => 'Page',
|
||||||
'type' => 'number'
|
'type' => 'number',
|
||||||
|
'defaultValue' => 1,
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
|
|
||||||
protected function getMetadata($id){
|
private $feedName = '';
|
||||||
$metadata = array();
|
|
||||||
$html2 = getSimpleHTMLDOM(self::URI . 'video/' . $id);
|
|
||||||
if(!$html2) {
|
|
||||||
return $metadata;
|
|
||||||
}
|
|
||||||
|
|
||||||
$metadata['title'] = $html2->find('meta[property=og:title]', 0)->getAttribute('content');
|
private $apiUrl = 'https://api.dailymotion.com';
|
||||||
$metadata['timestamp'] = strtotime(
|
private $apiFields = 'created_time,description,id,owner.screenname,tags,thumbnail_url,title,url';
|
||||||
$html2->find('meta[property=video:release_date]', 0)->getAttribute('content')
|
|
||||||
);
|
|
||||||
$metadata['thumbnailUri'] = $html2->find('meta[property=og:image]', 0)->getAttribute('content');
|
|
||||||
$metadata['uri'] = $html2->find('meta[property=og:url]', 0)->getAttribute('content');
|
|
||||||
return $metadata;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function getIcon() {
|
public function getIcon() {
|
||||||
return 'https://static1-ssl.dmcdn.net/images/neon/favicons/android-icon-36x36.png.vf806ca4ed0deed812';
|
return 'https://static1-ssl.dmcdn.net/images/neon/favicons/android-icon-36x36.png.vf806ca4ed0deed812';
|
||||||
}
|
}
|
||||||
|
|
||||||
public function collectData() {
|
public function collectData() {
|
||||||
$html = '';
|
|
||||||
$limit = 5;
|
if ($this->queriedContext === 'By username' || $this->queriedContext === 'By playlist id') {
|
||||||
$count = 0;
|
|
||||||
|
$apiJson = getContents($this->getApiUrl())
|
||||||
|
or returnServerError('Could not request: ' . $this->getApiUrl());
|
||||||
|
|
||||||
|
$apiData = json_decode($apiJson, true);
|
||||||
|
|
||||||
|
$this->feedName = $this->getPlaylistTitle($this->getInput('p'));
|
||||||
|
|
||||||
|
foreach ($apiData['list'] as $apiItem) {
|
||||||
|
$item = array();
|
||||||
|
|
||||||
|
$item['uri'] = $apiItem['url'];
|
||||||
|
$item['uid'] = $apiItem['id'];
|
||||||
|
$item['title'] = $apiItem['title'];
|
||||||
|
$item['timestamp'] = $apiItem['created_time'];
|
||||||
|
$item['author'] = $apiItem['owner.screenname'];
|
||||||
|
$item['content'] = '<p><a href="' . $apiItem['url'] . '">
|
||||||
|
<img src="' . $apiItem['thumbnail_url'] . '"></a></p><p>' . $apiItem['description'] . '</p>';
|
||||||
|
$item['categories'] = $apiItem['tags'];
|
||||||
|
$item['enclosures'][] = $apiItem['thumbnail_url'];
|
||||||
|
|
||||||
|
$this->items[] = $item;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($this->queriedContext === 'From search results') {
|
||||||
|
|
||||||
$html = getSimpleHTMLDOM($this->getURI())
|
$html = getSimpleHTMLDOM($this->getURI())
|
||||||
or returnServerError('Could not request Dailymotion.');
|
or returnServerError('Could not request Dailymotion.');
|
||||||
|
|
||||||
foreach($html->find('div.media a.preview_link') as $element) {
|
foreach($html->find('div.media a.preview_link') as $element) {
|
||||||
if($count < $limit) {
|
|
||||||
$item = array();
|
$item = array();
|
||||||
|
|
||||||
$item['id'] = str_replace('/video/', '', strtok($element->href, '_'));
|
$item['id'] = str_replace('/video/', '', strtok($element->href, '_'));
|
||||||
$metadata = $this->getMetadata($item['id']);
|
$metadata = $this->getMetadata($item['id']);
|
||||||
|
|
||||||
if(empty($metadata)) {
|
if(empty($metadata)) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
$item['uri'] = $metadata['uri'];
|
$item['uri'] = $metadata['uri'];
|
||||||
$item['title'] = $metadata['title'];
|
$item['title'] = $metadata['title'];
|
||||||
$item['timestamp'] = $metadata['timestamp'];
|
$item['timestamp'] = $metadata['timestamp'];
|
||||||
|
@ -83,7 +100,10 @@ class DailymotionBridge extends BridgeAbstract {
|
||||||
. '</a>';
|
. '</a>';
|
||||||
|
|
||||||
$this->items[] = $item;
|
$this->items[] = $item;
|
||||||
$count++;
|
|
||||||
|
if (count($this->items) >= 5) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -95,6 +115,11 @@ class DailymotionBridge extends BridgeAbstract {
|
||||||
break;
|
break;
|
||||||
case 'By playlist id':
|
case 'By playlist id':
|
||||||
$specific = strtok($this->getInput('p'), '_');
|
$specific = strtok($this->getInput('p'), '_');
|
||||||
|
|
||||||
|
if ($this->feedName) {
|
||||||
|
$specific = $this->feedName;
|
||||||
|
}
|
||||||
|
|
||||||
break;
|
break;
|
||||||
case 'From search results':
|
case 'From search results':
|
||||||
$specific = $this->getInput('s');
|
$specific = $this->getInput('s');
|
||||||
|
@ -102,26 +127,77 @@ class DailymotionBridge extends BridgeAbstract {
|
||||||
default: return parent::getName();
|
default: return parent::getName();
|
||||||
}
|
}
|
||||||
|
|
||||||
return $specific . ' : Dailymotion Bridge';
|
return $specific . ' : Dailymotion';
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getURI(){
|
public function getURI(){
|
||||||
$uri = self::URI;
|
$uri = self::URI;
|
||||||
switch($this->queriedContext) {
|
switch($this->queriedContext) {
|
||||||
case 'By username':
|
case 'By username':
|
||||||
$uri .= 'user/' . urlencode($this->getInput('u')) . '/1';
|
$uri .= 'user/' . urlencode($this->getInput('u'));
|
||||||
break;
|
break;
|
||||||
case 'By playlist id':
|
case 'By playlist id':
|
||||||
$uri .= 'playlist/' . urlencode(strtok($this->getInput('p'), '_'));
|
$uri .= 'playlist/' . urlencode(strtok($this->getInput('p'), '_'));
|
||||||
break;
|
break;
|
||||||
case 'From search results':
|
case 'From search results':
|
||||||
$uri .= 'search/' . urlencode($this->getInput('s'));
|
$uri .= 'search/' . urlencode($this->getInput('s'));
|
||||||
if($this->getInput('pa')) {
|
|
||||||
$uri .= '/' . $this->getInput('pa');
|
if(!is_null($this->getInput('pa'))) {
|
||||||
|
$pa = $this->getInput('pa');
|
||||||
|
|
||||||
|
if ($this->getInput('pa') < 1) {
|
||||||
|
$pa = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
$uri .= '/' . $pa;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
default: return parent::getURI();
|
default: return parent::getURI();
|
||||||
}
|
}
|
||||||
return $uri;
|
return $uri;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private function getMetadata($id) {
|
||||||
|
$metadata = array();
|
||||||
|
|
||||||
|
$html = getSimpleHTMLDOM(self::URI . 'video/' . $id);
|
||||||
|
|
||||||
|
if(!$html) {
|
||||||
|
return $metadata;
|
||||||
|
}
|
||||||
|
|
||||||
|
$metadata['title'] = $html->find('meta[property=og:title]', 0)->getAttribute('content');
|
||||||
|
$metadata['timestamp'] = strtotime(
|
||||||
|
$html->find('meta[property=video:release_date]', 0)->getAttribute('content')
|
||||||
|
);
|
||||||
|
$metadata['thumbnailUri'] = $html->find('meta[property=og:image]', 0)->getAttribute('content');
|
||||||
|
$metadata['uri'] = $html->find('meta[property=og:url]', 0)->getAttribute('content');
|
||||||
|
return $metadata;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function getPlaylistTitle($id) {
|
||||||
|
$title = '';
|
||||||
|
|
||||||
|
$url = self::URI . 'playlist/' . $id;
|
||||||
|
|
||||||
|
$html = getSimpleHTMLDOM($url)
|
||||||
|
or returnServerError('Could not request: ' . $url);
|
||||||
|
|
||||||
|
$title = $html->find('meta[property=og:title]', 0)->getAttribute('content');
|
||||||
|
return $title;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function getApiUrl() {
|
||||||
|
|
||||||
|
switch($this->queriedContext) {
|
||||||
|
case 'By username':
|
||||||
|
return $this->apiUrl . '/user/' . $this->getInput('u')
|
||||||
|
. '/videos?fields=' . urlencode($this->apiFields) . '&availability=1&sort=recent&limit=5';
|
||||||
|
break;
|
||||||
|
case 'By playlist id':
|
||||||
|
return $this->apiUrl . '/playlist/' . $this->getInput('p')
|
||||||
|
. '/videos?fields=' . urlencode($this->apiFields) . '&limit=5';
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -40,7 +40,7 @@ class DanbooruBridge extends BridgeAbstract {
|
||||||
defaultLinkTo($element, $this->getURI());
|
defaultLinkTo($element, $this->getURI());
|
||||||
|
|
||||||
$item = array();
|
$item = array();
|
||||||
$item['uri'] = $element->find('a', 0)->href;
|
$item['uri'] = html_entity_decode($element->find('a', 0)->href);
|
||||||
$item['postid'] = (int)preg_replace('/[^0-9]/', '', $element->getAttribute(static::IDATTRIBUTE));
|
$item['postid'] = (int)preg_replace('/[^0-9]/', '', $element->getAttribute(static::IDATTRIBUTE));
|
||||||
$item['timestamp'] = time();
|
$item['timestamp'] = time();
|
||||||
$thumbnailUri = $element->find('img', 0)->src;
|
$thumbnailUri = $element->find('img', 0)->src;
|
||||||
|
|
79
bridges/DarkReadingBridge.php
Normal file
79
bridges/DarkReadingBridge.php
Normal file
|
@ -0,0 +1,79 @@
|
||||||
|
<?php
|
||||||
|
class DarkReadingBridge extends FeedExpander {
|
||||||
|
const MAINTAINER = 'ORelio';
|
||||||
|
const NAME = 'Dark Reading Bridge';
|
||||||
|
const URI = 'https://www.darkreading.com/';
|
||||||
|
const DESCRIPTION = 'Returns the newest articles from Dark Reading';
|
||||||
|
|
||||||
|
const PARAMETERS = array( array(
|
||||||
|
'feed' => array(
|
||||||
|
'name' => 'Feed',
|
||||||
|
'type' => 'list',
|
||||||
|
'values' => array(
|
||||||
|
'All Dark Reading Stories' => '000_AllArticles',
|
||||||
|
'Attacks/Breaches' => '644_Attacks/Breaches',
|
||||||
|
'Application Security' => '645_Application%20Security',
|
||||||
|
'Database Security' => '646_Database%20Security',
|
||||||
|
'Cloud' => '647_Cloud',
|
||||||
|
'Endpoint' => '648_Endpoint',
|
||||||
|
'Authentication' => '649_Authentication',
|
||||||
|
'Privacy' => '650_Privacy',
|
||||||
|
'Mobile' => '651_Mobile',
|
||||||
|
'Perimeter' => '652_Perimeter',
|
||||||
|
'Risk' => '653_Risk',
|
||||||
|
'Compliance' => '654_Compliance',
|
||||||
|
'Operations' => '655_Operations',
|
||||||
|
'Careers and People' => '656_Careers%20and%20People',
|
||||||
|
'Identity and Access Management' => '657_Identity%20and%20Access%20Management',
|
||||||
|
'Analytics' => '658_Analytics',
|
||||||
|
'Threat Intelligence' => '659_Threat%20Intelligence',
|
||||||
|
'Security Monitoring' => '660_Security%20Monitoring',
|
||||||
|
'Vulnerabilities / Threats' => '661_Vulnerabilities%20/%20Threats',
|
||||||
|
'Advanced Threats' => '662_Advanced%20Threats',
|
||||||
|
'Insider Threats' => '663_Insider%20Threats',
|
||||||
|
'Vulnerability Management' => '664_Vulnerability%20Management',
|
||||||
|
)
|
||||||
|
)
|
||||||
|
));
|
||||||
|
|
||||||
|
public function collectData(){
|
||||||
|
$feed = $this->getInput('feed');
|
||||||
|
$feed_splitted = explode('_', $feed);
|
||||||
|
$feed_id = $feed_splitted[0];
|
||||||
|
$feed_name = $feed_splitted[1];
|
||||||
|
if(empty($feed) || !ctype_digit($feed_id) || !preg_match('/[A-Za-z%20\/]/', $feed_name)) {
|
||||||
|
returnClientError('Invalid feed, please check the "feed" parameter.');
|
||||||
|
}
|
||||||
|
$feed_url = $this->getURI() . 'rss_simple.asp';
|
||||||
|
if ($feed_id != '000') {
|
||||||
|
$feed_url .= '?f_n=' . $feed_id . '&f_ln=' . $feed_name;
|
||||||
|
}
|
||||||
|
$this->collectExpandableDatas($feed_url);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function parseItem($newsItem){
|
||||||
|
$item = parent::parseItem($newsItem);
|
||||||
|
$article = getSimpleHTMLDOMCached($item['uri'])
|
||||||
|
or returnServerError('Could not request Dark Reading: ' . $item['uri']);
|
||||||
|
$item['content'] = $this->extractArticleContent($article);
|
||||||
|
$item['enclosures'] = array(); //remove author profile picture
|
||||||
|
return $item;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function extractArticleContent($article){
|
||||||
|
$content = $article->find('div#article-main', 0)->innertext;
|
||||||
|
|
||||||
|
foreach (array(
|
||||||
|
'<div class="divsplitter',
|
||||||
|
'<div style="float: left; margin-right: 2px;',
|
||||||
|
'<div class="more-insights',
|
||||||
|
'<div id="more-insights',
|
||||||
|
) as $div_start) {
|
||||||
|
$content = stripRecursiveHTMLSection($content, 'div', $div_start);
|
||||||
|
}
|
||||||
|
|
||||||
|
$content = stripWithDelimiters($content, '<h1 ', '</h1>');
|
||||||
|
|
||||||
|
return $content;
|
||||||
|
}
|
||||||
|
}
|
27
bridges/DavesTrailerPageBridge.php
Normal file
27
bridges/DavesTrailerPageBridge.php
Normal file
|
@ -0,0 +1,27 @@
|
||||||
|
<?php
|
||||||
|
class DavesTrailerPageBridge extends BridgeAbstract {
|
||||||
|
const MAINTAINER = 'johnnygroovy';
|
||||||
|
const NAME = 'Daves Trailer Page Bridge';
|
||||||
|
const URI = 'https://www.davestrailerpage.co.uk/';
|
||||||
|
const DESCRIPTION = 'Last trailers in HD thanks to Dave.';
|
||||||
|
|
||||||
|
public function collectData(){
|
||||||
|
$html = getSimpleHTMLDOM(static::URI)
|
||||||
|
or returnClientError('No results for this query.');
|
||||||
|
|
||||||
|
foreach ($html->find('tr[!align]') as $tr) {
|
||||||
|
$item = array();
|
||||||
|
|
||||||
|
// title
|
||||||
|
$item['title'] = $tr->find('td', 0)->find('b', 0)->plaintext;
|
||||||
|
|
||||||
|
// content
|
||||||
|
$item['content'] = $tr->find('ul', 1);
|
||||||
|
|
||||||
|
// uri
|
||||||
|
$item['uri'] = $tr->find('a', 3)->getAttribute('href');
|
||||||
|
|
||||||
|
$this->items[] = $item;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -1145,7 +1145,7 @@ class PepperBridgeAbstract extends BridgeAbstract {
|
||||||
} else {
|
} else {
|
||||||
foreach ($list as $deal) {
|
foreach ($list as $deal) {
|
||||||
$item = array();
|
$item = array();
|
||||||
$item['uri'] = $deal->find('div[class=threadGrid-title]', 0)->find('a', 0)->href;
|
$item['uri'] = $deal->find('div[class*=threadGrid-title]', 0)->find('a', 0)->href;
|
||||||
$item['title'] = $deal->find('a[class*=' . $selectorLink . ']', 0
|
$item['title'] = $deal->find('a[class*=' . $selectorLink . ']', 0
|
||||||
)->plaintext;
|
)->plaintext;
|
||||||
$item['author'] = $deal->find('span.thread-username', 0)->plaintext;
|
$item['author'] = $deal->find('span.thread-username', 0)->plaintext;
|
||||||
|
|
|
@ -1,169 +0,0 @@
|
||||||
<?php
|
|
||||||
class DemonoidBridge extends BridgeAbstract {
|
|
||||||
|
|
||||||
const MAINTAINER = 'metaMMA';
|
|
||||||
const NAME = 'Demonoid';
|
|
||||||
const URI = 'https://www.demonoid.pw/';
|
|
||||||
const DESCRIPTION = 'Returns results from search';
|
|
||||||
|
|
||||||
const PARAMETERS = array(
|
|
||||||
'Keywords' => array(
|
|
||||||
'q' => array(
|
|
||||||
'name' => 'keywords',
|
|
||||||
'exampleValue' => 'keyword1 keyword2…',
|
|
||||||
'required' => true,
|
|
||||||
),
|
|
||||||
'category' => array(
|
|
||||||
'name' => 'Category',
|
|
||||||
'type' => 'list',
|
|
||||||
'values' => array(
|
|
||||||
'All' => 0,
|
|
||||||
'Movies' => 1,
|
|
||||||
'Music' => 2,
|
|
||||||
'TV' => 3,
|
|
||||||
'Games' => 4,
|
|
||||||
'Applications' => 5,
|
|
||||||
'Pictures' => 8,
|
|
||||||
'Anime' => 9,
|
|
||||||
'Comics' => 10,
|
|
||||||
'Books' => 11,
|
|
||||||
'Audiobooks' => 17
|
|
||||||
)
|
|
||||||
)
|
|
||||||
),
|
|
||||||
'Category Only' => array(
|
|
||||||
'catOnly' => array(
|
|
||||||
'name' => 'Category',
|
|
||||||
'type' => 'list',
|
|
||||||
'values' => array(
|
|
||||||
'All' => 0,
|
|
||||||
'Movies' => 1,
|
|
||||||
'Music' => 2,
|
|
||||||
'TV' => 3,
|
|
||||||
'Games' => 4,
|
|
||||||
'Applications' => 5,
|
|
||||||
'Pictures' => 8,
|
|
||||||
'Anime' => 9,
|
|
||||||
'Comics' => 10,
|
|
||||||
'Books' => 11,
|
|
||||||
'Audiobooks' => 17
|
|
||||||
)
|
|
||||||
)
|
|
||||||
),
|
|
||||||
'User ID' => array(
|
|
||||||
'userid' => array(
|
|
||||||
'name' => 'user id',
|
|
||||||
'exampleValue' => '00000',
|
|
||||||
'required' => true,
|
|
||||||
'type' => 'number'
|
|
||||||
),
|
|
||||||
'category' => array(
|
|
||||||
'name' => 'Category',
|
|
||||||
'type' => 'list',
|
|
||||||
'values' => array(
|
|
||||||
'All' => 0,
|
|
||||||
'Movies' => 1,
|
|
||||||
'Music' => 2,
|
|
||||||
'TV' => 3,
|
|
||||||
'Games' => 4,
|
|
||||||
'Applications' => 5,
|
|
||||||
'Pictures' => 8,
|
|
||||||
'Anime' => 9,
|
|
||||||
'Comics' => 10,
|
|
||||||
'Books' => 11,
|
|
||||||
'Audiobooks' => 17
|
|
||||||
)
|
|
||||||
)
|
|
||||||
)
|
|
||||||
);
|
|
||||||
|
|
||||||
public function collectData() {
|
|
||||||
|
|
||||||
if(!empty($this->getInput('q'))) {
|
|
||||||
|
|
||||||
$html = getSimpleHTMLDOM(
|
|
||||||
self::URI .
|
|
||||||
'files/?category=' .
|
|
||||||
rawurlencode($this->getInput('category')) .
|
|
||||||
'&subcategory=All&quality=All&seeded=2&external=2&query=' .
|
|
||||||
urlencode($this->getInput('q')) .
|
|
||||||
'&uid=0&sort='
|
|
||||||
) or returnServerError('Could not request Demonoid.');
|
|
||||||
|
|
||||||
} elseif(!empty($this->getInput('catOnly'))) {
|
|
||||||
|
|
||||||
$html = getSimpleHTMLDOM(
|
|
||||||
self::URI .
|
|
||||||
'files/?uid=0&category=' .
|
|
||||||
rawurlencode($this->getInput('catOnly')) .
|
|
||||||
'&subcategory=0&language=0&seeded=2&quality=0&query=&sort='
|
|
||||||
) or returnServerError('Could not request Demonoid.');
|
|
||||||
|
|
||||||
} elseif(!empty($this->getInput('userid'))) {
|
|
||||||
|
|
||||||
$html = getSimpleHTMLDOM(
|
|
||||||
self::URI .
|
|
||||||
'files/?uid=' .
|
|
||||||
rawurlencode($this->getInput('userid')) .
|
|
||||||
'&seeded=2'
|
|
||||||
) or returnServerError('Could not request Demonoid.');
|
|
||||||
|
|
||||||
} else {
|
|
||||||
returnServerError('Invalid parameters !');
|
|
||||||
}
|
|
||||||
|
|
||||||
if(preg_match('~No torrents found~', $html)) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
$table = $html->find('td[class=ctable_content_no_pad]', 0);
|
|
||||||
$cursorCount = 4;
|
|
||||||
$elementCount = 0;
|
|
||||||
while($elementCount != 40) {
|
|
||||||
$elementCount++;
|
|
||||||
$currentElement = $table->find('tr', $cursorCount);
|
|
||||||
if(preg_match('~items total~', $currentElement)) {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
$item = array();
|
|
||||||
//Do we have a date ?
|
|
||||||
if(preg_match('~Added.*?(.*)~', $currentElement->plaintext, $dateStr)) {
|
|
||||||
if(preg_match('~today~', $dateStr[0])) {
|
|
||||||
date_default_timezone_set('UTC');
|
|
||||||
$timestamp = mktime(0, 0, 0, gmdate('n'), gmdate('j'), gmdate('Y'));
|
|
||||||
} else {
|
|
||||||
preg_match('~(?<=ed on ).*\d+~', $currentElement->plaintext, $fullDateStr);
|
|
||||||
date_default_timezone_set('UTC');
|
|
||||||
$dateObj = strptime($fullDateStr[0], '%A, %b %d, %Y');
|
|
||||||
$timestamp = mktime(0, 0, 0, $dateObj['tm_mon'] + 1, $dateObj['tm_mday'], 1900 + $dateObj['tm_year']);
|
|
||||||
}
|
|
||||||
$cursorCount++;
|
|
||||||
}
|
|
||||||
|
|
||||||
$content = $table->find('tr', $cursorCount)->find('a', 1);
|
|
||||||
$cursorCount++;
|
|
||||||
$torrentInfo = $table->find('tr', $cursorCount);
|
|
||||||
$item['timestamp'] = $timestamp;
|
|
||||||
$item['title'] = $content->plaintext;
|
|
||||||
$item['id'] = self::URI . $content->href;
|
|
||||||
$item['uri'] = self::URI . $content->href;
|
|
||||||
$item['author'] = $torrentInfo->find('a[class=user]', 0)->plaintext;
|
|
||||||
$item['seeders'] = $torrentInfo->find('font[class=green]', 0)->plaintext;
|
|
||||||
$item['leechers'] = $torrentInfo->find('font[class=red]', 0)->plaintext;
|
|
||||||
$item['size'] = $torrentInfo->find('td', 3)->plaintext;
|
|
||||||
$item['content'] = 'Uploaded by ' . $item['author']
|
|
||||||
. ' , Size ' . $item['size']
|
|
||||||
. '<br>seeders: '
|
|
||||||
. $item['seeders']
|
|
||||||
. ' | leechers: '
|
|
||||||
. $item['leechers']
|
|
||||||
. '<br><a href="'
|
|
||||||
. $item['id']
|
|
||||||
. '">info page</a>';
|
|
||||||
|
|
||||||
$this->items[] = $item;
|
|
||||||
|
|
||||||
$cursorCount++;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -116,6 +116,12 @@ class DesoutterBridge extends BridgeAbstract {
|
||||||
'name' => 'Load full articles',
|
'name' => 'Load full articles',
|
||||||
'type' => 'checkbox',
|
'type' => 'checkbox',
|
||||||
'title' => 'Enable to load the full article for each item'
|
'title' => 'Enable to load the full article for each item'
|
||||||
|
),
|
||||||
|
'limit' => array(
|
||||||
|
'name' => 'Limit',
|
||||||
|
'type' => 'number',
|
||||||
|
'defaultValue' => 3,
|
||||||
|
'title' => "Maximum number of items to return in the feed.\n0 = unlimited"
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
|
@ -156,19 +162,23 @@ class DesoutterBridge extends BridgeAbstract {
|
||||||
|
|
||||||
$this->title = html_entity_decode($html->find('title', 0)->plaintext, ENT_QUOTES);
|
$this->title = html_entity_decode($html->find('title', 0)->plaintext, ENT_QUOTES);
|
||||||
|
|
||||||
|
$limit = $this->getInput('limit') ?: 0;
|
||||||
|
|
||||||
foreach($html->find('article') as $article) {
|
foreach($html->find('article') as $article) {
|
||||||
$item = array();
|
$item = array();
|
||||||
|
|
||||||
$item['uri'] = $article->find('[itemprop="name"]', 0)->href;
|
$item['uri'] = $article->find('a', 0)->href;
|
||||||
$item['title'] = $article->find('[itemprop="name"]', 0)->title;
|
$item['title'] = $article->find('a[title]', 0)->title;
|
||||||
|
|
||||||
if($this->getInput('full')) {
|
if($this->getInput('full')) {
|
||||||
$item['content'] = $this->getFullNewsArticle($item['uri']);
|
$item['content'] = $this->getFullNewsArticle($item['uri']);
|
||||||
} else {
|
} else {
|
||||||
$item['content'] = $article->find('[itemprop="description"]', 0)->plaintext;
|
$item['content'] = $article->find('div.tile-body p', 0)->plaintext;
|
||||||
}
|
}
|
||||||
|
|
||||||
$this->items[] = $item;
|
$this->items[] = $item;
|
||||||
|
|
||||||
|
if ($limit > 0 && count($this->items) >= $limit) break;
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
60
bridges/DiarioDoAlentejoBridge.php
Normal file
60
bridges/DiarioDoAlentejoBridge.php
Normal file
|
@ -0,0 +1,60 @@
|
||||||
|
<?php
|
||||||
|
class DiarioDoAlentejoBridge extends BridgeAbstract {
|
||||||
|
const MAINTAINER = 'somini';
|
||||||
|
const NAME = 'Diário do Alentejo';
|
||||||
|
const URI = 'https://www.diariodoalentejo.pt';
|
||||||
|
const DESCRIPTION = 'Semanário Regionalista Independente';
|
||||||
|
const CACHE_TIMEOUT = 28800; // 8h
|
||||||
|
|
||||||
|
/* This is used to hack around obtaining a timestamp. It's just a list of Month names in Portuguese ... */
|
||||||
|
const PT_MONTH_NAMES = array(
|
||||||
|
'janeiro',
|
||||||
|
'fevereiro',
|
||||||
|
'março',
|
||||||
|
'abril',
|
||||||
|
'maio',
|
||||||
|
'junho',
|
||||||
|
'julho',
|
||||||
|
'agosto',
|
||||||
|
'setembro',
|
||||||
|
'outubro',
|
||||||
|
'novembro',
|
||||||
|
'dezembro');
|
||||||
|
|
||||||
|
public function getIcon() {
|
||||||
|
return 'https://www.diariodoalentejo.pt/images/favicon/apple-touch-icon.png';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function collectData(){
|
||||||
|
/* This is slow as molasses (>30s!), keep the cache timeout high to avoid killing the host */
|
||||||
|
$html = getSimpleHTMLDOMCached($this->getURI() . '/pt/noticias-listagem.aspx')
|
||||||
|
or returnServerError('Could not load content');
|
||||||
|
|
||||||
|
foreach($html->find('.list_news .item') as $element) {
|
||||||
|
$item = array();
|
||||||
|
|
||||||
|
$item_link = $element->find('.body h2.title a', 0);
|
||||||
|
/* Another broken URL, see also `bridges/ComboiosDePortugalBridge.php` */
|
||||||
|
$item['uri'] = self::URI . implode('/', array_map('urlencode', explode('/', $item_link->href)));
|
||||||
|
$item['title'] = $item_link->innertext;
|
||||||
|
|
||||||
|
$item['timestamp'] = str_ireplace(
|
||||||
|
array_map(function($name) { return ' ' . $name . ' '; }, self::PT_MONTH_NAMES),
|
||||||
|
array_map(function($num) { return sprintf('-%02d-', $num); }, range(1, sizeof(self::PT_MONTH_NAMES))),
|
||||||
|
$element->find('span.date', 0)->innertext);
|
||||||
|
|
||||||
|
/* Fix the Image URL */
|
||||||
|
$item_image = $element->find('img.thumb', 0);
|
||||||
|
$item_image->src = preg_replace('/.*&img=([^&]+).*/', '\1', $item_image->getAttribute('data-src'));
|
||||||
|
|
||||||
|
/* Content: */
|
||||||
|
/* - Image */
|
||||||
|
/* - Category */
|
||||||
|
$content = $item_image .
|
||||||
|
'<center>' . $element->find('a.category', 0) . '</center>';
|
||||||
|
$item['content'] = defaultLinkTo($content, self::URI);
|
||||||
|
|
||||||
|
$this->items[] = $item;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,9 +0,0 @@
|
||||||
<?php
|
|
||||||
require_once('Shimmie2Bridge.php');
|
|
||||||
|
|
||||||
class DollbooruBridge extends Shimmie2Bridge {
|
|
||||||
const MAINTAINER = 'mitsukarenai';
|
|
||||||
const NAME = 'Dollbooru';
|
|
||||||
const URI = 'http://dollbooru.org/';
|
|
||||||
const DESCRIPTION = 'Returns images from given page';
|
|
||||||
}
|
|
6195
bridges/DownDetectorBridge.php
Normal file
6195
bridges/DownDetectorBridge.php
Normal file
File diff suppressed because it is too large
Load diff
|
@ -19,7 +19,7 @@ favicon-63b2904a073c89b52b19aa08cebc16a154bcf83fee8ecc6439968b1e6db569c7.ico';
|
||||||
$json = $this->loadEmbeddedJsonData($html);
|
$json = $this->loadEmbeddedJsonData($html);
|
||||||
|
|
||||||
foreach($html->find('li[id^="screenshot-"]') as $shot) {
|
foreach($html->find('li[id^="screenshot-"]') as $shot) {
|
||||||
$item = [];
|
$item = array();
|
||||||
|
|
||||||
$additional_data = $this->findJsonForShot($shot, $json);
|
$additional_data = $this->findJsonForShot($shot, $json);
|
||||||
if ($additional_data === null) {
|
if ($additional_data === null) {
|
||||||
|
@ -38,14 +38,14 @@ favicon-63b2904a073c89b52b19aa08cebc16a154bcf83fee8ecc6439968b1e6db569c7.ico';
|
||||||
|
|
||||||
$preview_path = $shot->find('picture source', 0)->attr['srcset'];
|
$preview_path = $shot->find('picture source', 0)->attr['srcset'];
|
||||||
$item['content'] .= $this->getImageTag($preview_path, $item['title']);
|
$item['content'] .= $this->getImageTag($preview_path, $item['title']);
|
||||||
$item['enclosures'] = [$this->getFullSizeImagePath($preview_path)];
|
$item['enclosures'] = array($this->getFullSizeImagePath($preview_path));
|
||||||
|
|
||||||
$this->items[] = $item;
|
$this->items[] = $item;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private function loadEmbeddedJsonData($html){
|
private function loadEmbeddedJsonData($html){
|
||||||
$json = [];
|
$json = array();
|
||||||
$scripts = $html->find('script');
|
$scripts = $html->find('script');
|
||||||
|
|
||||||
foreach($scripts as $script) {
|
foreach($scripts as $script) {
|
||||||
|
|
|
@ -40,7 +40,7 @@ class EconomistBridge extends BridgeAbstract {
|
||||||
if ($nextprev)
|
if ($nextprev)
|
||||||
$nextprev->outertext = '';
|
$nextprev->outertext = '';
|
||||||
|
|
||||||
$section = [ $article->find('h3[itemprop="articleSection"]', 0)->plaintext ];
|
$section = array( $article->find('h3[itemprop="articleSection"]', 0)->plaintext );
|
||||||
|
|
||||||
$item = array();
|
$item = array();
|
||||||
$item['title'] = $header->find('span', 0)->innertext . ': '
|
$item['title'] = $header->find('span', 0)->innertext . ': '
|
||||||
|
|
|
@ -47,5 +47,8 @@ class EliteDangerousGalnetBridge extends BridgeAbstract {
|
||||||
|
|
||||||
$this->items[] = $item;
|
$this->items[] = $item;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//Remove duplicates that sometimes show up on the website
|
||||||
|
$this->items = array_unique($this->items, SORT_REGULAR);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -95,7 +95,7 @@ class ElloBridge extends BridgeAbstract {
|
||||||
|
|
||||||
private function getEnclosures($post, $postData) {
|
private function getEnclosures($post, $postData) {
|
||||||
|
|
||||||
$assets = [];
|
$assets = array();
|
||||||
foreach($post->links->assets as $asset) {
|
foreach($post->links->assets as $asset) {
|
||||||
foreach($postData->linked->assets as $assetLink) {
|
foreach($postData->linked->assets as $assetLink) {
|
||||||
if($asset == $assetLink->id) {
|
if($asset == $assetLink->id) {
|
||||||
|
@ -120,9 +120,11 @@ class ElloBridge extends BridgeAbstract {
|
||||||
}
|
}
|
||||||
|
|
||||||
private function getAPIKey() {
|
private function getAPIKey() {
|
||||||
$cache = Cache::create(Configuration::getConfig('cache', 'type'));
|
$cacheFac = new CacheFactory();
|
||||||
|
$cacheFac->setWorkingDir(PATH_LIB_CACHES);
|
||||||
|
$cache = $cacheFac->create(Configuration::getConfig('cache', 'type'));
|
||||||
$cache->setScope(get_called_class());
|
$cache->setScope(get_called_class());
|
||||||
$cache->setKey(['key']);
|
$cache->setKey(array('key'));
|
||||||
$key = $cache->loadData();
|
$key = $cache->loadData();
|
||||||
|
|
||||||
if($key == null) {
|
if($key == null) {
|
||||||
|
|
|
@ -3,7 +3,7 @@ class ElsevierBridge extends BridgeAbstract {
|
||||||
|
|
||||||
const MAINTAINER = 'Pierre Mazière';
|
const MAINTAINER = 'Pierre Mazière';
|
||||||
const NAME = 'Elsevier journals recent articles';
|
const NAME = 'Elsevier journals recent articles';
|
||||||
const URI = 'http://www.journals.elsevier.com/';
|
const URI = 'https://www.journals.elsevier.com/';
|
||||||
const CACHE_TIMEOUT = 43200; //12h
|
const CACHE_TIMEOUT = 43200; //12h
|
||||||
const DESCRIPTION = 'Returns the recent articles published in Elsevier journals';
|
const DESCRIPTION = 'Returns the recent articles published in Elsevier journals';
|
||||||
|
|
||||||
|
|
26
bridges/EngadgetBridge.php
Normal file
26
bridges/EngadgetBridge.php
Normal file
|
@ -0,0 +1,26 @@
|
||||||
|
<?php
|
||||||
|
class EngadgetBridge extends FeedExpander {
|
||||||
|
|
||||||
|
const MAINTAINER = 'IceWreck';
|
||||||
|
const NAME = 'Engadget Bridge';
|
||||||
|
const URI = 'https://www.engadget.com/';
|
||||||
|
const CACHE_TIMEOUT = 3600;
|
||||||
|
const DESCRIPTION = 'Article content for Engadget.';
|
||||||
|
|
||||||
|
public function collectData(){
|
||||||
|
$this->collectExpandableDatas(static::URI . 'rss.xml', 15);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function parseItem($newsItem){
|
||||||
|
$item = parent::parseItem($newsItem);
|
||||||
|
// $articlePage gets the entire page's contents
|
||||||
|
$articlePage = getSimpleHTMLDOM($newsItem->link);
|
||||||
|
// figure contain's the main article image
|
||||||
|
$article = $articlePage->find('figure', 0);
|
||||||
|
// .article-text has the actual article
|
||||||
|
foreach($articlePage->find('.article-text') as $element)
|
||||||
|
$article = $article . $element;
|
||||||
|
$item['content'] = $article;
|
||||||
|
return $item;
|
||||||
|
}
|
||||||
|
}
|
70
bridges/EsquerdaNetBridge.php
Normal file
70
bridges/EsquerdaNetBridge.php
Normal file
|
@ -0,0 +1,70 @@
|
||||||
|
<?php
|
||||||
|
class EsquerdaNetBridge extends FeedExpander {
|
||||||
|
const MAINTAINER = 'somini';
|
||||||
|
const NAME = 'Esquerda.net';
|
||||||
|
const URI = 'https://www.esquerda.net';
|
||||||
|
const DESCRIPTION = 'Esquerda.net';
|
||||||
|
const PARAMETERS = array(
|
||||||
|
array(
|
||||||
|
'feed' => array(
|
||||||
|
'name' => 'Feed',
|
||||||
|
'type' => 'list',
|
||||||
|
'defaultValue' => 'Geral',
|
||||||
|
'values' => array(
|
||||||
|
'Geral' => 'geral',
|
||||||
|
'Dossier' => 'artigos-dossier',
|
||||||
|
'Vídeo' => 'video',
|
||||||
|
'Opinião' => 'opinioes',
|
||||||
|
'Rádio' => 'radio',
|
||||||
|
)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
public function getURI() {
|
||||||
|
$type = $this->getInput('feed');
|
||||||
|
return self::URI . '/rss/' . $type;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getIcon() {
|
||||||
|
return 'https://www.esquerda.net/sites/default/files/favicon_0.ico';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function collectData(){
|
||||||
|
parent::collectExpandableDatas($this->getURI());
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function parseItem($newsItem){
|
||||||
|
# Fix Publish date
|
||||||
|
$badDate = $newsItem->pubDate;
|
||||||
|
preg_match('|(?P<day>\d\d)/(?P<month>\d\d)/(?P<year>\d\d\d\d) - (?P<hour>\d\d):(?P<minute>\d\d)|', $badDate, $d);
|
||||||
|
$newsItem->pubDate = sprintf('%s-%s-%sT%s:%s', $d['year'], $d['month'], $d['day'], $d['hour'], $d['minute']);
|
||||||
|
$item = parent::parseItem($newsItem);
|
||||||
|
# Include all the content
|
||||||
|
$uri = $item['uri'];
|
||||||
|
$html = getSimpleHTMLDOMCached($uri)
|
||||||
|
or returnServerError('Could not load content for ' . $uri);
|
||||||
|
$content = $html->find('div#content div.content', 0);
|
||||||
|
## Fix author
|
||||||
|
$authorHTML = $html->find('.field-name-field-op-author a', 0);
|
||||||
|
if ($authorHTML) {
|
||||||
|
$item['author'] = $authorHTML->innertext;
|
||||||
|
$authorHTML->remove();
|
||||||
|
}
|
||||||
|
## Remove crap
|
||||||
|
$content->find('.field-name-addtoany', 0)->remove();
|
||||||
|
## Fix links
|
||||||
|
$content = defaultLinkTo($content, self::URI);
|
||||||
|
## Fix Images
|
||||||
|
foreach($content->find('img') as $img) {
|
||||||
|
$altSrc = $img->getAttribute('data-src');
|
||||||
|
if ($altSrc) {
|
||||||
|
$img->setAttribute('src', $altSrc);
|
||||||
|
}
|
||||||
|
$img->width = null;
|
||||||
|
$img->height = null;
|
||||||
|
}
|
||||||
|
$item['content'] = $content;
|
||||||
|
return $item;
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,7 +1,7 @@
|
||||||
<?php
|
<?php
|
||||||
class ExtremeDownloadBridge extends BridgeAbstract {
|
class ExtremeDownloadBridge extends BridgeAbstract {
|
||||||
const NAME = 'Extreme Download';
|
const NAME = 'Extreme Download';
|
||||||
const URI = 'https://ww1.extreme-d0wn.com/';
|
const URI = 'https://wvw.extreme-down.xyz/';
|
||||||
const DESCRIPTION = 'Suivi de série sur Extreme Download';
|
const DESCRIPTION = 'Suivi de série sur Extreme Download';
|
||||||
const MAINTAINER = 'sysadminstory';
|
const MAINTAINER = 'sysadminstory';
|
||||||
const PARAMETERS = array(
|
const PARAMETERS = array(
|
||||||
|
|
|
@ -2,7 +2,7 @@
|
||||||
class FB2Bridge extends BridgeAbstract {
|
class FB2Bridge extends BridgeAbstract {
|
||||||
|
|
||||||
const MAINTAINER = 'teromene';
|
const MAINTAINER = 'teromene';
|
||||||
const NAME = 'Facebook Alternate';
|
const NAME = 'Facebook Bridge | Touch Site';
|
||||||
const URI = 'https://www.facebook.com/';
|
const URI = 'https://www.facebook.com/';
|
||||||
const CACHE_TIMEOUT = 1000;
|
const CACHE_TIMEOUT = 1000;
|
||||||
const DESCRIPTION = 'Input a page title or a profile log. For a profile log,
|
const DESCRIPTION = 'Input a page title or a profile log. For a profile log,
|
||||||
|
@ -12,7 +12,12 @@ class FB2Bridge extends BridgeAbstract {
|
||||||
'u' => array(
|
'u' => array(
|
||||||
'name' => 'Username',
|
'name' => 'Username',
|
||||||
'required' => true
|
'required' => true
|
||||||
)
|
),
|
||||||
|
'abbrev_name' => array(
|
||||||
|
'name' => 'Abbreviate author name in title',
|
||||||
|
'type' => 'checkbox',
|
||||||
|
'defaultValue' => true,
|
||||||
|
),
|
||||||
));
|
));
|
||||||
|
|
||||||
public function getIcon() {
|
public function getIcon() {
|
||||||
|
@ -102,12 +107,12 @@ EOD
|
||||||
else
|
else
|
||||||
$timestamp = 0;
|
$timestamp = 0;
|
||||||
|
|
||||||
$item['uri'] = html_entity_decode('http://touch.facebook.com'
|
$item['uri'] = html_entity_decode('https://touch.facebook.com'
|
||||||
. $content->find("div[class='_52jc _5qc4 _78cz _24u0 _36xo']", 0)->find('a', 0)->getAttribute('href'), ENT_QUOTES);
|
. $content->find("div[class='_52jc _5qc4 _78cz _24u0 _36xo']", 0)->find('a', 0)->getAttribute('href'), ENT_QUOTES);
|
||||||
|
|
||||||
//Decode images
|
//Decode images
|
||||||
$imagecleaned = preg_replace_callback('/<i [^>]* style="[^"]*url\(\'(.*?)\'\).*?><\/i>/m', function ($matches) {
|
$imagecleaned = preg_replace_callback('/<i [^>]* style="[^"]*url\(\'(.*?)\'\).*?><\/i>/m', function ($matches) {
|
||||||
return "<img src='" . str_replace(['\\3a ', '\\3d ', '\\26 '], [':', '=', '&'], $matches[1]) . "' />";
|
return "<img src='" . str_replace(array('\\3a ', '\\3d ', '\\26 '), array(':', '=', '&'), $matches[1]) . "' />";
|
||||||
}, $content);
|
}, $content);
|
||||||
$content = str_get_html($imagecleaned);
|
$content = str_get_html($imagecleaned);
|
||||||
|
|
||||||
|
@ -159,7 +164,11 @@ EOD
|
||||||
$content = preg_replace('/<img src=\'.*?safe_image\.php.*?\' \/>/m', '', $content);
|
$content = preg_replace('/<img src=\'.*?safe_image\.php.*?\' \/>/m', '', $content);
|
||||||
|
|
||||||
//Remove the double section tags
|
//Remove the double section tags
|
||||||
$content = str_replace(['<section><section>', '</section></section>'], ['<section>', '</section>'], $content);
|
$content = str_replace(
|
||||||
|
array('<section><section>', '</section></section>'),
|
||||||
|
array('<section>', '</section>'),
|
||||||
|
$content
|
||||||
|
);
|
||||||
|
|
||||||
//Move the section tag link upper, if it is down
|
//Move the section tag link upper, if it is down
|
||||||
$content = str_get_html($content);
|
$content = str_get_html($content);
|
||||||
|
@ -182,8 +191,10 @@ EOD
|
||||||
$item['content'] = html_entity_decode($content, ENT_QUOTES);
|
$item['content'] = html_entity_decode($content, ENT_QUOTES);
|
||||||
|
|
||||||
$title = $author;
|
$title = $author;
|
||||||
|
if ($this->getInput('abbrev_name') === true) {
|
||||||
if (strlen($title) > 24)
|
if (strlen($title) > 24)
|
||||||
$title = substr($title, 0, strpos(wordwrap($title, 24), "\n")) . '...';
|
$title = substr($title, 0, strpos(wordwrap($title, 24), "\n")) . '...';
|
||||||
|
}
|
||||||
$title = $title . ' | ' . strip_tags($content);
|
$title = $title . ' | ' . strip_tags($content);
|
||||||
if (strlen($title) > 64)
|
if (strlen($title) > 64)
|
||||||
$title = substr($title, 0, strpos(wordwrap($title, 64), "\n")) . '...';
|
$title = substr($title, 0, strpos(wordwrap($title, 64), "\n")) . '...';
|
||||||
|
@ -281,10 +292,20 @@ EOD
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getName(){
|
public function getName(){
|
||||||
return (isset($this->name) ? $this->name . ' - ' : '') . 'Facebook Bridge';
|
$username = $this->getInput('u');
|
||||||
|
if (isset($username)) {
|
||||||
|
return $this->getInput('u') . ' | Facebook';
|
||||||
|
} else {
|
||||||
|
return self::NAME;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getURI(){
|
public function getURI(){
|
||||||
return 'http://facebook.com';
|
$username = $this->getInput('u');
|
||||||
|
if (isset($username)) {
|
||||||
|
return 'https://facebook.com/' . $this->getInput('u') . '/posts';
|
||||||
|
} else {
|
||||||
|
return self::URI;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
36
bridges/FabriceBellardBridge.php
Normal file
36
bridges/FabriceBellardBridge.php
Normal file
|
@ -0,0 +1,36 @@
|
||||||
|
<?php
|
||||||
|
class FabriceBellardBridge extends BridgeAbstract {
|
||||||
|
const NAME = 'Fabrice Bellard';
|
||||||
|
const URI = 'https://bellard.org/';
|
||||||
|
const DESCRIPTION = "Fabrice Bellard's Home Page";
|
||||||
|
const MAINTAINER = 'somini';
|
||||||
|
|
||||||
|
public function collectData() {
|
||||||
|
$html = getSimpleHTMLDOM(self::URI)
|
||||||
|
or returnServerError('Could not load content');
|
||||||
|
|
||||||
|
foreach ($html->find('p') as $obj) {
|
||||||
|
$item = array();
|
||||||
|
|
||||||
|
$html = defaultLinkTo($html, $this->getURI());
|
||||||
|
|
||||||
|
$links = $obj->find('a');
|
||||||
|
if (count($links) > 0) {
|
||||||
|
$link_uri = $links[0]->href;
|
||||||
|
} else {
|
||||||
|
$link_uri = $this->getURI();
|
||||||
|
}
|
||||||
|
|
||||||
|
/* try to make sure the link is valid */
|
||||||
|
if ($link_uri[-1] !== '/' && strpos($link_uri, '/') === false) {
|
||||||
|
$link_uri = $link_uri . '/';
|
||||||
|
}
|
||||||
|
|
||||||
|
$item['title'] = strip_tags($obj->innertext);
|
||||||
|
$item['uri'] = $link_uri;
|
||||||
|
$item['content'] = $obj->innertext;
|
||||||
|
|
||||||
|
$this->items[] = $item;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -2,7 +2,7 @@
|
||||||
class FacebookBridge extends BridgeAbstract {
|
class FacebookBridge extends BridgeAbstract {
|
||||||
|
|
||||||
const MAINTAINER = 'teromene, logmanoriginal';
|
const MAINTAINER = 'teromene, logmanoriginal';
|
||||||
const NAME = 'Facebook Bridge';
|
const NAME = 'Facebook Bridge | Main Site';
|
||||||
const URI = 'https://www.facebook.com/';
|
const URI = 'https://www.facebook.com/';
|
||||||
const CACHE_TIMEOUT = 300; // 5min
|
const CACHE_TIMEOUT = 300; // 5min
|
||||||
const DESCRIPTION = 'Input a page title or a profile log. For a profile log,
|
const DESCRIPTION = 'Input a page title or a profile log. For a profile log,
|
||||||
|
@ -66,14 +66,13 @@ class FacebookBridge extends BridgeAbstract {
|
||||||
|
|
||||||
case 'User':
|
case 'User':
|
||||||
if(!empty($this->authorName)) {
|
if(!empty($this->authorName)) {
|
||||||
return isset($this->extraInfos['name']) ? $this->extraInfos['name'] : $this->authorName
|
return isset($this->extraInfos['name']) ? $this->extraInfos['name'] : $this->authorName;
|
||||||
. ' - ' . static::NAME;
|
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case 'Group':
|
case 'Group':
|
||||||
if(!empty($this->groupName)) {
|
if(!empty($this->groupName)) {
|
||||||
return $this->groupName . ' - ' . static::NAME;
|
return $this->groupName;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
@ -82,6 +81,34 @@ class FacebookBridge extends BridgeAbstract {
|
||||||
return parent::getName();
|
return parent::getName();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function detectParameters($url){
|
||||||
|
$params = array();
|
||||||
|
|
||||||
|
// By profile
|
||||||
|
$regex = '/^(https?:\/\/)?(www\.)?facebook\.com\/profile\.php\?id\=([^\/?&\n]+)?(.*)/';
|
||||||
|
if(preg_match($regex, $url, $matches) > 0) {
|
||||||
|
$params['u'] = urldecode($matches[3]);
|
||||||
|
return $params;
|
||||||
|
}
|
||||||
|
|
||||||
|
// By group
|
||||||
|
$regex = '/^(https?:\/\/)?(www\.)?facebook\.com\/groups\/([^\/?\n]+)?(.*)/';
|
||||||
|
if(preg_match($regex, $url, $matches) > 0) {
|
||||||
|
$params['g'] = urldecode($matches[3]);
|
||||||
|
return $params;
|
||||||
|
}
|
||||||
|
|
||||||
|
// By username
|
||||||
|
$regex = '/^(https?:\/\/)?(www\.)?facebook\.com\/([^\/?\n]+)/';
|
||||||
|
|
||||||
|
if(preg_match($regex, $url, $matches) > 0) {
|
||||||
|
$params['u'] = urldecode($matches[3]);
|
||||||
|
return $params;
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
public function getURI() {
|
public function getURI() {
|
||||||
$uri = self::URI;
|
$uri = self::URI;
|
||||||
|
|
||||||
|
@ -142,7 +169,11 @@ class FacebookBridge extends BridgeAbstract {
|
||||||
|
|
||||||
private function collectGroupData() {
|
private function collectGroupData() {
|
||||||
|
|
||||||
$header = array('Accept-Language: ' . getEnv('HTTP_ACCEPT_LANGUAGE') . "\r\n");
|
if(getEnv('HTTP_ACCEPT_LANGUAGE')) {
|
||||||
|
$header = array('Accept-Language: ' . getEnv('HTTP_ACCEPT_LANGUAGE'));
|
||||||
|
} else {
|
||||||
|
$header = array();
|
||||||
|
}
|
||||||
|
|
||||||
$html = getSimpleHTMLDOM($this->getURI(), $header)
|
$html = getSimpleHTMLDOM($this->getURI(), $header)
|
||||||
or returnServerError('Failed loading facebook page: ' . $this->getURI());
|
or returnServerError('Failed loading facebook page: ' . $this->getURI());
|
||||||
|
@ -505,7 +536,11 @@ EOD;
|
||||||
// Retrieve page contents
|
// Retrieve page contents
|
||||||
if(is_null($html)) {
|
if(is_null($html)) {
|
||||||
|
|
||||||
|
if(getEnv('HTTP_ACCEPT_LANGUAGE')) {
|
||||||
$header = array('Accept-Language: ' . getEnv('HTTP_ACCEPT_LANGUAGE'));
|
$header = array('Accept-Language: ' . getEnv('HTTP_ACCEPT_LANGUAGE'));
|
||||||
|
} else {
|
||||||
|
$header = array();
|
||||||
|
}
|
||||||
|
|
||||||
$html = getSimpleHTMLDOM($this->getURI(), $header)
|
$html = getSimpleHTMLDOM($this->getURI(), $header)
|
||||||
or returnServerError('No results for this query.');
|
or returnServerError('No results for this query.');
|
||||||
|
@ -580,6 +615,8 @@ EOD;
|
||||||
'._5mly', // Remove embedded videos (the preview image remains)
|
'._5mly', // Remove embedded videos (the preview image remains)
|
||||||
'._2ezg', // Remove "Views ..."
|
'._2ezg', // Remove "Views ..."
|
||||||
'.hidden_elem', // Remove hidden elements (they are hidden anyway)
|
'.hidden_elem', // Remove hidden elements (they are hidden anyway)
|
||||||
|
'.timestampContent', // Remove relative timestamp
|
||||||
|
'._6spk', // Remove redundant separator
|
||||||
);
|
);
|
||||||
|
|
||||||
foreach($content_filters as $filter) {
|
foreach($content_filters as $filter) {
|
||||||
|
@ -664,9 +701,16 @@ EOD;
|
||||||
|
|
||||||
$uri = $post->find('abbr')[0]->parent()->getAttribute('href');
|
$uri = $post->find('abbr')[0]->parent()->getAttribute('href');
|
||||||
|
|
||||||
if (false !== strpos($uri, '?')) {
|
// Extract fbid and patch link
|
||||||
|
if (strpos($uri, '?') !== false) {
|
||||||
|
$query = substr($uri, strpos($uri, '?') + 1);
|
||||||
|
parse_str($query, $query_params);
|
||||||
|
if (isset($query_params['story_fbid'])) {
|
||||||
|
$uri = self::URI . $query_params['story_fbid'];
|
||||||
|
} else {
|
||||||
$uri = substr($uri, 0, strpos($uri, '?'));
|
$uri = substr($uri, 0, strpos($uri, '?'));
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
//Build and add final item
|
//Build and add final item
|
||||||
$item['uri'] = htmlspecialchars_decode($uri, ENT_QUOTES);
|
$item['uri'] = htmlspecialchars_decode($uri, ENT_QUOTES);
|
||||||
|
|
164
bridges/FicbookBridge.php
Normal file
164
bridges/FicbookBridge.php
Normal file
|
@ -0,0 +1,164 @@
|
||||||
|
<?php
|
||||||
|
class FicbookBridge extends BridgeAbstract {
|
||||||
|
|
||||||
|
const NAME = 'Ficbook Bridge';
|
||||||
|
const URI = 'https://ficbook.net/';
|
||||||
|
const DESCRIPTION = 'No description provided';
|
||||||
|
const MAINTAINER = 'logmanoriginal';
|
||||||
|
|
||||||
|
const PARAMETERS = array(
|
||||||
|
'Site News' => array(),
|
||||||
|
'Fiction Updates' => array(
|
||||||
|
'fiction_id' => array(
|
||||||
|
'name' => 'Fanfiction ID',
|
||||||
|
'type' => 'text',
|
||||||
|
'pattern' => '[0-9]+',
|
||||||
|
'required' => true,
|
||||||
|
'title' => 'Insert fanfiction ID',
|
||||||
|
'exampleValue' => '5783919',
|
||||||
|
),
|
||||||
|
'include_contents' => array(
|
||||||
|
'name' => 'Include contents',
|
||||||
|
'type' => 'checkbox',
|
||||||
|
'title' => 'Activate to include contents in the feed',
|
||||||
|
),
|
||||||
|
),
|
||||||
|
'Fiction Comments' => array(
|
||||||
|
'fiction_id' => array(
|
||||||
|
'name' => 'Fanfiction ID',
|
||||||
|
'type' => 'text',
|
||||||
|
'pattern' => '[0-9]+',
|
||||||
|
'required' => true,
|
||||||
|
'title' => 'Insert fanfiction ID',
|
||||||
|
'exampleValue' => '5783919',
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
|
||||||
|
public function getURI() {
|
||||||
|
switch($this->queriedContext) {
|
||||||
|
case 'Site News': {
|
||||||
|
// For some reason this is not HTTPS
|
||||||
|
return 'http://ficbook.net/sitenews';
|
||||||
|
}
|
||||||
|
case 'Fiction Updates': {
|
||||||
|
return self::URI
|
||||||
|
. 'readfic/'
|
||||||
|
. urlencode($this->getInput('fiction_id'));
|
||||||
|
}
|
||||||
|
case 'Fiction Comments': {
|
||||||
|
return self::URI
|
||||||
|
. 'readfic/'
|
||||||
|
. urlencode($this->getInput('fiction_id'))
|
||||||
|
. '/comments#content';
|
||||||
|
}
|
||||||
|
default: return parent::getURI();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public function collectData() {
|
||||||
|
|
||||||
|
$header = array('Accept-Language: en-US');
|
||||||
|
|
||||||
|
$html = getSimpleHTMLDOM($this->getURI(), $header)
|
||||||
|
or returnServerError('Could not request ' . $this->getURI());
|
||||||
|
|
||||||
|
$html = defaultLinkTo($html, self::URI);
|
||||||
|
|
||||||
|
switch($this->queriedContext) {
|
||||||
|
case 'Site News': return $this->collectSiteNews($html);
|
||||||
|
case 'Fiction Updates': return $this->collectUpdatesData($html);
|
||||||
|
case 'Fiction Comments': return $this->collectCommentsData($html);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
private function collectSiteNews($html) {
|
||||||
|
foreach($html->find('.news_view') as $news) {
|
||||||
|
$this->items[] = array(
|
||||||
|
'title' => $news->find('h1.title', 0)->plaintext,
|
||||||
|
'timestamp' => strtotime($this->fixDate($news->find('span[title]', 0)->title)),
|
||||||
|
'content' => $news->find('.news_text', 0),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private function collectCommentsData($html) {
|
||||||
|
foreach($html->find('article.post') as $article) {
|
||||||
|
$this->items[] = array(
|
||||||
|
'uri' => $article->find('.comment_link_to_fic > a', 0)->href,
|
||||||
|
'title' => $article->find('.comment_author', 0)->plaintext,
|
||||||
|
'author' => $article->find('.comment_author', 0)->plaintext,
|
||||||
|
'timestamp' => strtotime($this->fixDate($article->find('time[datetime]', 0)->datetime)),
|
||||||
|
'content' => $article->find('.comment_message', 0),
|
||||||
|
'enclosures' => array($article->find('img', 0)->src),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private function collectUpdatesData($html) {
|
||||||
|
foreach($html->find('ul.table-of-contents > li') as $chapter) {
|
||||||
|
$item = array(
|
||||||
|
'uri' => $chapter->find('a', 0)->href,
|
||||||
|
'title' => $chapter->find('a', 0)->plaintext,
|
||||||
|
'timestamp' => strtotime($this->fixDate($chapter->find('span[title]', 0)->title)),
|
||||||
|
);
|
||||||
|
|
||||||
|
if($this->getInput('include_contents')) {
|
||||||
|
$content = getSimpleHTMLDOMCached($item['uri']);
|
||||||
|
$item['content'] = $content->find('#content', 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->items[] = $item;
|
||||||
|
|
||||||
|
// Sort by time, descending
|
||||||
|
usort($this->items, function($a, $b){ return $b['timestamp'] - $a['timestamp']; });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private function fixDate($date) {
|
||||||
|
|
||||||
|
// FIXME: This list was generated using Google tranlator. Someone who
|
||||||
|
// actually knows russian should check this list! Please keep in mind
|
||||||
|
// that month names must match exactly the names returned by Ficbook.
|
||||||
|
$ru_month = array(
|
||||||
|
'января',
|
||||||
|
'февраля',
|
||||||
|
'марта',
|
||||||
|
'апреля',
|
||||||
|
'мая',
|
||||||
|
'июня',
|
||||||
|
'июля',
|
||||||
|
'августа',
|
||||||
|
'Сентября',
|
||||||
|
'октября',
|
||||||
|
'Ноября',
|
||||||
|
'Декабря',
|
||||||
|
);
|
||||||
|
|
||||||
|
$en_month = array(
|
||||||
|
'January',
|
||||||
|
'February',
|
||||||
|
'March',
|
||||||
|
'April',
|
||||||
|
'May',
|
||||||
|
'June',
|
||||||
|
'July',
|
||||||
|
'August',
|
||||||
|
'September',
|
||||||
|
'October',
|
||||||
|
'November',
|
||||||
|
'December',
|
||||||
|
);
|
||||||
|
|
||||||
|
$fixed_date = str_replace($ru_month, $en_month, $date);
|
||||||
|
|
||||||
|
if($fixed_date === $date) {
|
||||||
|
Debug::log('Unable to fix date: ' . $date);
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $fixed_date;
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
|
@ -62,11 +62,16 @@ class FindACrewBridge extends BridgeAbstract {
|
||||||
foreach ($annonces as $annonce) {
|
foreach ($annonces as $annonce) {
|
||||||
$item = array();
|
$item = array();
|
||||||
|
|
||||||
$img = parent::getURI() . $annonce->find('.lst-pic img', 0)->getAttribute('src');
|
$link = parent::getURI() . $annonce->find('.lst-ctrls a', 0)->href;
|
||||||
|
$htmlDetail = getSimpleHTMLDOMCached($link . '?mdl=2'); // add ?mdl=2 for xhr content not full html page
|
||||||
|
|
||||||
|
$img = parent::getURI() . $htmlDetail->find('img.img-responsive', 0)->getAttribute('src');
|
||||||
$item['title'] = $annonce->find('.lst-tags span', 0)->plaintext;
|
$item['title'] = $annonce->find('.lst-tags span', 0)->plaintext;
|
||||||
$item['uri'] = parent::getURI() . $annonce->find('.lst-ctrls a', 0)->href;
|
$item['uri'] = $link;
|
||||||
$content = $annonce->find('.lst-dtl', 0)->innertext;
|
$content = $htmlDetail->find('.panel-body div.clearfix.row > div', 1)->innertext;
|
||||||
$item['content'] = "<img src='$img' /><br>$content";
|
$content .= $htmlDetail->find('.panel-body > div', 1)->innertext;
|
||||||
|
$content = defaultLinkTo($content, parent::getURI());
|
||||||
|
$item['content'] = $content;
|
||||||
$item['enclosures'] = array($img);
|
$item['enclosures'] = array($img);
|
||||||
$item['categories'] = array($annonce->find('.css_AccLocCur', 0)->plaintext);
|
$item['categories'] = array($annonce->find('.css_AccLocCur', 0)->plaintext);
|
||||||
$this->items[] = $item;
|
$this->items[] = $item;
|
||||||
|
|
27
bridges/FreeCodeCampBridge.php
Normal file
27
bridges/FreeCodeCampBridge.php
Normal file
|
@ -0,0 +1,27 @@
|
||||||
|
<?php
|
||||||
|
class FreeCodeCampBridge extends FeedExpander {
|
||||||
|
|
||||||
|
const MAINTAINER = 'IceWreck';
|
||||||
|
const NAME = 'FreeCodecamp Bridge';
|
||||||
|
const URI = 'https://www.freecodecamp.org';
|
||||||
|
const CACHE_TIMEOUT = 3600;
|
||||||
|
const DESCRIPTION = 'RSS feed for FreeCodeCamp';
|
||||||
|
// Freecodecamp removed their old full content rss feed and replaced it with one liner content.
|
||||||
|
|
||||||
|
public function collectData(){
|
||||||
|
$this->collectExpandableDatas('https://www.freecodecamp.org/news/rss/', 15);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function parseItem($newsItem){
|
||||||
|
$item = parent::parseItem($newsItem);
|
||||||
|
// $articlePage gets the entire page's contents
|
||||||
|
$articlePage = getSimpleHTMLDOM($newsItem->link);
|
||||||
|
// figure contain's the main article image
|
||||||
|
$article = $articlePage->find('figure', 0);
|
||||||
|
// the actual article
|
||||||
|
foreach($articlePage->find('.post-full-content') as $element)
|
||||||
|
$article = $article . $element;
|
||||||
|
$item['content'] = $article;
|
||||||
|
return $item;
|
||||||
|
}
|
||||||
|
}
|
918
bridges/FurAffinityBridge.php
Normal file
918
bridges/FurAffinityBridge.php
Normal file
|
@ -0,0 +1,918 @@
|
||||||
|
<?php
|
||||||
|
class FurAffinityBridge extends BridgeAbstract {
|
||||||
|
const NAME = 'FurAffinity Bridge';
|
||||||
|
const URI = 'https://www.furaffinity.net';
|
||||||
|
const CACHE_TIMEOUT = 300; // 5min
|
||||||
|
const DESCRIPTION = 'Returns posts from various sections of FurAffinity';
|
||||||
|
const MAINTAINER = 'Roliga';
|
||||||
|
const PARAMETERS = array(
|
||||||
|
'Search' => array(
|
||||||
|
'q' => array(
|
||||||
|
'name' => 'Query',
|
||||||
|
'required' => true
|
||||||
|
),
|
||||||
|
'rating-general' => array(
|
||||||
|
'name' => 'General',
|
||||||
|
'type' => 'checkbox',
|
||||||
|
'defaultValue' => 'checked'
|
||||||
|
),
|
||||||
|
'rating-mature' => array(
|
||||||
|
'name' => 'Mature',
|
||||||
|
'type' => 'checkbox',
|
||||||
|
),
|
||||||
|
'rating-adult' => array(
|
||||||
|
'name' => 'Adult',
|
||||||
|
'type' => 'checkbox',
|
||||||
|
),
|
||||||
|
'range' => array(
|
||||||
|
'name' => 'Time range',
|
||||||
|
'type' => 'list',
|
||||||
|
'values' => array(
|
||||||
|
'A Day' => 'day',
|
||||||
|
'3 Days' => '3days',
|
||||||
|
'A Week' => 'week',
|
||||||
|
'A Month' => 'month',
|
||||||
|
'All time' => 'all'
|
||||||
|
),
|
||||||
|
'defaultValue' => 'all'
|
||||||
|
),
|
||||||
|
'type-art' => array(
|
||||||
|
'name' => 'Art',
|
||||||
|
'type' => 'checkbox',
|
||||||
|
'defaultValue' => 'checked'
|
||||||
|
),
|
||||||
|
'type-flash' => array(
|
||||||
|
'name' => 'Flash',
|
||||||
|
'type' => 'checkbox',
|
||||||
|
'defaultValue' => 'checked'
|
||||||
|
),
|
||||||
|
'type-photo' => array(
|
||||||
|
'name' => 'Photography',
|
||||||
|
'type' => 'checkbox',
|
||||||
|
'defaultValue' => 'checked'
|
||||||
|
),
|
||||||
|
'type-music' => array(
|
||||||
|
'name' => 'Music',
|
||||||
|
'type' => 'checkbox',
|
||||||
|
'defaultValue' => 'checked'
|
||||||
|
),
|
||||||
|
'type-story' => array(
|
||||||
|
'name' => 'Story',
|
||||||
|
'type' => 'checkbox',
|
||||||
|
'defaultValue' => 'checked'
|
||||||
|
),
|
||||||
|
'type-poetry' => array(
|
||||||
|
'name' => 'Poetry',
|
||||||
|
'type' => 'checkbox',
|
||||||
|
'defaultValue' => 'checked'
|
||||||
|
),
|
||||||
|
'mode' => array(
|
||||||
|
'name' => 'Match mode',
|
||||||
|
'type' => 'list',
|
||||||
|
'values' => array(
|
||||||
|
'All of the words' => 'all',
|
||||||
|
'Any of the words' => 'any',
|
||||||
|
'Extended' => 'extended'
|
||||||
|
),
|
||||||
|
'defaultValue' => 'extended'
|
||||||
|
),
|
||||||
|
'limit' => array(
|
||||||
|
'name' => 'Limit',
|
||||||
|
'type' => 'number',
|
||||||
|
'defaultValue' => 10,
|
||||||
|
'title' => 'Limit number of submissions to return. -1 for unlimited.'
|
||||||
|
),
|
||||||
|
'full' => array(
|
||||||
|
'name' => 'Full view',
|
||||||
|
'title' => 'Include description, tags, date and larger image in article. Uses more bandwidth.',
|
||||||
|
'type' => 'checkbox',
|
||||||
|
'defaultValue' => 'checked'
|
||||||
|
),
|
||||||
|
'cache' => array(
|
||||||
|
'name' => 'Cache submission pages',
|
||||||
|
'title' => 'Reduces requests to FA when Full view is enabled. Changes to submission details may be delayed.',
|
||||||
|
'type' => 'checkbox',
|
||||||
|
'defaultValue' => 'checked'
|
||||||
|
)
|
||||||
|
),
|
||||||
|
'Browse' => array(
|
||||||
|
'cat' => array(
|
||||||
|
'name' => 'Category',
|
||||||
|
'type' => 'list',
|
||||||
|
'values' => array(
|
||||||
|
'Visual Art' => array(
|
||||||
|
'All' => 1,
|
||||||
|
'Artwork (Digital)' => 2,
|
||||||
|
'Artwork (Traditional)' => 3,
|
||||||
|
'Cellshading' => 4,
|
||||||
|
'Crafting' => 5,
|
||||||
|
'Designs' => 6,
|
||||||
|
'Flash' => 7,
|
||||||
|
'Fursuiting' => 8,
|
||||||
|
'Icons' => 9,
|
||||||
|
'Mosaics' => 10,
|
||||||
|
'Photography' => 11,
|
||||||
|
'Sculpting' => 12
|
||||||
|
),
|
||||||
|
'Readable Art' => array(
|
||||||
|
'Story' => 13,
|
||||||
|
'Poetry' => 14,
|
||||||
|
'Prose' => 15
|
||||||
|
),
|
||||||
|
'Audio Art' => array(
|
||||||
|
'Music' => 16,
|
||||||
|
'Podcasts' => 17
|
||||||
|
),
|
||||||
|
'Downloadable' => array(
|
||||||
|
'Skins' => 18,
|
||||||
|
'Handhelds' => 19,
|
||||||
|
'Resources' => 20
|
||||||
|
),
|
||||||
|
'Other Stuff' => array(
|
||||||
|
'Adoptables' => 21,
|
||||||
|
'Auctions' => 22,
|
||||||
|
'Contests' => 23,
|
||||||
|
'Current Events' => 24,
|
||||||
|
'Desktops' => 25,
|
||||||
|
'Stockart' => 26,
|
||||||
|
'Screenshots' => 27,
|
||||||
|
'Scraps' => 28,
|
||||||
|
'Wallpaper' => 29,
|
||||||
|
'YCH / Sale' => 30,
|
||||||
|
'Other' => 31
|
||||||
|
)
|
||||||
|
),
|
||||||
|
'defaultValue' => 1
|
||||||
|
),
|
||||||
|
'atype' => array(
|
||||||
|
'name' => 'Type',
|
||||||
|
'type' => 'list',
|
||||||
|
'values' => array(
|
||||||
|
'General Things' => array(
|
||||||
|
'All' => 1,
|
||||||
|
'Abstract' => 2,
|
||||||
|
'Animal related (non-anthro)' => 3,
|
||||||
|
'Anime' => 4,
|
||||||
|
'Comics' => 5,
|
||||||
|
'Doodle' => 6,
|
||||||
|
'Fanart' => 7,
|
||||||
|
'Fantasy' => 8,
|
||||||
|
'Human' => 9,
|
||||||
|
'Portraits' => 10,
|
||||||
|
'Scenery' => 11,
|
||||||
|
'Still Life' => 12,
|
||||||
|
'Tutorials' => 13,
|
||||||
|
'Miscellaneous' => 14
|
||||||
|
),
|
||||||
|
'Fetish / Furry specialty' => array(
|
||||||
|
'Baby fur' => 101,
|
||||||
|
'Bondage' => 102,
|
||||||
|
'Digimon' => 103,
|
||||||
|
'Fat Furs' => 104,
|
||||||
|
'Fetish Other' => 105,
|
||||||
|
'Fursuit' => 106,
|
||||||
|
'Gore / Macabre Art' => 119,
|
||||||
|
'Hyper' => 107,
|
||||||
|
'Inflation' => 108,
|
||||||
|
'Macro / Micro' => 109,
|
||||||
|
'Muscle' => 110,
|
||||||
|
'My Little Pony / Brony' => 111,
|
||||||
|
'Paw' => 112,
|
||||||
|
'Pokemon' => 113,
|
||||||
|
'Pregnancy' => 114,
|
||||||
|
'Sonic' => 115,
|
||||||
|
'Transformation' => 116,
|
||||||
|
'Vore' => 117,
|
||||||
|
'Water Sports' => 118,
|
||||||
|
'General Furry Art' => 100
|
||||||
|
),
|
||||||
|
'Music' => array(
|
||||||
|
'Techno' => 201,
|
||||||
|
'Trance' => 202,
|
||||||
|
'House' => 203,
|
||||||
|
'90s' => 204,
|
||||||
|
'80s' => 205,
|
||||||
|
'70s' => 206,
|
||||||
|
'60s' => 207,
|
||||||
|
'Pre-60s' => 208,
|
||||||
|
'Classical' => 209,
|
||||||
|
'Game Music' => 210,
|
||||||
|
'Rock' => 211,
|
||||||
|
'Pop' => 212,
|
||||||
|
'Rap' => 213,
|
||||||
|
'Industrial' => 214,
|
||||||
|
'Other Music' => 200
|
||||||
|
)
|
||||||
|
),
|
||||||
|
'defaultValue' => 1
|
||||||
|
),
|
||||||
|
'species' => array(
|
||||||
|
'name' => 'Species',
|
||||||
|
'type' => 'list',
|
||||||
|
'values' => array(
|
||||||
|
'Unspecified / Any' => 1,
|
||||||
|
'Amphibian' => array(
|
||||||
|
'Frog' => 1001,
|
||||||
|
'Newt' => 1002,
|
||||||
|
'Salamander' => 1003,
|
||||||
|
'Amphibian (Other)' => 1000
|
||||||
|
),
|
||||||
|
'Aquatic' => array(
|
||||||
|
'Cephalopod' => 2001,
|
||||||
|
'Dolphin' => 2002,
|
||||||
|
'Fish' => 2005,
|
||||||
|
'Porpoise' => 2004,
|
||||||
|
'Seal' => 6068,
|
||||||
|
'Shark' => 2006,
|
||||||
|
'Whale' => 2003,
|
||||||
|
'Aquatic (Other)' => 2000
|
||||||
|
),
|
||||||
|
'Avian' => array(
|
||||||
|
'Corvid' => 3001,
|
||||||
|
'Crow' => 3002,
|
||||||
|
'Duck' => 3003,
|
||||||
|
'Eagle' => 3004,
|
||||||
|
'Falcon' => 3005,
|
||||||
|
'Goose' => 3006,
|
||||||
|
'Gryphon' => 3007,
|
||||||
|
'Hawk' => 3008,
|
||||||
|
'Owl' => 3009,
|
||||||
|
'Phoenix' => 3010,
|
||||||
|
'Swan' => 3011,
|
||||||
|
'Avian (Other)' => 3000
|
||||||
|
),
|
||||||
|
'Bears & Ursines' => array(
|
||||||
|
'Bear' => 6002
|
||||||
|
),
|
||||||
|
'Camelids' => array(
|
||||||
|
'Camel' => 6074,
|
||||||
|
'Llama' => 6036
|
||||||
|
),
|
||||||
|
'Canines & Lupines' => array(
|
||||||
|
'Coyote' => 6008,
|
||||||
|
'Doberman' => 6009,
|
||||||
|
'Dog' => 6010,
|
||||||
|
'Dingo' => 6011,
|
||||||
|
'German Shepherd' => 6012,
|
||||||
|
'Jackal' => 6013,
|
||||||
|
'Husky' => 6014,
|
||||||
|
'Wolf' => 6016,
|
||||||
|
'Canine (Other)' => 6017
|
||||||
|
),
|
||||||
|
'Cervines' => array(
|
||||||
|
'Cervine (Other)' => 6018
|
||||||
|
),
|
||||||
|
'Cows & Bovines' => array(
|
||||||
|
'Antelope' => 6004,
|
||||||
|
'Cows' => 6003,
|
||||||
|
'Gazelle' => 6005,
|
||||||
|
'Goat' => 6006,
|
||||||
|
'Bovines (General)' => 6007
|
||||||
|
),
|
||||||
|
'Dragons' => array(
|
||||||
|
'Eastern Dragon' => 4001,
|
||||||
|
'Hydra' => 4002,
|
||||||
|
'Serpent' => 4003,
|
||||||
|
'Western Dragon' => 4004,
|
||||||
|
'Wyvern' => 4005,
|
||||||
|
'Dragon (Other)' => 4000
|
||||||
|
),
|
||||||
|
'Equestrians' => array(
|
||||||
|
'Donkey' => 6019,
|
||||||
|
'Horse' => 6034,
|
||||||
|
'Pony' => 6073,
|
||||||
|
'Zebra' => 6071
|
||||||
|
),
|
||||||
|
'Exotic & Mythicals' => array(
|
||||||
|
'Argonian' => 5002,
|
||||||
|
'Chakat' => 5003,
|
||||||
|
'Chocobo' => 5004,
|
||||||
|
'Citra' => 5005,
|
||||||
|
'Crux' => 5006,
|
||||||
|
'Daemon' => 5007,
|
||||||
|
'Digimon' => 5008,
|
||||||
|
'Dracat' => 5009,
|
||||||
|
'Draenei' => 5010,
|
||||||
|
'Elf' => 5011,
|
||||||
|
'Gargoyle' => 5012,
|
||||||
|
'Iksar' => 5013,
|
||||||
|
'Kaiju/Monster' => 5015,
|
||||||
|
'Langurhali' => 5014,
|
||||||
|
'Moogle' => 5017,
|
||||||
|
'Naga' => 5016,
|
||||||
|
'Orc' => 5018,
|
||||||
|
'Pokemon' => 5019,
|
||||||
|
'Satyr' => 5020,
|
||||||
|
'Sergal' => 5021,
|
||||||
|
'Tanuki' => 5022,
|
||||||
|
'Unicorn' => 5023,
|
||||||
|
'Xenomorph' => 5024,
|
||||||
|
'Alien (Other)' => 5001,
|
||||||
|
'Exotic (Other)' => 5000
|
||||||
|
),
|
||||||
|
'Felines' => array(
|
||||||
|
'Domestic Cat' => 6020,
|
||||||
|
'Cheetah' => 6021,
|
||||||
|
'Cougar' => 6022,
|
||||||
|
'Jaguar' => 6023,
|
||||||
|
'Leopard' => 6024,
|
||||||
|
'Lion' => 6025,
|
||||||
|
'Lynx' => 6026,
|
||||||
|
'Ocelot' => 6027,
|
||||||
|
'Panther' => 6028,
|
||||||
|
'Tiger' => 6029,
|
||||||
|
'Feline (Other)' => 6030
|
||||||
|
),
|
||||||
|
'Insects' => array(
|
||||||
|
'Arachnid' => 8000,
|
||||||
|
'Mantid' => 8004,
|
||||||
|
'Scorpion' => 8005,
|
||||||
|
'Insect (Other)' => 8003
|
||||||
|
),
|
||||||
|
'Mammals (Other)' => array(
|
||||||
|
'Bat' => 6001,
|
||||||
|
'Giraffe' => 6031,
|
||||||
|
'Hedgehog' => 6032,
|
||||||
|
'Hippopotamus' => 6033,
|
||||||
|
'Hyena' => 6035,
|
||||||
|
'Panda' => 6052,
|
||||||
|
'Pig/Swine' => 6053,
|
||||||
|
'Rabbit/Hare' => 6059,
|
||||||
|
'Raccoon' => 6060,
|
||||||
|
'Red Panda' => 6062,
|
||||||
|
'Meerkat' => 6043,
|
||||||
|
'Mongoose' => 6044,
|
||||||
|
'Rhinoceros' => 6063,
|
||||||
|
'Mammals (Other)' => 6000
|
||||||
|
),
|
||||||
|
'Marsupials' => array(
|
||||||
|
'Opossum' => 6037,
|
||||||
|
'Kangaroo' => 6038,
|
||||||
|
'Koala' => 6039,
|
||||||
|
'Quoll' => 6040,
|
||||||
|
'Wallaby' => 6041,
|
||||||
|
'Marsupial (Other)' => 6042
|
||||||
|
),
|
||||||
|
'Mustelids' => array(
|
||||||
|
'Badger' => 6045,
|
||||||
|
'Ferret' => 6046,
|
||||||
|
'Mink' => 6048,
|
||||||
|
'Otter' => 6047,
|
||||||
|
'Skunk' => 6069,
|
||||||
|
'Weasel' => 6049,
|
||||||
|
'Mustelid (Other)' => 6051
|
||||||
|
),
|
||||||
|
'Primates' => array(
|
||||||
|
'Gorilla' => 6054,
|
||||||
|
'Human' => 6055,
|
||||||
|
'Lemur' => 6056,
|
||||||
|
'Monkey' => 6057,
|
||||||
|
'Primate (Other)' => 6058
|
||||||
|
),
|
||||||
|
'Reptillian' => array(
|
||||||
|
'Alligator & Crocodile' => 7001,
|
||||||
|
'Gecko' => 7003,
|
||||||
|
'Iguana' => 7004,
|
||||||
|
'Lizard' => 7005,
|
||||||
|
'Snakes & Serpents' => 7006,
|
||||||
|
'Turtle' => 7007,
|
||||||
|
'Reptilian (Other)' => 7000
|
||||||
|
),
|
||||||
|
'Rodents' => array(
|
||||||
|
'Beaver' => 6064,
|
||||||
|
'Mouse' => 6065,
|
||||||
|
'Rat' => 6061,
|
||||||
|
'Squirrel' => 6070,
|
||||||
|
'Rodent (Other)' => 6067
|
||||||
|
),
|
||||||
|
'Vulpines' => array(
|
||||||
|
'Fennec' => 6072,
|
||||||
|
'Fox' => 6075,
|
||||||
|
'Vulpine (Other)' => 6015
|
||||||
|
),
|
||||||
|
'Other' => array(
|
||||||
|
'Dinosaur' => 8001,
|
||||||
|
'Wolverine' => 6050
|
||||||
|
)
|
||||||
|
),
|
||||||
|
'defaultValue' => 1
|
||||||
|
),
|
||||||
|
'gender' => array(
|
||||||
|
'name' => 'Gender',
|
||||||
|
'type' => 'list',
|
||||||
|
'values' => array(
|
||||||
|
'Any' => 0,
|
||||||
|
'Male' => 2,
|
||||||
|
'Female' => 3,
|
||||||
|
'Herm' => 4,
|
||||||
|
'Transgender' => 5,
|
||||||
|
'Multiple characters' => 6,
|
||||||
|
'Other / Not Specified' => 7
|
||||||
|
),
|
||||||
|
'defaultValue' => 0
|
||||||
|
),
|
||||||
|
'rating_general' => array(
|
||||||
|
'name' => 'General',
|
||||||
|
'type' => 'checkbox',
|
||||||
|
'defaultValue' => 'checked'
|
||||||
|
),
|
||||||
|
'rating_mature' => array(
|
||||||
|
'name' => 'Mature',
|
||||||
|
'type' => 'checkbox',
|
||||||
|
),
|
||||||
|
'rating_adult' => array(
|
||||||
|
'name' => 'Adult',
|
||||||
|
'type' => 'checkbox',
|
||||||
|
),
|
||||||
|
'limit-browse' => array(
|
||||||
|
'name' => 'Limit',
|
||||||
|
'type' => 'number',
|
||||||
|
'required' => true,
|
||||||
|
'defaultValue' => 10,
|
||||||
|
'title' => 'Limit number of submissions to return. -1 for unlimited.'
|
||||||
|
),
|
||||||
|
'full' => array(
|
||||||
|
'name' => 'Full view',
|
||||||
|
'title' => 'Include description, tags, date and larger image in article. Uses more bandwidth.',
|
||||||
|
'type' => 'checkbox',
|
||||||
|
'defaultValue' => 'checked'
|
||||||
|
),
|
||||||
|
'cache' => array(
|
||||||
|
'name' => 'Cache submission pages',
|
||||||
|
'title' => 'Reduces requests to FA when Full view is enabled. Changes to submission details may be delayed.',
|
||||||
|
'type' => 'checkbox',
|
||||||
|
'defaultValue' => 'checked'
|
||||||
|
)
|
||||||
|
|
||||||
|
),
|
||||||
|
'Journals' => array(
|
||||||
|
'username-journals' => array(
|
||||||
|
'name' => 'Username',
|
||||||
|
'required' => true,
|
||||||
|
'title' => 'Lowercase username as seen in URLs'
|
||||||
|
),
|
||||||
|
'limit' => array(
|
||||||
|
'name' => 'Limit',
|
||||||
|
'type' => 'number',
|
||||||
|
'defaultValue' => -1,
|
||||||
|
'title' => 'Limit number of journals to return. -1 for unlimited.'
|
||||||
|
)
|
||||||
|
|
||||||
|
),
|
||||||
|
'Single Journal' => array(
|
||||||
|
'journal-id' => array(
|
||||||
|
'name' => 'Journal ID',
|
||||||
|
'required' => true,
|
||||||
|
'type' => 'number',
|
||||||
|
'title' => 'Number seen in journal URL'
|
||||||
|
)
|
||||||
|
),
|
||||||
|
'Gallery' => array(
|
||||||
|
'username-gallery' => array(
|
||||||
|
'name' => 'Username',
|
||||||
|
'required' => true,
|
||||||
|
'title' => 'Lowercase username as seen in URLs'
|
||||||
|
),
|
||||||
|
'limit' => array(
|
||||||
|
'name' => 'Limit',
|
||||||
|
'type' => 'number',
|
||||||
|
'defaultValue' => 10,
|
||||||
|
'title' => 'Limit number of submissions to return. -1 for unlimited.'
|
||||||
|
),
|
||||||
|
'full' => array(
|
||||||
|
'name' => 'Full view',
|
||||||
|
'title' => 'Include description, tags, date and larger image in article. Uses more bandwidth.',
|
||||||
|
'type' => 'checkbox',
|
||||||
|
'defaultValue' => 'checked'
|
||||||
|
),
|
||||||
|
'cache' => array(
|
||||||
|
'name' => 'Cache submission pages',
|
||||||
|
'title' => 'Reduces requests to FA when Full view is enabled. Changes to submission details may be delayed.',
|
||||||
|
'type' => 'checkbox',
|
||||||
|
'defaultValue' => 'checked'
|
||||||
|
)
|
||||||
|
),
|
||||||
|
'Scraps' => array(
|
||||||
|
'username-scraps' => array(
|
||||||
|
'name' => 'Username',
|
||||||
|
'required' => true,
|
||||||
|
'title' => 'Lowercase username as seen in URLs'
|
||||||
|
),
|
||||||
|
'limit' => array(
|
||||||
|
'name' => 'Limit',
|
||||||
|
'type' => 'number',
|
||||||
|
'defaultValue' => 10,
|
||||||
|
'title' => 'Limit number of submissions to return. -1 for unlimited.'
|
||||||
|
),
|
||||||
|
'full' => array(
|
||||||
|
'name' => 'Full view',
|
||||||
|
'title' => 'Include description, tags, date and larger image in article. Uses more bandwidth.',
|
||||||
|
'type' => 'checkbox',
|
||||||
|
'defaultValue' => 'checked'
|
||||||
|
),
|
||||||
|
'cache' => array(
|
||||||
|
'name' => 'Cache submission pages',
|
||||||
|
'title' => 'Reduces requests to FA when Full view is enabled. Changes to submission details may be delayed.',
|
||||||
|
'type' => 'checkbox',
|
||||||
|
'defaultValue' => 'checked'
|
||||||
|
)
|
||||||
|
),
|
||||||
|
'Favorites' => array(
|
||||||
|
'username-favorites' => array(
|
||||||
|
'name' => 'Username',
|
||||||
|
'required' => true,
|
||||||
|
'title' => 'Lowercase username as seen in URLs'
|
||||||
|
),
|
||||||
|
'limit' => array(
|
||||||
|
'name' => 'Limit',
|
||||||
|
'type' => 'number',
|
||||||
|
'defaultValue' => 10,
|
||||||
|
'title' => 'Limit number of submissions to return. -1 for unlimited.'
|
||||||
|
),
|
||||||
|
'full' => array(
|
||||||
|
'name' => 'Full view',
|
||||||
|
'title' => 'Include description, tags, date and larger image in article. Uses more bandwidth.',
|
||||||
|
'type' => 'checkbox',
|
||||||
|
'defaultValue' => 'checked'
|
||||||
|
),
|
||||||
|
'cache' => array(
|
||||||
|
'name' => 'Cache submission pages',
|
||||||
|
'title' => 'Reduces requests to FA when Full view is enabled. Changes to submission details may be delayed.',
|
||||||
|
'type' => 'checkbox',
|
||||||
|
'defaultValue' => 'checked'
|
||||||
|
)
|
||||||
|
),
|
||||||
|
'Gallery Folder' => array(
|
||||||
|
'username-folder' => array(
|
||||||
|
'name' => 'Username',
|
||||||
|
'required' => true,
|
||||||
|
'title' => 'Lowercase username as seen in URLs'
|
||||||
|
),
|
||||||
|
'folder-id' => array(
|
||||||
|
'name' => 'Folder ID',
|
||||||
|
'required' => true,
|
||||||
|
'type' => 'number',
|
||||||
|
'title' => 'Number seen in folder URL'
|
||||||
|
),
|
||||||
|
'limit' => array(
|
||||||
|
'name' => 'Limit',
|
||||||
|
'type' => 'number',
|
||||||
|
'defaultValue' => 10,
|
||||||
|
'title' => 'Limit number of submissions to return. -1 for unlimited.'
|
||||||
|
),
|
||||||
|
'full' => array(
|
||||||
|
'name' => 'Full view',
|
||||||
|
'title' => 'Include description, tags, date and larger image in article. Uses more bandwidth.',
|
||||||
|
'type' => 'checkbox',
|
||||||
|
'defaultValue' => 'checked'
|
||||||
|
),
|
||||||
|
'cache' => array(
|
||||||
|
'name' => 'Cache submission pages',
|
||||||
|
'title' => 'Reduces requests to FA when Full view is enabled. Changes to submission details may be delayed.',
|
||||||
|
'type' => 'checkbox',
|
||||||
|
'defaultValue' => 'checked'
|
||||||
|
)
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* This was aquired by creating a new user on FA then
|
||||||
|
* extracting the cookie from the browsers dev console.
|
||||||
|
*/
|
||||||
|
const FA_AUTH_COOKIE = 'b=4ce65691-b50f-4742-a990-bf28d6de16ee; a=ca6e4566-9d81-4263-9444-653b142e35f8';
|
||||||
|
|
||||||
|
public function detectParameters($url) {
|
||||||
|
$params = array();
|
||||||
|
|
||||||
|
// Single journal
|
||||||
|
$regex = '/^(https?:\/\/)?(www\.)?furaffinity.net\/journal\/(\d+)/';
|
||||||
|
if(preg_match($regex, $url, $matches) > 0) {
|
||||||
|
$params['journal-id'] = urldecode($matches[3]);
|
||||||
|
return $params;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Journals
|
||||||
|
$regex = '/^(https?:\/\/)?(www\.)?furaffinity.net\/journals\/([^\/&?\n]+)/';
|
||||||
|
if(preg_match($regex, $url, $matches) > 0) {
|
||||||
|
$params['username-journals'] = urldecode($matches[3]);
|
||||||
|
return $params;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Gallery folder
|
||||||
|
$regex = '/^(https?:\/\/)?(www\.)?furaffinity.net\/gallery\/([^\/&?\n]+)\/folder\/(\d+)/';
|
||||||
|
if(preg_match($regex, $url, $matches) > 0) {
|
||||||
|
$params['username-folder'] = urldecode($matches[3]);
|
||||||
|
$params['folder-id'] = urldecode($matches[4]);
|
||||||
|
$params['full'] = 'on';
|
||||||
|
return $params;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Gallery (must be after gallery folder)
|
||||||
|
$regex = '/^(https?:\/\/)?(www\.)?furaffinity.net\/(gallery|scraps|favorites)\/([^\/&?\n]+)/';
|
||||||
|
if(preg_match($regex, $url, $matches) > 0) {
|
||||||
|
$params['username-' . $matches[3]] = urldecode($matches[4]);
|
||||||
|
$params['full'] = 'on';
|
||||||
|
return $params;
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getName() {
|
||||||
|
switch($this->queriedContext) {
|
||||||
|
case 'Search':
|
||||||
|
return 'Search For '
|
||||||
|
. $this->getInput('q');
|
||||||
|
case 'Browse':
|
||||||
|
return 'Browse';
|
||||||
|
case 'Journals':
|
||||||
|
return $this->getInput('username-journals');
|
||||||
|
case 'Single Journal':
|
||||||
|
return 'Journal '
|
||||||
|
. $this->getInput('journal-id');
|
||||||
|
case 'Gallery':
|
||||||
|
return $this->getInput('username-gallery');
|
||||||
|
case 'Scraps':
|
||||||
|
return $this->getInput('username-scraps');
|
||||||
|
case 'Favorites':
|
||||||
|
return $this->getInput('username-favorites');
|
||||||
|
case 'Gallery Folder':
|
||||||
|
return $this->getInput('username-folder')
|
||||||
|
. '\'s Folder '
|
||||||
|
. $this->getInput('folder-id');
|
||||||
|
default: return parent::getName();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getDescription() {
|
||||||
|
switch($this->queriedContext) {
|
||||||
|
case 'Search':
|
||||||
|
return 'FurAffinity Search For '
|
||||||
|
. $this->getInput('q');
|
||||||
|
case 'Browse':
|
||||||
|
return 'FurAffinity Browse';
|
||||||
|
case 'Journals':
|
||||||
|
return 'FurAffinity Journals By '
|
||||||
|
. $this->getInput('username-journals');
|
||||||
|
case 'Single Journal':
|
||||||
|
return 'FurAffinity Journal '
|
||||||
|
. $this->getInput('journal-id');
|
||||||
|
case 'Gallery':
|
||||||
|
return 'FurAffinity Gallery By '
|
||||||
|
. $this->getInput('username-gallery');
|
||||||
|
case 'Scraps':
|
||||||
|
return 'FurAffinity Scraps By '
|
||||||
|
. $this->getInput('username-scraps');
|
||||||
|
case 'Favorites':
|
||||||
|
return 'FurAffinity Favorites By '
|
||||||
|
. $this->getInput('username-favorites');
|
||||||
|
case 'Gallery Folder':
|
||||||
|
return 'FurAffinity Gallery Folder '
|
||||||
|
. $this->getInput('folder-id')
|
||||||
|
. ' By '
|
||||||
|
. $this->getInput('username-folder');
|
||||||
|
default: return parent::getDescription();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getURI() {
|
||||||
|
switch($this->queriedContext) {
|
||||||
|
case 'Search':
|
||||||
|
return SELF::URI
|
||||||
|
. '/search';
|
||||||
|
case 'Browse':
|
||||||
|
return SELF::URI
|
||||||
|
. '/browse';
|
||||||
|
case 'Journals':
|
||||||
|
return SELF::URI
|
||||||
|
. '/journals/'
|
||||||
|
. $this->getInput('username-journals');
|
||||||
|
case 'Single Journal':
|
||||||
|
return SELF::URI
|
||||||
|
. '/journal/'
|
||||||
|
. $this->getInput('journal-id');
|
||||||
|
case 'Gallery':
|
||||||
|
return SELF::URI
|
||||||
|
. '/gallery/'
|
||||||
|
. $this->getInput('username-gallery');
|
||||||
|
case 'Scraps':
|
||||||
|
return SELF::URI
|
||||||
|
. '/scraps/'
|
||||||
|
. $this->getInput('username-scraps');
|
||||||
|
case 'Favorites':
|
||||||
|
return SELF::URI
|
||||||
|
. '/favorites/'
|
||||||
|
. $this->getInput('username-favorites');
|
||||||
|
case 'Gallery Folder':
|
||||||
|
return SELF::URI
|
||||||
|
. '/gallery/'
|
||||||
|
. $this->getInput('username-folder')
|
||||||
|
. '/folder/'
|
||||||
|
. $this->getInput('folder-id');
|
||||||
|
default: return parent::getURI();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public function collectData() {
|
||||||
|
switch($this->queriedContext) {
|
||||||
|
case 'Search':
|
||||||
|
$data = array(
|
||||||
|
'q' => $this->getInput('q'),
|
||||||
|
'perpage' => 72,
|
||||||
|
'rating-general' => ($this->getInput('rating-general') === true ? 'on' : 0),
|
||||||
|
'rating-mature' => ($this->getInput('rating-mature') === true ? 'on' : 0),
|
||||||
|
'rating-adult' => ($this->getInput('rating-adult') === true ? 'on' : 0),
|
||||||
|
'range' => $this->getInput('range'),
|
||||||
|
'type-art' => ($this->getInput('type-art') === true ? 'on' : 0),
|
||||||
|
'type-flash' => ($this->getInput('type-flash') === true ? 'on' : 0),
|
||||||
|
'type-photo' => ($this->getInput('type-photo') === true ? 'on' : 0),
|
||||||
|
'type-music' => ($this->getInput('type-music') === true ? 'on' : 0),
|
||||||
|
'type-story' => ($this->getInput('type-story') === true ? 'on' : 0),
|
||||||
|
'type-poetry' => ($this->getInput('type-poetry') === true ? 'on' : 0),
|
||||||
|
'mode' => $this->getInput('mode')
|
||||||
|
);
|
||||||
|
$html = $this->postFASimpleHTMLDOM($data);
|
||||||
|
$limit = (is_int($this->getInput('limit')) ? $this->getInput('limit') : 10);
|
||||||
|
$this->itemsFromSubmissionList($html, $limit);
|
||||||
|
break;
|
||||||
|
case 'Browse':
|
||||||
|
$data = array(
|
||||||
|
'cat' => $this->getInput('cat'),
|
||||||
|
'atype' => $this->getInput('atype'),
|
||||||
|
'species' => $this->getInput('species'),
|
||||||
|
'gender' => $this->getInput('gender'),
|
||||||
|
'perpage' => 72,
|
||||||
|
'rating_general' => ($this->getInput('rating_general') === true ? 'on' : 0),
|
||||||
|
'rating_mature' => ($this->getInput('rating_mature') === true ? 'on' : 0),
|
||||||
|
'rating_adult' => ($this->getInput('rating_adult') === true ? 'on' : 0)
|
||||||
|
);
|
||||||
|
$html = $this->postFASimpleHTMLDOM($data);
|
||||||
|
$limit = (is_int($this->getInput('limit-browse')) ? $this->getInput('limit-browse') : 10);
|
||||||
|
$this->itemsFromSubmissionList($html, $limit);
|
||||||
|
break;
|
||||||
|
case 'Journals':
|
||||||
|
$html = $this->getFASimpleHTMLDOM($this->getURI());
|
||||||
|
$limit = (is_int($this->getInput('limit')) ? $this->getInput('limit') : -1);
|
||||||
|
$this->itemsFromJournalList($html, $limit);
|
||||||
|
break;
|
||||||
|
case 'Single Journal':
|
||||||
|
$html = $this->getFASimpleHTMLDOM($this->getURI());
|
||||||
|
$this->itemsFromJournal($html);
|
||||||
|
break;
|
||||||
|
case 'Gallery':
|
||||||
|
case 'Scraps':
|
||||||
|
case 'Favorites':
|
||||||
|
case 'Gallery Folder':
|
||||||
|
$html = $this->getFASimpleHTMLDOM($this->getURI());
|
||||||
|
$limit = (is_int($this->getInput('limit')) ? $this->getInput('limit') : 10);
|
||||||
|
$this->itemsFromSubmissionList($html, $limit);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private function postFASimpleHTMLDOM($data) {
|
||||||
|
$opts = array(
|
||||||
|
CURLOPT_CUSTOMREQUEST => 'POST',
|
||||||
|
CURLOPT_POSTFIELDS => http_build_query($data)
|
||||||
|
);
|
||||||
|
$header = array(
|
||||||
|
'Host: ' . parse_url(self::URI, PHP_URL_HOST),
|
||||||
|
'Content-Type: application/x-www-form-urlencoded',
|
||||||
|
'Cookie: ' . self::FA_AUTH_COOKIE
|
||||||
|
);
|
||||||
|
|
||||||
|
$html = getSimpleHTMLDOM($this->getURI(), $header, $opts);
|
||||||
|
$html = defaultLinkTo($html, $this->getURI());
|
||||||
|
|
||||||
|
return $html;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function getFASimpleHTMLDOM($url, $cache = false) {
|
||||||
|
$header = array(
|
||||||
|
'Cookie: ' . self::FA_AUTH_COOKIE
|
||||||
|
);
|
||||||
|
|
||||||
|
if($cache) {
|
||||||
|
$html = getSimpleHTMLDOMCached($url, 86400, $header); // 24 hours
|
||||||
|
} else {
|
||||||
|
$html = getSimpleHTMLDOM($url, $header);
|
||||||
|
}
|
||||||
|
|
||||||
|
$html = defaultLinkTo($html, $url);
|
||||||
|
|
||||||
|
return $html;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function itemsFromJournalList($html, $limit) {
|
||||||
|
foreach($html->find('table[id^=jid:]') as $journal) {
|
||||||
|
# allows limit = -1 to mean 'unlimited'
|
||||||
|
if($limit-- === 0) break;
|
||||||
|
|
||||||
|
$item = array();
|
||||||
|
|
||||||
|
$this->setReferrerPolicy($journal);
|
||||||
|
|
||||||
|
$item['uri'] = $journal->find('a', 0)->href;
|
||||||
|
$item['title'] = html_entity_decode($journal->find('a', 0)->plaintext);
|
||||||
|
$item['author'] = $this->getInput('username-journals');
|
||||||
|
$item['timestamp'] = strtotime(
|
||||||
|
$journal->find('span.popup_date', 0)->plaintext);
|
||||||
|
$item['content'] = $journal
|
||||||
|
->find('.alt1 table div.no_overflow', 0)
|
||||||
|
->innertext;
|
||||||
|
|
||||||
|
$this->items[] = $item;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private function itemsFromJournal($html) {
|
||||||
|
$this->setReferrerPolicy($html);
|
||||||
|
$item = array();
|
||||||
|
|
||||||
|
$item['uri'] = $this->getURI();
|
||||||
|
|
||||||
|
$title = $html->find('.journal-title-box .no_overflow', 0)->plaintext;
|
||||||
|
$title = html_entity_decode($title);
|
||||||
|
$title = trim($title, " \t\n\r\0\x0B" . chr(0xC2) . chr(0xA0));
|
||||||
|
$item['title'] = $title;
|
||||||
|
|
||||||
|
$item['author'] = $html->find('.journal-title-box a', 0)->plaintext;
|
||||||
|
$item['timestamp'] = strtotime(
|
||||||
|
$html->find('.journal-title-box span.popup_date', 0)->plaintext);
|
||||||
|
$item['content'] = $html->find('.journal-body', 0)->innertext;
|
||||||
|
|
||||||
|
$this->items[] = $item;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function itemsFromSubmissionList($html, $limit) {
|
||||||
|
$cache = ($this->getInput('cache') === true);
|
||||||
|
|
||||||
|
foreach($html->find('section.gallery figure') as $figure) {
|
||||||
|
# allows limit = -1 to mean 'unlimited'
|
||||||
|
if($limit-- === 0) break;
|
||||||
|
|
||||||
|
$item = array();
|
||||||
|
|
||||||
|
$submissionURL = $figure->find('b u a', 0)->href;
|
||||||
|
$imgURL = 'https:' . $figure->find('b u a img', 0)->src;
|
||||||
|
|
||||||
|
$item['uri'] = $submissionURL;
|
||||||
|
$item['title'] = html_entity_decode(
|
||||||
|
$figure->find('figcaption p a[href*=/view/]', 0)->title);
|
||||||
|
$item['author'] = $figure->find('figcaption p a[href*=/user/]', 0)->title;
|
||||||
|
|
||||||
|
if($this->getInput('full') === true) {
|
||||||
|
$submissionHTML = $this->getFASimpleHTMLDOM($submissionURL, $cache);
|
||||||
|
|
||||||
|
$stats = $submissionHTML->find('.stats-container', 0);
|
||||||
|
$item['timestamp'] = strtotime($stats->find('.popup_date', 0)->title);
|
||||||
|
$item['enclosures'] = array(
|
||||||
|
$submissionHTML->find('.actions a[href^=https://d.facdn]', 0)->href
|
||||||
|
);
|
||||||
|
foreach($stats->find('#keywords a') as $keyword) {
|
||||||
|
$item['categories'][] = $keyword->plaintext;
|
||||||
|
}
|
||||||
|
|
||||||
|
$previewSrc = $submissionHTML->find('#submissionImg', 0)
|
||||||
|
->{'data-preview-src'};
|
||||||
|
if($previewSrc) {
|
||||||
|
$imgURL = 'https:' . $previewSrc;
|
||||||
|
}
|
||||||
|
|
||||||
|
$description = $submissionHTML
|
||||||
|
->find('.maintable .maintable tr td.alt1', -1);
|
||||||
|
$this->setReferrerPolicy($description);
|
||||||
|
$description = $description->innertext;
|
||||||
|
|
||||||
|
$item['content'] = <<<EOD
|
||||||
|
<a href="$submissionURL">
|
||||||
|
<img src="{$imgURL}" referrerpolicy="no-referrer" />
|
||||||
|
</a>
|
||||||
|
<p>
|
||||||
|
{$description}
|
||||||
|
</p>
|
||||||
|
EOD;
|
||||||
|
} else {
|
||||||
|
$item['content'] = <<<EOD
|
||||||
|
<a href="$submissionURL">
|
||||||
|
<img src="$imgURL" referrerpolicy="no-referrer" />
|
||||||
|
</a>
|
||||||
|
EOD;
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->items[] = $item;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private function setReferrerPolicy(&$html) {
|
||||||
|
foreach($html->find('img') as $img) {
|
||||||
|
/*
|
||||||
|
* Note: Without the no-referrer policy their CDN sometimes denies requests.
|
||||||
|
* We can't control this for enclosures sadly.
|
||||||
|
* At least tt-rss adds the referrerpolicy on its own.
|
||||||
|
* Alternatively we could not use https for images, but that's not ideal.
|
||||||
|
*/
|
||||||
|
$img->referrerpolicy = 'no-referrer';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
110
bridges/FurAffinityUserBridge.php
Normal file
110
bridges/FurAffinityUserBridge.php
Normal file
|
@ -0,0 +1,110 @@
|
||||||
|
<?php
|
||||||
|
class FurAffinityUserBridge extends BridgeAbstract {
|
||||||
|
const NAME = 'FurAffinity User Gallery';
|
||||||
|
const URI = 'https://www.furaffinity.net';
|
||||||
|
const MAINTAINER = 'CyberJacob';
|
||||||
|
const PARAMETERS = array(
|
||||||
|
array(
|
||||||
|
'searchUsername' => array(
|
||||||
|
'name' => 'Search Username',
|
||||||
|
'type' => 'text',
|
||||||
|
'required' => true,
|
||||||
|
'title' => 'Username to fetch the gallery for'
|
||||||
|
),
|
||||||
|
'loginUsername' => array(
|
||||||
|
'name' => 'Login Username',
|
||||||
|
'type' => 'text',
|
||||||
|
'required' => true
|
||||||
|
),
|
||||||
|
'loginPassword' => array(
|
||||||
|
'name' => 'Login Password',
|
||||||
|
'type' => 'text',
|
||||||
|
'required' => true
|
||||||
|
)
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
public function collectData() {
|
||||||
|
$cookies = self::login();
|
||||||
|
$url = self::URI . '/gallery/' . $this->getInput('searchUsername');
|
||||||
|
|
||||||
|
$html = getSimpleHTMLDOM($url, $cookies)
|
||||||
|
or returnServerError('Could not load the user\'s galary page.');
|
||||||
|
|
||||||
|
$submissions = $html->find('section[id=gallery-gallery]', 0)->find('figure');
|
||||||
|
foreach($submissions as $submission) {
|
||||||
|
$item = array();
|
||||||
|
$item['title'] = $submission->find('figcaption', 0)->find('a', 0)->plaintext;
|
||||||
|
|
||||||
|
$thumbnail = $submission->find('a', 0);
|
||||||
|
$thumbnail->href = self::URI . $thumbnail->href;
|
||||||
|
|
||||||
|
$item['content'] = $submission->find('a', 0);
|
||||||
|
|
||||||
|
$this->items[] = $item;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getName() {
|
||||||
|
return self::NAME . ' for ' . $this->getInput('searchUsername');
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getURI() {
|
||||||
|
return self::URI . '/user/' . $this->getInput('searchUsername');
|
||||||
|
}
|
||||||
|
|
||||||
|
private function login() {
|
||||||
|
$ch = curl_init(self::URI . '/login/');
|
||||||
|
curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
|
||||||
|
curl_setopt($ch, CURLOPT_FOLLOWLOCATION, true);
|
||||||
|
|
||||||
|
curl_setopt($ch, CURLOPT_USERAGENT, ini_get('user_agent'));
|
||||||
|
curl_setopt($ch, CURLOPT_ENCODING, '');
|
||||||
|
curl_setopt($ch, CURLOPT_PROTOCOLS, CURLPROTO_HTTP | CURLPROTO_HTTPS);
|
||||||
|
|
||||||
|
$fields = implode('&', array(
|
||||||
|
'action=login',
|
||||||
|
'retard_protection=1',
|
||||||
|
'name=' . urlencode($this->getInput('loginUsername')),
|
||||||
|
'pass=' . urlencode($this->getInput('loginPassword')),
|
||||||
|
'login=Login to Faraffinity'
|
||||||
|
));
|
||||||
|
|
||||||
|
curl_setopt($ch, CURLOPT_POST, 5);
|
||||||
|
curl_setopt($ch, CURLOPT_POSTFIELDS, $fields);
|
||||||
|
|
||||||
|
if(defined('PROXY_URL') && !defined('NOPROXY')) {
|
||||||
|
curl_setopt($ch, CURLOPT_PROXY, PROXY_URL);
|
||||||
|
}
|
||||||
|
|
||||||
|
curl_setopt($ch, CURLOPT_HEADER, true);
|
||||||
|
curl_setopt($ch, CURLINFO_HEADER_OUT, true);
|
||||||
|
|
||||||
|
$data = curl_exec($ch);
|
||||||
|
|
||||||
|
$errorCode = curl_getinfo($ch, CURLINFO_HTTP_CODE);
|
||||||
|
|
||||||
|
$curlError = curl_error($ch);
|
||||||
|
$curlErrno = curl_errno($ch);
|
||||||
|
$curlInfo = curl_getinfo($ch);
|
||||||
|
|
||||||
|
if($data === false)
|
||||||
|
fDebug::log("Cant't download {$url} cUrl error: {$curlError} ({$curlErrno})");
|
||||||
|
|
||||||
|
curl_close($ch);
|
||||||
|
|
||||||
|
if($errorCode != 200) {
|
||||||
|
returnServerError(error_get_last());
|
||||||
|
} else {
|
||||||
|
preg_match_all('/^Set-Cookie:\s*([^;]*)/mi', $data, $matches);
|
||||||
|
$cookies = array();
|
||||||
|
|
||||||
|
foreach($matches[1] as $item) {
|
||||||
|
parse_str($item, $cookie);
|
||||||
|
$cookies = array_merge($cookies, $cookie);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $cookies;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -40,6 +40,11 @@ class GQMagazineBridge extends BridgeAbstract
|
||||||
'data-original' => 'src'
|
'data-original' => 'src'
|
||||||
);
|
);
|
||||||
|
|
||||||
|
const POSSIBLE_TITLES = array(
|
||||||
|
'h2',
|
||||||
|
'h3'
|
||||||
|
);
|
||||||
|
|
||||||
private function getDomain() {
|
private function getDomain() {
|
||||||
$domain = $this->getInput('domain');
|
$domain = $this->getInput('domain');
|
||||||
if (empty($domain))
|
if (empty($domain))
|
||||||
|
@ -54,6 +59,17 @@ class GQMagazineBridge extends BridgeAbstract
|
||||||
return $this->getDomain() . '/' . $this->getInput('page');
|
return $this->getDomain() . '/' . $this->getInput('page');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private function findTitleOf($link) {
|
||||||
|
foreach (self::POSSIBLE_TITLES as $tag) {
|
||||||
|
$title = $link->parent()->find($tag, 0);
|
||||||
|
if($title !== null) {
|
||||||
|
if($title->plaintext !== null) {
|
||||||
|
return $title->plaintext;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public function collectData()
|
public function collectData()
|
||||||
{
|
{
|
||||||
$html = getSimpleHTMLDOM($this->getURI()) or returnServerError('Could not request ' . $this->getURI());
|
$html = getSimpleHTMLDOM($this->getURI()) or returnServerError('Could not request ' . $this->getURI());
|
||||||
|
@ -61,22 +77,26 @@ class GQMagazineBridge extends BridgeAbstract
|
||||||
// Since GQ don't want simple class scrapping, let's do it the hard way and ... discover content !
|
// Since GQ don't want simple class scrapping, let's do it the hard way and ... discover content !
|
||||||
$main = $html->find('main', 0);
|
$main = $html->find('main', 0);
|
||||||
foreach ($main->find('a') as $link) {
|
foreach ($main->find('a') as $link) {
|
||||||
|
if(strpos($link, $this->getInput('page')))
|
||||||
|
continue;
|
||||||
$uri = $link->href;
|
$uri = $link->href;
|
||||||
$title = $link->find('h2', 0);
|
$date = $link->parent()->find('time', 0);
|
||||||
$date = $link->find('time', 0);
|
|
||||||
|
|
||||||
$item = array();
|
$item = array();
|
||||||
$author = $link->find('span[itemprop=name]', 0);
|
$author = $link->parent()->find('span[itemprop=name]', 0);
|
||||||
|
if($author !== null) {
|
||||||
$item['author'] = $author->plaintext;
|
$item['author'] = $author->plaintext;
|
||||||
$item['title'] = $title->plaintext;
|
$item['title'] = $this->findTitleOf($link);
|
||||||
if(substr($uri, 0, 1) === 'h') { // absolute uri
|
switch(substr($uri, 0, 1)) {
|
||||||
|
case 'h': // absolute uri
|
||||||
$item['uri'] = $uri;
|
$item['uri'] = $uri;
|
||||||
} else if(substr($uri, 0, 1) === '/') { // domain relative url
|
break;
|
||||||
|
case '/': // domain relative uri
|
||||||
$item['uri'] = $this->getDomain() . $uri;
|
$item['uri'] = $this->getDomain() . $uri;
|
||||||
} else {
|
break;
|
||||||
|
default:
|
||||||
$item['uri'] = $this->getDomain() . '/' . $uri;
|
$item['uri'] = $this->getDomain() . '/' . $uri;
|
||||||
}
|
}
|
||||||
|
|
||||||
$article = $this->loadFullArticle($item['uri']);
|
$article = $this->loadFullArticle($item['uri']);
|
||||||
if($article) {
|
if($article) {
|
||||||
$item['content'] = $this->replaceUriInHtmlElement($article);
|
$item['content'] = $this->replaceUriInHtmlElement($article);
|
||||||
|
@ -88,6 +108,7 @@ class GQMagazineBridge extends BridgeAbstract
|
||||||
$this->items[] = $item;
|
$this->items[] = $item;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Loads the full article and returns the contents
|
* Loads the full article and returns the contents
|
||||||
|
@ -96,16 +117,7 @@ class GQMagazineBridge extends BridgeAbstract
|
||||||
*/
|
*/
|
||||||
private function loadFullArticle($uri){
|
private function loadFullArticle($uri){
|
||||||
$html = getSimpleHTMLDOMCached($uri);
|
$html = getSimpleHTMLDOMCached($uri);
|
||||||
// Once again, that generated css classes madness is an obstacle ... which i can go over easily
|
return $html->find('section[data-test-id=MainContentWrapper]', 0);
|
||||||
foreach($html->find('div') as $div) {
|
|
||||||
// List the CSS classes of that div
|
|
||||||
$classes = $div->class;
|
|
||||||
// I can't directly lookup that class since GQ since to generate random names like "ArticleBodySection-fkggUW"
|
|
||||||
if(strpos($classes, 'ArticleBodySection') !== false) {
|
|
||||||
return $div;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return null;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -5,7 +5,7 @@ class GiphyBridge extends BridgeAbstract {
|
||||||
|
|
||||||
const MAINTAINER = 'kraoc';
|
const MAINTAINER = 'kraoc';
|
||||||
const NAME = 'Giphy Bridge';
|
const NAME = 'Giphy Bridge';
|
||||||
const URI = 'http://giphy.com/';
|
const URI = 'https://giphy.com/';
|
||||||
const CACHE_TIMEOUT = 300; //5min
|
const CACHE_TIMEOUT = 300; //5min
|
||||||
const DESCRIPTION = 'Bridge for giphy.com';
|
const DESCRIPTION = 'Bridge for giphy.com';
|
||||||
|
|
||||||
|
|
27
bridges/GiteaBridge.php
Normal file
27
bridges/GiteaBridge.php
Normal file
|
@ -0,0 +1,27 @@
|
||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* Gitea is a fork of Gogs which may diverge in the future.
|
||||||
|
* https://docs.gitea.io/en-us/
|
||||||
|
*/
|
||||||
|
require_once 'GogsBridge.php';
|
||||||
|
|
||||||
|
class GiteaBridge extends GogsBridge {
|
||||||
|
|
||||||
|
const NAME = 'Gitea';
|
||||||
|
const URI = 'https://gitea.io';
|
||||||
|
const DESCRIPTION = 'Returns the latest issues, commits or releases';
|
||||||
|
const MAINTAINER = 'logmanoriginal';
|
||||||
|
const CACHE_TIMEOUT = 300; // 5 minutes
|
||||||
|
|
||||||
|
protected function collectReleasesData($html) {
|
||||||
|
$releases = $html->find('#release-list > li')
|
||||||
|
or returnServerError('Unable to find releases');
|
||||||
|
|
||||||
|
foreach($releases as $release) {
|
||||||
|
$this->items[] = array(
|
||||||
|
'uri' => $release->find('a', 0)->href,
|
||||||
|
'title' => 'Release ' . $release->find('h3', 0)->plaintext,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -66,10 +66,21 @@ class GithubIssueBridge extends BridgeAbstract {
|
||||||
return parent::getURI();
|
return parent::getURI();
|
||||||
}
|
}
|
||||||
|
|
||||||
protected function extractIssueEvent($issueNbr, $title, $comment){
|
private function buildGitHubIssueCommentUri($issue_number, $comment_id) {
|
||||||
$comment = $comment->firstChild();
|
// https://github.com/<user>/<project>/issues/<issue-number>#<id>
|
||||||
$uri = static::URI . $this->getInput('u') . '/' . $this->getInput('p')
|
return static::URI
|
||||||
. '/issues/' . $issueNbr . '#' . $comment->getAttribute('id');
|
. $this->getInput('u')
|
||||||
|
. '/'
|
||||||
|
. $this->getInput('p')
|
||||||
|
. '/issues/'
|
||||||
|
. $issue_number
|
||||||
|
. '#'
|
||||||
|
. $comment_id;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function extractIssueEvent($issueNbr, $title, $comment){
|
||||||
|
|
||||||
|
$uri = $this->buildGitHubIssueCommentUri($issueNbr, $comment->id);
|
||||||
|
|
||||||
$author = $comment->find('.author', 0)->plaintext;
|
$author = $comment->find('.author', 0)->plaintext;
|
||||||
|
|
||||||
|
@ -94,22 +105,21 @@ class GithubIssueBridge extends BridgeAbstract {
|
||||||
return $item;
|
return $item;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected function extractIssueComment($issueNbr, $title, $comment){
|
private function extractIssueComment($issueNbr, $title, $comment){
|
||||||
$uri = static::URI . $this->getInput('u') . '/'
|
|
||||||
. $this->getInput('p') . '/issues/' . $issueNbr;
|
$uri = $this->buildGitHubIssueCommentUri($issueNbr, $comment->parent->id);
|
||||||
|
|
||||||
$author = $comment->find('.author', 0)->plaintext;
|
$author = $comment->find('.author', 0)->plaintext;
|
||||||
|
|
||||||
$title .= ' / ' . trim(
|
$title .= ' / ' . trim(
|
||||||
$comment->find('.comment .timeline-comment-header-text', 0)->plaintext
|
$comment->find('.timeline-comment-header-text', 0)->plaintext
|
||||||
);
|
);
|
||||||
|
|
||||||
$content = $comment->find('.comment-body', 0)->innertext;
|
$content = $comment->find('.comment-body', 0)->innertext;
|
||||||
|
|
||||||
$item = array();
|
$item = array();
|
||||||
$item['author'] = $author;
|
$item['author'] = $author;
|
||||||
$item['uri'] = $uri
|
$item['uri'] = $uri;
|
||||||
. '#' . $comment->firstChild()->nextSibling()->getAttribute('id');
|
|
||||||
$item['title'] = html_entity_decode($title, ENT_QUOTES, 'UTF-8');
|
$item['title'] = html_entity_decode($title, ENT_QUOTES, 'UTF-8');
|
||||||
$item['timestamp'] = strtotime(
|
$item['timestamp'] = strtotime(
|
||||||
$comment->find('relative-time', 0)->getAttribute('datetime')
|
$comment->find('relative-time', 0)->getAttribute('datetime')
|
||||||
|
@ -118,25 +128,32 @@ class GithubIssueBridge extends BridgeAbstract {
|
||||||
return $item;
|
return $item;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected function extractIssueComments($issue){
|
private function extractIssueComments($issue){
|
||||||
$items = array();
|
$items = array();
|
||||||
$title = $issue->find('.gh-header-title', 0)->plaintext;
|
$title = $issue->find('.gh-header-title', 0)->plaintext;
|
||||||
$issueNbr = trim(
|
$issueNbr = trim(
|
||||||
substr($issue->find('.gh-header-number', 0)->plaintext, 1)
|
substr($issue->find('.gh-header-number', 0)->plaintext, 1)
|
||||||
);
|
);
|
||||||
$comments = $issue->find('.js-discussion', 0);
|
|
||||||
foreach($comments->children() as $comment) {
|
$comments = $issue->find('
|
||||||
|
[id^="issue-"] > .comment,
|
||||||
|
[id^="issuecomment-"] > .comment,
|
||||||
|
[id^="event-"],
|
||||||
|
[id^="ref-"]
|
||||||
|
');
|
||||||
|
foreach($comments as $comment) {
|
||||||
|
|
||||||
if (!$comment->hasChildNodes()) {
|
if (!$comment->hasChildNodes()) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
$comment = $comment->firstChild();
|
|
||||||
$classes = explode(' ', $comment->getAttribute('class'));
|
if (!$comment->hasClass('discussion-item-header')) {
|
||||||
if (in_array('timeline-comment-wrapper', $classes)) {
|
|
||||||
$item = $this->extractIssueComment($issueNbr, $title, $comment);
|
$item = $this->extractIssueComment($issueNbr, $title, $comment);
|
||||||
$items[] = $item;
|
$items[] = $item;
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
while (in_array('discussion-item', $classes)) {
|
|
||||||
|
while ($comment->hasClass('discussion-item-header')) {
|
||||||
$item = $this->extractIssueEvent($issueNbr, $title, $comment);
|
$item = $this->extractIssueEvent($issueNbr, $title, $comment);
|
||||||
$items[] = $item;
|
$items[] = $item;
|
||||||
$comment = $comment->nextSibling();
|
$comment = $comment->nextSibling();
|
||||||
|
@ -145,6 +162,7 @@ class GithubIssueBridge extends BridgeAbstract {
|
||||||
}
|
}
|
||||||
$classes = explode(' ', $comment->getAttribute('class'));
|
$classes = explode(' ', $comment->getAttribute('class'));
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
return $items;
|
return $items;
|
||||||
}
|
}
|
||||||
|
@ -192,8 +210,13 @@ class GithubIssueBridge extends BridgeAbstract {
|
||||||
ENT_QUOTES,
|
ENT_QUOTES,
|
||||||
'UTF-8'
|
'UTF-8'
|
||||||
);
|
);
|
||||||
$comments = trim($issue->find('.col-5', 0)->plaintext);
|
|
||||||
$item['content'] .= "\n" . 'Comments: ' . ($comments ? $comments : '0');
|
$comment_count = 0;
|
||||||
|
if($span = $issue->find('a[aria-label*="comment"] span', 0)) {
|
||||||
|
$comment_count = $span->plaintext;
|
||||||
|
}
|
||||||
|
|
||||||
|
$item['content'] .= "\n" . 'Comments: ' . $comment_count;
|
||||||
$item['uri'] = self::URI
|
$item['uri'] = self::URI
|
||||||
. $issue->find('.js-navigation-open', 0)->getAttribute('href');
|
. $issue->find('.js-navigation-open', 0)->getAttribute('href');
|
||||||
$this->items[] = $item;
|
$this->items[] = $item;
|
||||||
|
@ -216,4 +239,43 @@ class GithubIssueBridge extends BridgeAbstract {
|
||||||
$item['title'] = preg_replace('/\s+/', ' ', $item['title']);
|
$item['title'] = preg_replace('/\s+/', ' ', $item['title']);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function detectParameters($url) {
|
||||||
|
|
||||||
|
if(filter_var($url, FILTER_VALIDATE_URL, FILTER_FLAG_PATH_REQUIRED) === false
|
||||||
|
|| strpos($url, self::URI) !== 0) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$url_components = parse_url($url);
|
||||||
|
$path_segments = array_values(array_filter(explode('/', $url_components['path'])));
|
||||||
|
|
||||||
|
switch(count($path_segments)) {
|
||||||
|
case 2: { // Project issues
|
||||||
|
list($user, $project) = $path_segments;
|
||||||
|
$show_comments = 'off';
|
||||||
|
} break;
|
||||||
|
case 3: { // Project issues with issue comments
|
||||||
|
if($path_segments[2] !== 'issues') {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
list($user, $project) = $path_segments;
|
||||||
|
$show_comments = 'on';
|
||||||
|
} break;
|
||||||
|
case 4: { // Issue comments
|
||||||
|
list($user, $project, /* issues */, $issue) = $path_segments;
|
||||||
|
} break;
|
||||||
|
default: {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return array(
|
||||||
|
'u' => $user,
|
||||||
|
'p' => $project,
|
||||||
|
'c' => isset($show_comments) ? $show_comments : null,
|
||||||
|
'i' => isset($issue) ? $issue : null,
|
||||||
|
);
|
||||||
|
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -141,7 +141,7 @@ class GlassdoorBridge extends BridgeAbstract {
|
||||||
}
|
}
|
||||||
|
|
||||||
private function collectReviewData($html, $limit) {
|
private function collectReviewData($html, $limit) {
|
||||||
$reviews = $html->find('#EmployerReviews li[id^="empReview]')
|
$reviews = $html->find('#ReviewsFeed li[id^="empReview]')
|
||||||
or returnServerError('Unable to find reviews!');
|
or returnServerError('Unable to find reviews!');
|
||||||
|
|
||||||
foreach($reviews as $review) {
|
foreach($reviews as $review) {
|
||||||
|
@ -153,7 +153,19 @@ class GlassdoorBridge extends BridgeAbstract {
|
||||||
$item['timestamp'] = strtotime($review->find('time', 0)->datetime);
|
$item['timestamp'] = strtotime($review->find('time', 0)->datetime);
|
||||||
|
|
||||||
$mainText = $review->find('p.mainText', 0)->plaintext;
|
$mainText = $review->find('p.mainText', 0)->plaintext;
|
||||||
$description = $review->find('div.prosConsAdvice', 0)->innertext;
|
|
||||||
|
$description = '';
|
||||||
|
foreach($review->find('div.description p') as $p) {
|
||||||
|
|
||||||
|
if ($p->hasClass('strong')) {
|
||||||
|
$p->tag = 'strong';
|
||||||
|
$p->removeClass('strong');
|
||||||
|
}
|
||||||
|
|
||||||
|
$description .= $p;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
$item['content'] = "<p>{$mainText}</p><p>{$description}</p>";
|
$item['content'] = "<p>{$mainText}</p><p>{$description}</p>";
|
||||||
|
|
||||||
$this->items[] = $item;
|
$this->items[] = $item;
|
||||||
|
|
206
bridges/GogsBridge.php
Normal file
206
bridges/GogsBridge.php
Normal file
|
@ -0,0 +1,206 @@
|
||||||
|
<?php
|
||||||
|
class GogsBridge extends BridgeAbstract {
|
||||||
|
|
||||||
|
const NAME = 'Gogs';
|
||||||
|
const URI = 'https://gogs.io';
|
||||||
|
const DESCRIPTION = 'Returns the latest issues, commits or releases';
|
||||||
|
const MAINTAINER = 'logmanoriginal';
|
||||||
|
const CACHE_TIMEOUT = 300; // 5 minutes
|
||||||
|
|
||||||
|
const PARAMETERS = array(
|
||||||
|
'global' => array(
|
||||||
|
'host' => array(
|
||||||
|
'name' => 'Host',
|
||||||
|
'exampleValue' => 'https://gogs.io',
|
||||||
|
'required' => true,
|
||||||
|
'title' => 'Host name without trailing slash',
|
||||||
|
),
|
||||||
|
'user' => array(
|
||||||
|
'name' => 'Username',
|
||||||
|
'exampleValue' => 'gogs',
|
||||||
|
'required' => true,
|
||||||
|
'title' => 'User name as it appears in the URL',
|
||||||
|
),
|
||||||
|
'project' => array(
|
||||||
|
'name' => 'Project name',
|
||||||
|
'exampleValue' => 'gogs',
|
||||||
|
'required' => true,
|
||||||
|
'title' => 'Project name as it appears in the URL',
|
||||||
|
),
|
||||||
|
),
|
||||||
|
'Commits' => array(
|
||||||
|
'branch' => array(
|
||||||
|
'name' => 'Branch name',
|
||||||
|
'defaultValue' => 'master',
|
||||||
|
'required' => true,
|
||||||
|
'title' => 'Branch name as it appears in the URL',
|
||||||
|
),
|
||||||
|
),
|
||||||
|
'Issues' => array(
|
||||||
|
'include_description' => array(
|
||||||
|
'name' => 'Include issue description',
|
||||||
|
'type' => 'checkbox',
|
||||||
|
'title' => 'Activate to include the issue description',
|
||||||
|
),
|
||||||
|
),
|
||||||
|
'Single issue' => array(
|
||||||
|
'issue' => array(
|
||||||
|
'name' => 'Issue number',
|
||||||
|
'type' => 'number',
|
||||||
|
'exampleValue' => 102,
|
||||||
|
'required' => true,
|
||||||
|
'title' => 'Issue number from the issues list',
|
||||||
|
),
|
||||||
|
),
|
||||||
|
'Releases' => array(),
|
||||||
|
);
|
||||||
|
|
||||||
|
private $title = '';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Note: detectParamters doesn't make sense for this bridge because there is
|
||||||
|
* no "single" host for this service. Anyone can host it.
|
||||||
|
*/
|
||||||
|
|
||||||
|
public function getURI() {
|
||||||
|
switch($this->queriedContext) {
|
||||||
|
case 'Commits': {
|
||||||
|
return $this->getInput('host')
|
||||||
|
. '/' . $this->getInput('user')
|
||||||
|
. '/' . $this->getInput('project')
|
||||||
|
. '/commits/' . $this->getInput('branch');
|
||||||
|
} break;
|
||||||
|
case 'Issues': {
|
||||||
|
return $this->getInput('host')
|
||||||
|
. '/' . $this->getInput('user')
|
||||||
|
. '/' . $this->getInput('project')
|
||||||
|
. '/issues/';
|
||||||
|
} break;
|
||||||
|
case 'Single issue': {
|
||||||
|
return $this->getInput('host')
|
||||||
|
. '/' . $this->getInput('user')
|
||||||
|
. '/' . $this->getInput('project')
|
||||||
|
. '/issues/' . $this->getInput('issue');
|
||||||
|
} break;
|
||||||
|
case 'Releases': {
|
||||||
|
return $this->getInput('host')
|
||||||
|
. '/' . $this->getInput('user')
|
||||||
|
. '/' . $this->getInput('project')
|
||||||
|
. '/releases/';
|
||||||
|
} break;
|
||||||
|
default: return parent::getURI();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getName() {
|
||||||
|
switch($this->queriedContext) {
|
||||||
|
case 'Commits':
|
||||||
|
case 'Issues':
|
||||||
|
case 'Releases': return $this->title . ' ' . $this->queriedContext;
|
||||||
|
case 'Single issue': return $this->title . ' Issue ' . $this->getInput('issue');
|
||||||
|
default: return parent::getName();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getIcon() {
|
||||||
|
return 'https://gogs.io/img/favicon.ico';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function collectData() {
|
||||||
|
|
||||||
|
$html = getSimpleHTMLDOM($this->getURI())
|
||||||
|
or returnServerError('Could not request ' . $this->getURI());
|
||||||
|
|
||||||
|
$html = defaultLinkTo($html, $this->getURI());
|
||||||
|
|
||||||
|
$this->title = $html->find('[property="og:title"]', 0)->content;
|
||||||
|
|
||||||
|
switch($this->queriedContext) {
|
||||||
|
case 'Commits': {
|
||||||
|
$this->collectCommitsData($html);
|
||||||
|
} break;
|
||||||
|
case 'Issues': {
|
||||||
|
$this->collectIssuesData($html);
|
||||||
|
} break;
|
||||||
|
case 'Single issue': {
|
||||||
|
$this->collectSingleIssueData($html);
|
||||||
|
} break;
|
||||||
|
case 'Releases': {
|
||||||
|
$this->collectReleasesData($html);
|
||||||
|
} break;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function collectCommitsData($html) {
|
||||||
|
$commits = $html->find('#commits-table tbody tr')
|
||||||
|
or returnServerError('Unable to find commits');
|
||||||
|
|
||||||
|
foreach($commits as $commit) {
|
||||||
|
$this->items[] = array(
|
||||||
|
'uri' => $commit->find('a.sha', 0)->href,
|
||||||
|
'title' => $commit->find('.message span', 0)->plaintext,
|
||||||
|
'author' => $commit->find('.author', 0)->plaintext,
|
||||||
|
'timestamp' => $commit->find('.time-since', 0)->title,
|
||||||
|
'uid' => $commit->find('.sha', 0)->plaintext,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function collectIssuesData($html) {
|
||||||
|
$issues = $html->find('.issue.list li')
|
||||||
|
or returnServerError('Unable to find issues');
|
||||||
|
|
||||||
|
foreach($issues as $issue) {
|
||||||
|
$uri = $issue->find('a', 0)->href;
|
||||||
|
|
||||||
|
$item = array(
|
||||||
|
'uri' => $uri,
|
||||||
|
'title' => $issue->find('.label', 0)->plaintext . ' | ' . $issue->find('a.title', 0)->plaintext,
|
||||||
|
'author' => $issue->find('.desc a', 0)->plaintext,
|
||||||
|
'timestamp' => $issue->find('.time-since', 0)->title,
|
||||||
|
'uid' => $issue->find('.label', 0)->plaintext,
|
||||||
|
);
|
||||||
|
|
||||||
|
if($this->getInput('include_description')) {
|
||||||
|
$issue_html = getSimpleHTMLDOMCached($uri, 3600)
|
||||||
|
or returnServerError('Unable to load issue description');
|
||||||
|
|
||||||
|
$issue_html = defaultLinkTo($issue_html, $uri);
|
||||||
|
|
||||||
|
$item['content'] = $issue_html->find('.comment .markdown', 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->items[] = $item;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function collectSingleIssueData($html) {
|
||||||
|
$comments = $html->find('.comments .comment')
|
||||||
|
or returnServerError('Unable to find comments');
|
||||||
|
|
||||||
|
foreach($comments as $comment) {
|
||||||
|
$this->items[] = array(
|
||||||
|
'uri' => $comment->find('a[href*="#issue"]', 0)->href,
|
||||||
|
'title' => $comment->find('span', 0)->plaintext,
|
||||||
|
'author' => $comment->find('.content a', 0)->plaintext,
|
||||||
|
'timestamp' => $comment->find('.time-since', 0)->title,
|
||||||
|
'content' => $comment->find('.markdown', 0),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->items = array_reverse($this->items);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function collectReleasesData($html) {
|
||||||
|
$releases = $html->find('#release-list li')
|
||||||
|
or returnServerError('Unable to find releases');
|
||||||
|
|
||||||
|
foreach($releases as $release) {
|
||||||
|
$this->items[] = array(
|
||||||
|
'uri' => $release->find('a', 0)->href,
|
||||||
|
'title' => 'Release ' . $release->find('h4', 0)->plaintext,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -25,13 +25,10 @@ class GoogleSearchBridge extends BridgeAbstract {
|
||||||
public function collectData(){
|
public function collectData(){
|
||||||
$html = '';
|
$html = '';
|
||||||
|
|
||||||
$html = getSimpleHTMLDOM(self::URI
|
$html = getSimpleHTMLDOM($this->getURI())
|
||||||
. 'search?q='
|
|
||||||
. urlencode($this->getInput('q'))
|
|
||||||
. '&num=100&complete=0&tbs=qdr:y,sbd:1')
|
|
||||||
or returnServerError('No results for this query.');
|
or returnServerError('No results for this query.');
|
||||||
|
|
||||||
$emIsRes = $html->find('div[id=ires]', 0);
|
$emIsRes = $html->find('div[id=res]', 0);
|
||||||
|
|
||||||
if(!is_null($emIsRes)) {
|
if(!is_null($emIsRes)) {
|
||||||
foreach($emIsRes->find('div[class=g]') as $element) {
|
foreach($emIsRes->find('div[class=g]') as $element) {
|
||||||
|
@ -54,6 +51,17 @@ class GoogleSearchBridge extends BridgeAbstract {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function getURI() {
|
||||||
|
if (!is_null($this->getInput('q'))) {
|
||||||
|
return self::URI
|
||||||
|
. 'search?q='
|
||||||
|
. urlencode($this->getInput('q'))
|
||||||
|
. '&num=100&complete=0&tbs=qdr:y,sbd:1';
|
||||||
|
}
|
||||||
|
|
||||||
|
return parent::getURI();
|
||||||
|
}
|
||||||
|
|
||||||
public function getName(){
|
public function getName(){
|
||||||
if(!is_null($this->getInput('q'))) {
|
if(!is_null($this->getInput('q'))) {
|
||||||
return $this->getInput('q') . ' - Google search';
|
return $this->getInput('q') . ' - Google search';
|
||||||
|
|
|
@ -2,7 +2,7 @@
|
||||||
class HDWallpapersBridge extends BridgeAbstract {
|
class HDWallpapersBridge extends BridgeAbstract {
|
||||||
const MAINTAINER = 'nel50n';
|
const MAINTAINER = 'nel50n';
|
||||||
const NAME = 'HD Wallpapers Bridge';
|
const NAME = 'HD Wallpapers Bridge';
|
||||||
const URI = 'http://www.hdwallpapers.in/';
|
const URI = 'https://www.hdwallpapers.in/';
|
||||||
const CACHE_TIMEOUT = 43200; //12h
|
const CACHE_TIMEOUT = 43200; //12h
|
||||||
const DESCRIPTION = 'Returns the latests wallpapers from HDWallpapers';
|
const DESCRIPTION = 'Returns the latests wallpapers from HDWallpapers';
|
||||||
|
|
||||||
|
@ -72,7 +72,7 @@ class HDWallpapersBridge extends BridgeAbstract {
|
||||||
public function getName(){
|
public function getName(){
|
||||||
if(!is_null($this->getInput('c')) && !is_null($this->getInput('r'))) {
|
if(!is_null($this->getInput('c')) && !is_null($this->getInput('r'))) {
|
||||||
return 'HDWallpapers - '
|
return 'HDWallpapers - '
|
||||||
. str_replace(['__', '_'], [' & ', ' '], $this->getInput('c'))
|
. str_replace(array('__', '_'), array(' & ', ' '), $this->getInput('c'))
|
||||||
. ' ['
|
. ' ['
|
||||||
. $this->getInput('r')
|
. $this->getInput('r')
|
||||||
. ']';
|
. ']';
|
||||||
|
|
|
@ -13,6 +13,11 @@ class HaveIBeenPwnedBridge extends BridgeAbstract {
|
||||||
'Date added to HIBP' => 'dateAdded',
|
'Date added to HIBP' => 'dateAdded',
|
||||||
),
|
),
|
||||||
'defaultValue' => 'dateAdded',
|
'defaultValue' => 'dateAdded',
|
||||||
|
),
|
||||||
|
'item_limit' => array(
|
||||||
|
'name' => 'Limit number of returned items',
|
||||||
|
'type' => 'number',
|
||||||
|
'defaultValue' => 20,
|
||||||
)
|
)
|
||||||
));
|
));
|
||||||
|
|
||||||
|
@ -52,13 +57,15 @@ class HaveIBeenPwnedBridge extends BridgeAbstract {
|
||||||
// Remove permalink
|
// Remove permalink
|
||||||
$breach->find('p', 1)->find('a', 0)->outertext = '';
|
$breach->find('p', 1)->find('a', 0)->outertext = '';
|
||||||
|
|
||||||
$item['title'] = $breach->find('h3', 0)->plaintext . ' - ' . $accounts[1] . ' breached accounts';
|
$item['title'] = html_entity_decode($breach->find('h3', 0)->plaintext, ENT_QUOTES)
|
||||||
|
. ' - ' . $accounts[1] . ' breached accounts';
|
||||||
$item['dateAdded'] = strtotime($dateAdded[1]);
|
$item['dateAdded'] = strtotime($dateAdded[1]);
|
||||||
$item['breachDate'] = strtotime($breachDate[1]);
|
$item['breachDate'] = strtotime($breachDate[1]);
|
||||||
$item['uri'] = self::URI . '/PwnedWebsites' . $permalink;
|
$item['uri'] = self::URI . '/PwnedWebsites' . $permalink;
|
||||||
|
|
||||||
$item['content'] = '<p>' . $breach->find('p', 0)->innertext . '<p>';
|
$item['content'] = '<p>' . $breach->find('p', 0)->innertext . '</p>';
|
||||||
$item['content'] .= '<p>' . $breach->find('p', 1)->innertext . '<p>';
|
$item['content'] .= '<p>' . $this->breachType($breach) . '</p>';
|
||||||
|
$item['content'] .= '<p>' . $breach->find('p', 1)->innertext . '</p>';
|
||||||
|
|
||||||
$this->breaches[] = $item;
|
$this->breaches[] = $item;
|
||||||
}
|
}
|
||||||
|
@ -67,6 +74,25 @@ class HaveIBeenPwnedBridge extends BridgeAbstract {
|
||||||
$this->createItems();
|
$this->createItems();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Extract data breach type(s)
|
||||||
|
*/
|
||||||
|
private function breachType($breach) {
|
||||||
|
|
||||||
|
$content = '';
|
||||||
|
|
||||||
|
if ($breach->find('h3 > i', 0)) {
|
||||||
|
|
||||||
|
foreach ($breach->find('h3 > i') as $i) {
|
||||||
|
$content .= $i->title . '.<br>';
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
return $content;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Order Breaches by date added or date breached
|
* Order Breaches by date added or date breached
|
||||||
*/
|
*/
|
||||||
|
@ -88,6 +114,12 @@ class HaveIBeenPwnedBridge extends BridgeAbstract {
|
||||||
*/
|
*/
|
||||||
private function createItems() {
|
private function createItems() {
|
||||||
|
|
||||||
|
$limit = $this->getInput('item_limit');
|
||||||
|
|
||||||
|
if ($limit < 1) {
|
||||||
|
$limit = 20;
|
||||||
|
}
|
||||||
|
|
||||||
foreach ($this->breaches as $breach) {
|
foreach ($this->breaches as $breach) {
|
||||||
$item = array();
|
$item = array();
|
||||||
|
|
||||||
|
@ -97,6 +129,10 @@ class HaveIBeenPwnedBridge extends BridgeAbstract {
|
||||||
$item['content'] = $breach['content'];
|
$item['content'] = $breach['content'];
|
||||||
|
|
||||||
$this->items[] = $item;
|
$this->items[] = $item;
|
||||||
|
|
||||||
|
if (count($this->items) >= $limit) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,7 +3,7 @@ class HentaiHavenBridge extends BridgeAbstract {
|
||||||
|
|
||||||
const MAINTAINER = 'albirew';
|
const MAINTAINER = 'albirew';
|
||||||
const NAME = 'Hentai Haven';
|
const NAME = 'Hentai Haven';
|
||||||
const URI = 'http://hentaihaven.org/';
|
const URI = 'https://hentaihaven.org/';
|
||||||
const CACHE_TIMEOUT = 21600; // 6h
|
const CACHE_TIMEOUT = 21600; // 6h
|
||||||
const DESCRIPTION = 'Returns releases from Hentai Haven';
|
const DESCRIPTION = 'Returns releases from Hentai Haven';
|
||||||
|
|
||||||
|
|
55
bridges/IGNBridge.php
Normal file
55
bridges/IGNBridge.php
Normal file
|
@ -0,0 +1,55 @@
|
||||||
|
<?php
|
||||||
|
class IGNBridge extends FeedExpander {
|
||||||
|
|
||||||
|
const MAINTAINER = 'IceWreck';
|
||||||
|
const NAME = 'IGN Bridge';
|
||||||
|
const URI = 'https://www.ign.com/';
|
||||||
|
const CACHE_TIMEOUT = 3600;
|
||||||
|
const DESCRIPTION = 'RSS Feed For IGN';
|
||||||
|
|
||||||
|
public function collectData(){
|
||||||
|
$this->collectExpandableDatas('http://feeds.ign.com/ign/all', 15);
|
||||||
|
}
|
||||||
|
|
||||||
|
// IGNs feed is both hidden and incomplete. This bridge tries to fix this.
|
||||||
|
|
||||||
|
protected function parseItem($newsItem){
|
||||||
|
$item = parent::parseItem($newsItem);
|
||||||
|
|
||||||
|
// $articlePage gets the entire page's contents
|
||||||
|
$articlePage = getSimpleHTMLDOM($newsItem->link);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* NOTE: Though articles and wiki/howtos have seperate styles of pages, there is no mechanism
|
||||||
|
* for handling them seperately as it just ignores the DOM querys which it does not find.
|
||||||
|
* (and their scraping)
|
||||||
|
*/
|
||||||
|
|
||||||
|
// For Articles
|
||||||
|
$article = $articlePage->find('section.article-page', 0);
|
||||||
|
// add in verdicts in articles, reviews etc
|
||||||
|
foreach($articlePage->find('div.article-section') as $element) {
|
||||||
|
$article = $article . $element;
|
||||||
|
}
|
||||||
|
|
||||||
|
// For Wikis and HowTos
|
||||||
|
$uselessWikiElements = array(
|
||||||
|
'.wiki-page-tools',
|
||||||
|
'.feedback-container',
|
||||||
|
'.paging-container'
|
||||||
|
);
|
||||||
|
foreach($articlePage->find('.wiki-page') as $wikiContents) {
|
||||||
|
$copy = clone $wikiContents;
|
||||||
|
// Remove useless elements present in IGN wiki/howtos
|
||||||
|
foreach($uselessWikiElements as $uslElement) {
|
||||||
|
$toRemove = $wikiContents->find($uslElement, 0);
|
||||||
|
$copy = str_replace($toRemove, '', $copy);
|
||||||
|
}
|
||||||
|
$article = $article . $copy;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add content to feed
|
||||||
|
$item['content'] = $article;
|
||||||
|
return $item;
|
||||||
|
}
|
||||||
|
}
|
245
bridges/IndeedBridge.php
Normal file
245
bridges/IndeedBridge.php
Normal file
|
@ -0,0 +1,245 @@
|
||||||
|
<?php
|
||||||
|
class IndeedBridge extends BridgeAbstract {
|
||||||
|
|
||||||
|
const NAME = 'Indeed';
|
||||||
|
const URI = 'https://www.indeed.com/';
|
||||||
|
const DESCRIPTION = 'Returns reviews and comments for a company of your choice';
|
||||||
|
const MAINTAINER = 'logmanoriginal';
|
||||||
|
const CACHE_TIMEOUT = 14400; // 4 hours
|
||||||
|
|
||||||
|
const PARAMETERS = array(
|
||||||
|
array(
|
||||||
|
'c' => array(
|
||||||
|
'name' => 'Company',
|
||||||
|
'type' => 'text',
|
||||||
|
'required' => true,
|
||||||
|
'title' => 'Company name',
|
||||||
|
'exampleValue' => 'GitHub',
|
||||||
|
)
|
||||||
|
),
|
||||||
|
'global' => array(
|
||||||
|
'language' => array(
|
||||||
|
'name' => 'Language Code',
|
||||||
|
'type' => 'list',
|
||||||
|
'title' => 'Choose your language code',
|
||||||
|
'defaultValue' => 'en-US',
|
||||||
|
'values' => array(
|
||||||
|
'es-AR' => 'es-AR',
|
||||||
|
'de-AT' => 'de-AT',
|
||||||
|
'en-AU' => 'en-AU',
|
||||||
|
'nl-BE' => 'nl-BE',
|
||||||
|
'fr-BE' => 'fr-BE',
|
||||||
|
'pt-BR' => 'pt-BR',
|
||||||
|
'en-CA' => 'en-CA',
|
||||||
|
'fr-CA' => 'fr-CA',
|
||||||
|
'de-CH' => 'de-CH',
|
||||||
|
'fr-CH' => 'fr-CH',
|
||||||
|
'es-CL' => 'es-CL',
|
||||||
|
'zh-CN' => 'zh-CN',
|
||||||
|
'es-CO' => 'es-CO',
|
||||||
|
'de-DE' => 'de-DE',
|
||||||
|
'es-ES' => 'es-ES',
|
||||||
|
'fr-FR' => 'fr-FR',
|
||||||
|
'en-GB' => 'en-GB',
|
||||||
|
'en-HK' => 'en-HK',
|
||||||
|
'en-IE' => 'en-IE',
|
||||||
|
'en-IN' => 'en-IN',
|
||||||
|
'it-IT' => 'it-IT',
|
||||||
|
'ja-JP' => 'ja-JP',
|
||||||
|
'ko-KR' => 'ko-KR',
|
||||||
|
'es-MX' => 'es-MX',
|
||||||
|
'nl-NL' => 'nl-NL',
|
||||||
|
'pl-PL' => 'pl-PL',
|
||||||
|
'en-SG' => 'en-SG',
|
||||||
|
'en-US' => 'en-US',
|
||||||
|
'en-ZA' => 'en-ZA',
|
||||||
|
'en-AE' => 'en-AE',
|
||||||
|
'da-DK' => 'da-DK',
|
||||||
|
'in-ID' => 'in-ID',
|
||||||
|
'en-MY' => 'en-MY',
|
||||||
|
'es-PE' => 'es-PE',
|
||||||
|
'en-PH' => 'en-PH',
|
||||||
|
'en-PK' => 'en-PK',
|
||||||
|
'ro-RO' => 'ro-RO',
|
||||||
|
'ru-RU' => 'ru-RU',
|
||||||
|
'tr-TR' => 'tr-TR',
|
||||||
|
'zh-TW' => 'zh-TW',
|
||||||
|
'vi-VN' => 'vi-VN',
|
||||||
|
'en-VN' => 'en-VN',
|
||||||
|
'ar-EG' => 'ar-EG',
|
||||||
|
'fr-MA' => 'fr-MA',
|
||||||
|
'en-NG' => 'en-NG',
|
||||||
|
)
|
||||||
|
),
|
||||||
|
'limit' => array(
|
||||||
|
'name' => 'Limit',
|
||||||
|
'type' => 'number',
|
||||||
|
'title' => 'Maximum number of items to return',
|
||||||
|
'exampleValue' => 20,
|
||||||
|
)
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
const SITES = array(
|
||||||
|
'es-AR' => 'https://ar.indeed.com/',
|
||||||
|
'de-AT' => 'https://at.indeed.com/',
|
||||||
|
'en-AU' => 'https://au.indeed.com/',
|
||||||
|
'nl-BE' => 'https://be.indeed.com/',
|
||||||
|
'fr-BE' => 'https://emplois.be.indeed.com/',
|
||||||
|
'pt-BR' => 'https://www.indeed.com.br/',
|
||||||
|
'en-CA' => 'https://ca.indeed.com/',
|
||||||
|
'fr-CA' => 'https://emplois.ca.indeed.com/',
|
||||||
|
'de-CH' => 'https://www.indeed.ch/',
|
||||||
|
'fr-CH' => 'https://emplois.indeed.ch/',
|
||||||
|
'es-CL' => 'https://www.indeed.cl/',
|
||||||
|
'zh-CN' => 'https://cn.indeed.com/',
|
||||||
|
'es-CO' => 'https://co.indeed.com/',
|
||||||
|
'de-DE' => 'https://de.indeed.com/',
|
||||||
|
'es-ES' => 'https://www.indeed.es/',
|
||||||
|
'fr-FR' => 'https://www.indeed.fr/',
|
||||||
|
'en-GB' => 'https://www.indeed.co.uk/',
|
||||||
|
'en-HK' => 'https://www.indeed.hk/',
|
||||||
|
'en-IE' => 'https://ie.indeed.com/',
|
||||||
|
'en-IN' => 'https://www.indeed.co.in/',
|
||||||
|
'it-IT' => 'https://it.indeed.com/',
|
||||||
|
'ja-JP' => 'https://jp.indeed.com/',
|
||||||
|
'ko-KR' => 'https://kr.indeed.com/',
|
||||||
|
'es-MX' => 'https://www.indeed.com.mx/',
|
||||||
|
'nl-NL' => 'https://www.indeed.nl/',
|
||||||
|
'pl-PL' => 'https://pl.indeed.com/',
|
||||||
|
'en-SG' => 'https://www.indeed.com.sg/',
|
||||||
|
'en-US' => 'https://www.indeed.com/',
|
||||||
|
'en-ZA' => 'https://www.indeed.co.za/',
|
||||||
|
'en-AE' => 'https://www.indeed.ae/',
|
||||||
|
'da-DK' => 'https://dk.indeed.com/',
|
||||||
|
'in-ID' => 'https://id.indeed.com/',
|
||||||
|
'en-MY' => 'https://www.indeed.com.my/',
|
||||||
|
'es-PE' => 'https://www.indeed.com.pe/',
|
||||||
|
'en-PH' => 'https://www.indeed.com.ph/',
|
||||||
|
'en-PK' => 'https://www.indeed.com.pk/',
|
||||||
|
'ro-RO' => 'https://ro.indeed.com/',
|
||||||
|
'ru-RU' => 'https://ru.indeed.com/',
|
||||||
|
'tr-TR' => 'https://tr.indeed.com/',
|
||||||
|
'zh-TW' => 'https://tw.indeed.com/',
|
||||||
|
'vi-VN' => 'https://vn.indeed.com/',
|
||||||
|
'en-VN' => 'https://jobs.vn.indeed.com/',
|
||||||
|
'ar-EG' => 'https://eg.indeed.com/',
|
||||||
|
'fr-MA' => 'https://ma.indeed.com/',
|
||||||
|
'en-NG' => 'https://ng.indeed.com/',
|
||||||
|
);
|
||||||
|
|
||||||
|
private $title;
|
||||||
|
|
||||||
|
public function collectData() {
|
||||||
|
|
||||||
|
$url = $this->getURI();
|
||||||
|
$limit = $this->getInput('limit') ?: 20;
|
||||||
|
|
||||||
|
do {
|
||||||
|
|
||||||
|
$html = getSimpleHTMLDOM($url)
|
||||||
|
or returnServerError('Could not request ' . $url);
|
||||||
|
|
||||||
|
$html = defaultLinkTo($html, $url);
|
||||||
|
|
||||||
|
$this->title = $html->find('h1', 0)->innertext;
|
||||||
|
|
||||||
|
// Use local translation of the word "Rating"
|
||||||
|
$rating_local = $html->find('a[data-id="rating_desc"]', 0)->plaintext;
|
||||||
|
|
||||||
|
foreach($html->find('#cmp-content [id^="cmp-review-"]') as $review) {
|
||||||
|
$item = array();
|
||||||
|
|
||||||
|
$rating = $review->find('.cmp-ratingNumber', 0)->plaintext;
|
||||||
|
$title = $review->find('.cmp-review-title > span', 0)->plaintext;
|
||||||
|
$comment = $this->beautifyComment($review->find('.cmp-review-content-container', 0));
|
||||||
|
|
||||||
|
$item['uri'] = $review->find('.cmp-review-share-popup-item-link--copylink', 0)->href;
|
||||||
|
$item['title'] = "{$rating_local} {$rating} / {$title}";
|
||||||
|
$item['timestamp'] = $review->find('.cmp-review-date-created', 0)->plaintext;
|
||||||
|
$item['author'] = $review->find('.cmp-reviewer', 0)->plaintext;
|
||||||
|
$item['content'] = $comment;
|
||||||
|
//$item['enclosures']
|
||||||
|
$item['categories'][] = $review->find('.cmp-reviewer-job-location', 0)->plaintext;
|
||||||
|
//$item['uid']
|
||||||
|
|
||||||
|
$this->items[] = $item;
|
||||||
|
|
||||||
|
if(count($this->items) >= $limit) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Break if no more pages available.
|
||||||
|
if($next = $html->find('a[data-tn-element="next-page"]', 0)) {
|
||||||
|
$url = $next->href;
|
||||||
|
} else {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
} while(count($this->items) < $limit);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getURI() {
|
||||||
|
if($this->getInput('language')
|
||||||
|
&& $this->getInput('c')) {
|
||||||
|
return self::SITES[$this->getInput('language')]
|
||||||
|
. 'cmp/'
|
||||||
|
. urlencode($this->getInput('c'))
|
||||||
|
. '/reviews';
|
||||||
|
}
|
||||||
|
|
||||||
|
return parent::getURI();
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getName() {
|
||||||
|
return $this->title ?: parent::getName();
|
||||||
|
}
|
||||||
|
|
||||||
|
public function detectParameters($url) {
|
||||||
|
/**
|
||||||
|
* Expected: https://<...>.indeed.<...>/cmp/<company>[/reviews][/...]
|
||||||
|
*
|
||||||
|
* Note that most users will be redirected to their localized version
|
||||||
|
* of the page, which adds the language code to the host. For example,
|
||||||
|
* "en.indeed.com" or "www.indeed.fr" (see link[rel="alternate"]). At
|
||||||
|
* least each of the sites have ".indeed." in the name.
|
||||||
|
*/
|
||||||
|
|
||||||
|
if(filter_var($url, FILTER_VALIDATE_URL, FILTER_FLAG_PATH_REQUIRED) === false
|
||||||
|
|| stristr($url, '.indeed.') === false) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$url_components = parse_url($url);
|
||||||
|
$path_segments = array_values(array_filter(explode('/', $url_components['path'])));
|
||||||
|
|
||||||
|
if(count($path_segments) < 2 || $path_segments[0] !== 'cmp') {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$language = array_search('https://' . $url_components['host'] . '/', self::SITES);
|
||||||
|
if($language === false) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$limit = self::PARAMETERS['global']['limit']['defaultValue'] ?: 20;
|
||||||
|
$company = $path_segments[1];
|
||||||
|
|
||||||
|
return array(
|
||||||
|
'c' => $company,
|
||||||
|
'language' => $language,
|
||||||
|
'limit' => $limit,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
private function beautifyComment($comment) {
|
||||||
|
foreach($comment->find('.cmp-bold') as $bold) {
|
||||||
|
$bold->tag = 'strong';
|
||||||
|
$bold->removeClass('cmp-bold');
|
||||||
|
}
|
||||||
|
|
||||||
|
return $comment;
|
||||||
|
}
|
||||||
|
}
|
|
@ -32,28 +32,61 @@ class InstagramBridge extends BridgeAbstract {
|
||||||
'required' => false,
|
'required' => false,
|
||||||
'values' => array(
|
'values' => array(
|
||||||
'All' => 'all',
|
'All' => 'all',
|
||||||
'Story' => 'story',
|
|
||||||
'Video' => 'video',
|
'Video' => 'video',
|
||||||
'Picture' => 'picture',
|
'Picture' => 'picture',
|
||||||
|
'Multiple' => 'multiple',
|
||||||
),
|
),
|
||||||
'defaultValue' => 'all'
|
'defaultValue' => 'all'
|
||||||
|
),
|
||||||
|
'direct_links' => array(
|
||||||
|
'name' => 'Use direct media links',
|
||||||
|
'type' => 'checkbox',
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
|
|
||||||
);
|
);
|
||||||
|
|
||||||
public function collectData(){
|
const USER_QUERY_HASH = '58b6785bea111c67129decbe6a448951';
|
||||||
|
const TAG_QUERY_HASH = '174a5243287c5f3a7de741089750ab3b';
|
||||||
|
const SHORTCODE_QUERY_HASH = '865589822932d1b43dfe312121dd353a';
|
||||||
|
|
||||||
if(is_null($this->getInput('u')) && $this->getInput('media_type') == 'story') {
|
protected function getInstagramUserId($username) {
|
||||||
returnClientError('Stories are not supported for hashtags nor locations!');
|
|
||||||
|
if(is_numeric($username)) return $username;
|
||||||
|
|
||||||
|
$cacheFac = new CacheFactory();
|
||||||
|
$cacheFac->setWorkingDir(PATH_LIB_CACHES);
|
||||||
|
$cache = $cacheFac->create(Configuration::getConfig('cache', 'type'));
|
||||||
|
$cache->setScope(get_called_class());
|
||||||
|
$cache->setKey(array($username));
|
||||||
|
$key = $cache->loadData();
|
||||||
|
|
||||||
|
if($key == null) {
|
||||||
|
$data = getContents(self::URI . 'web/search/topsearch/?query=' . $username);
|
||||||
|
|
||||||
|
foreach(json_decode($data)->users as $user) {
|
||||||
|
if($user->user->username === $username) {
|
||||||
|
$key = $user->user->pk;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
if($key == null) {
|
||||||
|
returnServerError('Unable to find username in search result.');
|
||||||
|
}
|
||||||
|
$cache->saveData($key);
|
||||||
|
}
|
||||||
|
return $key;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public function collectData(){
|
||||||
|
$directLink = !is_null($this->getInput('direct_links')) && $this->getInput('direct_links');
|
||||||
|
|
||||||
$data = $this->getInstagramJSON($this->getURI());
|
$data = $this->getInstagramJSON($this->getURI());
|
||||||
|
|
||||||
if(!is_null($this->getInput('u'))) {
|
if(!is_null($this->getInput('u'))) {
|
||||||
$userMedia = $data->entry_data->ProfilePage[0]->graphql->user->edge_owner_to_timeline_media->edges;
|
$userMedia = $data->data->user->edge_owner_to_timeline_media->edges;
|
||||||
} elseif(!is_null($this->getInput('h'))) {
|
} elseif(!is_null($this->getInput('h'))) {
|
||||||
$userMedia = $data->entry_data->TagPage[0]->graphql->hashtag->edge_hashtag_to_media->edges;
|
$userMedia = $data->data->hashtag->edge_hashtag_to_media->edges;
|
||||||
} elseif(!is_null($this->getInput('l'))) {
|
} elseif(!is_null($this->getInput('l'))) {
|
||||||
$userMedia = $data->entry_data->LocationsPage[0]->graphql->location->edge_location_to_media->edges;
|
$userMedia = $data->entry_data->LocationsPage[0]->graphql->location->edge_location_to_media->edges;
|
||||||
}
|
}
|
||||||
|
@ -61,23 +94,19 @@ class InstagramBridge extends BridgeAbstract {
|
||||||
foreach($userMedia as $media) {
|
foreach($userMedia as $media) {
|
||||||
$media = $media->node;
|
$media = $media->node;
|
||||||
|
|
||||||
if(!is_null($this->getInput('u'))) {
|
|
||||||
switch($this->getInput('media_type')) {
|
switch($this->getInput('media_type')) {
|
||||||
case 'all': break;
|
case 'all': break;
|
||||||
case 'video':
|
case 'video':
|
||||||
if($media->__typename != 'GraphVideo') continue 2;
|
if($media->__typename != 'GraphVideo' || !$media->is_video) continue 2;
|
||||||
break;
|
break;
|
||||||
case 'picture':
|
case 'picture':
|
||||||
if($media->__typename != 'GraphImage') continue 2;
|
if($media->__typename != 'GraphImage') continue 2;
|
||||||
break;
|
break;
|
||||||
case 'story':
|
case 'multiple':
|
||||||
if($media->__typename != 'GraphSidecar') continue 2;
|
if($media->__typename != 'GraphSidecar') continue 2;
|
||||||
break;
|
break;
|
||||||
default: break;
|
default: break;
|
||||||
}
|
}
|
||||||
} else {
|
|
||||||
if($this->getInput('media_type') == 'video' && !$media->is_video) continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
$item = array();
|
$item = array();
|
||||||
$item['uri'] = self::URI . 'p/' . $media->shortcode . '/';
|
$item['uri'] = self::URI . 'p/' . $media->shortcode . '/';
|
||||||
|
@ -86,65 +115,132 @@ class InstagramBridge extends BridgeAbstract {
|
||||||
$item['author'] = $media->owner->username;
|
$item['author'] = $media->owner->username;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (isset($media->edge_media_to_caption->edges[0]->node->text)) {
|
$textContent = $this->getTextContent($media);
|
||||||
$textContent = $media->edge_media_to_caption->edges[0]->node->text;
|
|
||||||
} else {
|
|
||||||
$textContent = '(no text)';
|
|
||||||
}
|
|
||||||
|
|
||||||
$item['title'] = ($media->is_video ? '▶ ' : '') . trim($textContent);
|
$item['title'] = ($media->is_video ? '▶ ' : '') . $textContent;
|
||||||
$titleLinePos = strpos(wordwrap($item['title'], 120), "\n");
|
$titleLinePos = strpos(wordwrap($item['title'], 120), "\n");
|
||||||
if ($titleLinePos != false) {
|
if ($titleLinePos != false) {
|
||||||
$item['title'] = substr($item['title'], 0, $titleLinePos) . '...';
|
$item['title'] = substr($item['title'], 0, $titleLinePos) . '...';
|
||||||
}
|
}
|
||||||
|
|
||||||
if(!is_null($this->getInput('u')) && $media->__typename == 'GraphSidecar') {
|
switch($media->__typename) {
|
||||||
$data = $this->getInstagramStory($item['uri']);
|
case 'GraphSidecar':
|
||||||
|
$data = $this->getInstagramSidecarData($item['uri'], $item['title']);
|
||||||
$item['content'] = $data[0];
|
$item['content'] = $data[0];
|
||||||
$item['enclosures'] = $data[1];
|
$item['enclosures'] = $data[1];
|
||||||
|
break;
|
||||||
|
case 'GraphImage':
|
||||||
|
if($directLink) {
|
||||||
|
$mediaURI = $media->display_url;
|
||||||
} else {
|
} else {
|
||||||
$mediaURI = self::URI . 'p/' . $media->shortcode . '/media?size=l';
|
$mediaURI = self::URI . 'p/' . $media->shortcode . '/media?size=l';
|
||||||
|
}
|
||||||
$item['content'] = '<a href="' . htmlentities($item['uri']) . '" target="_blank">';
|
$item['content'] = '<a href="' . htmlentities($item['uri']) . '" target="_blank">';
|
||||||
$item['content'] .= '<img src="' . htmlentities($mediaURI) . '" alt="' . $item['title'] . '" />';
|
$item['content'] .= '<img src="' . htmlentities($mediaURI) . '" alt="' . $item['title'] . '" />';
|
||||||
$item['content'] .= '</a><br><br>' . nl2br(htmlentities($textContent));
|
$item['content'] .= '</a><br><br>' . nl2br(htmlentities($textContent));
|
||||||
$item['enclosures'] = array($mediaURI);
|
$item['enclosures'] = array($mediaURI);
|
||||||
|
break;
|
||||||
|
case 'GraphVideo':
|
||||||
|
$data = $this->getInstagramVideoData($item['uri']);
|
||||||
|
$item['content'] = $data[0];
|
||||||
|
if($directLink) {
|
||||||
|
$item['enclosures'] = $data[1];
|
||||||
|
} else {
|
||||||
|
$item['enclosures'] = array(self::URI . 'p/' . $media->shortcode . '/media?size=l');
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
default: break;
|
||||||
}
|
}
|
||||||
|
|
||||||
$item['timestamp'] = $media->taken_at_timestamp;
|
$item['timestamp'] = $media->taken_at_timestamp;
|
||||||
|
|
||||||
$this->items[] = $item;
|
$this->items[] = $item;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
protected function getInstagramStory($uri) {
|
// returns Sidecar(a post which has multiple media)'s contents and enclosures
|
||||||
|
protected function getInstagramSidecarData($uri, $postTitle) {
|
||||||
|
$mediaInfo = $this->getSinglePostData($uri);
|
||||||
|
|
||||||
$data = $this->getInstagramJSON($uri);
|
$textContent = $this->getTextContent($mediaInfo);
|
||||||
$mediaInfo = $data->entry_data->PostPage[0]->graphql->shortcode_media;
|
|
||||||
|
|
||||||
//Process the first element, that isn't in the node graph
|
$enclosures = array();
|
||||||
if (count($mediaInfo->edge_media_to_caption->edges) > 0) {
|
$content = '';
|
||||||
$caption = $mediaInfo->edge_media_to_caption->edges[0]->node->text;
|
foreach($mediaInfo->edge_sidecar_to_children->edges as $singleMedia) {
|
||||||
|
$singleMedia = $singleMedia->node;
|
||||||
|
if($singleMedia->is_video) {
|
||||||
|
if(in_array($singleMedia->video_url, $enclosures)) continue; // check if not added yet
|
||||||
|
$content .= '<video controls><source src="' . $singleMedia->video_url . '" type="video/mp4"></video><br>';
|
||||||
|
array_push($enclosures, $singleMedia->video_url);
|
||||||
} else {
|
} else {
|
||||||
$caption = '';
|
if(in_array($singleMedia->display_url, $enclosures)) continue; // check if not added yet
|
||||||
}
|
$content .= '<a href="' . $singleMedia->display_url . '" target="_blank">';
|
||||||
|
$content .= '<img src="' . $singleMedia->display_url . '" alt="' . $postTitle . '" />';
|
||||||
$enclosures = [$mediaInfo->display_url];
|
$content .= '</a><br>';
|
||||||
$content = '<img src="' . htmlentities($mediaInfo->display_url) . '" alt="' . $caption . '" />';
|
array_push($enclosures, $singleMedia->display_url);
|
||||||
|
|
||||||
foreach($mediaInfo->edge_sidecar_to_children->edges as $media) {
|
|
||||||
$display_url = $media->node->display_url;
|
|
||||||
if(!in_array($display_url, $enclosures)) { // add only if not added yet
|
|
||||||
$content .= '<img src="' . htmlentities($display_url) . '" alt="' . $caption . '" />';
|
|
||||||
$enclosures[] = $display_url;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
$content .= '<br>' . nl2br(htmlentities($textContent));
|
||||||
|
|
||||||
return [$content, $enclosures];
|
return array($content, $enclosures);
|
||||||
|
}
|
||||||
|
|
||||||
|
// returns Video post's contents and enclosures
|
||||||
|
protected function getInstagramVideoData($uri) {
|
||||||
|
$mediaInfo = $this->getSinglePostData($uri);
|
||||||
|
|
||||||
|
$textContent = $this->getTextContent($mediaInfo);
|
||||||
|
$content = '<video controls><source src="' . $mediaInfo->video_url . '" type="video/mp4"></video><br>';
|
||||||
|
$content .= '<br>' . nl2br(htmlentities($textContent));
|
||||||
|
|
||||||
|
return array($content, array($mediaInfo->video_url));
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function getTextContent($media) {
|
||||||
|
$textContent = '(no text)';
|
||||||
|
//Process the first element, that isn't in the node graph
|
||||||
|
if (count($media->edge_media_to_caption->edges) > 0) {
|
||||||
|
$textContent = trim($media->edge_media_to_caption->edges[0]->node->text);
|
||||||
|
}
|
||||||
|
return $textContent;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function getSinglePostData($uri) {
|
||||||
|
$shortcode = explode('/', $uri)[4];
|
||||||
|
$data = getContents(self::URI .
|
||||||
|
'graphql/query/?query_hash=' .
|
||||||
|
self::SHORTCODE_QUERY_HASH .
|
||||||
|
'&variables={"shortcode"%3A"' .
|
||||||
|
$shortcode .
|
||||||
|
'"}');
|
||||||
|
|
||||||
|
return json_decode($data)->data->shortcode_media;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected function getInstagramJSON($uri) {
|
protected function getInstagramJSON($uri) {
|
||||||
|
|
||||||
|
if(!is_null($this->getInput('u'))) {
|
||||||
|
|
||||||
|
$userId = $this->getInstagramUserId($this->getInput('u'));
|
||||||
|
|
||||||
|
$data = getContents(self::URI .
|
||||||
|
'graphql/query/?query_hash=' .
|
||||||
|
self::USER_QUERY_HASH .
|
||||||
|
'&variables={"id"%3A"' .
|
||||||
|
$userId .
|
||||||
|
'"%2C"first"%3A10}');
|
||||||
|
return json_decode($data);
|
||||||
|
|
||||||
|
} elseif(!is_null($this->getInput('h'))) {
|
||||||
|
$data = getContents(self::URI .
|
||||||
|
'graphql/query/?query_hash=' .
|
||||||
|
self::TAG_QUERY_HASH .
|
||||||
|
'&variables={"tag_name"%3A"' .
|
||||||
|
$this->getInput('h') .
|
||||||
|
'"%2C"first"%3A10}');
|
||||||
|
return json_decode($data);
|
||||||
|
|
||||||
|
} else {
|
||||||
|
|
||||||
$html = getContents($uri)
|
$html = getContents($uri)
|
||||||
or returnServerError('Could not request Instagram.');
|
or returnServerError('Could not request Instagram.');
|
||||||
$scriptRegex = '/window\._sharedData = (.*);<\/script>/';
|
$scriptRegex = '/window\._sharedData = (.*);<\/script>/';
|
||||||
|
@ -155,6 +251,8 @@ class InstagramBridge extends BridgeAbstract {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
public function getName(){
|
public function getName(){
|
||||||
if(!is_null($this->getInput('u'))) {
|
if(!is_null($this->getInput('u'))) {
|
||||||
return $this->getInput('u') . ' - Instagram Bridge';
|
return $this->getInput('u') . ' - Instagram Bridge';
|
||||||
|
|
|
@ -1,8 +1,7 @@
|
||||||
<?php
|
<?php
|
||||||
/**
|
/**
|
||||||
* This class implements a bridge for http://www.instructables.com, supporting
|
* This class implements a bridge for http://www.instructables.com, supporting
|
||||||
* general feeds and feeds by category. Instructables doesn't support HTTPS as
|
* general feeds and feeds by category.
|
||||||
* of now (23.06.2018), so all connections are insecure!
|
|
||||||
*
|
*
|
||||||
* Remarks:
|
* Remarks:
|
||||||
* - For some reason it is very important to have the category URI end with a
|
* - For some reason it is very important to have the category URI end with a
|
||||||
|
@ -13,7 +12,7 @@
|
||||||
*/
|
*/
|
||||||
class InstructablesBridge extends BridgeAbstract {
|
class InstructablesBridge extends BridgeAbstract {
|
||||||
const NAME = 'Instructables Bridge';
|
const NAME = 'Instructables Bridge';
|
||||||
const URI = 'http://www.instructables.com';
|
const URI = 'https://www.instructables.com';
|
||||||
const DESCRIPTION = 'Returns general feeds and feeds by category';
|
const DESCRIPTION = 'Returns general feeds and feeds by category';
|
||||||
const MAINTAINER = 'logmanoriginal';
|
const MAINTAINER = 'logmanoriginal';
|
||||||
const PARAMETERS = array(
|
const PARAMETERS = array(
|
||||||
|
@ -22,219 +21,201 @@ class InstructablesBridge extends BridgeAbstract {
|
||||||
'name' => 'Category',
|
'name' => 'Category',
|
||||||
'type' => 'list',
|
'type' => 'list',
|
||||||
'values' => array(
|
'values' => array(
|
||||||
'Play' => array(
|
'Circuits' => array(
|
||||||
'All' => '/play/',
|
'All' => '/circuits/',
|
||||||
'KNEX' => '/play/knex/',
|
'Apple' => '/circuits/apple/projects/',
|
||||||
'Offbeat' => '/play/offbeat/',
|
'Arduino' => '/circuits/arduino/projects/',
|
||||||
'Lego' => '/play/lego/',
|
'Art' => '/circuits/art/projects/',
|
||||||
'Airsoft' => '/play/airsoft/',
|
'Assistive Tech' => '/circuits/assistive-tech/projects/',
|
||||||
'Card Games' => '/play/card-games/',
|
'Audio' => '/circuits/audio/projects/',
|
||||||
'Guitars' => '/play/guitars/',
|
'Cameras' => '/circuits/cameras/projects/',
|
||||||
'Instruments' => '/play/instruments/',
|
'Clocks' => '/circuits/clocks/projects/',
|
||||||
'Magic Tricks' => '/play/magic-tricks/',
|
'Computers' => '/circuits/computers/projects/',
|
||||||
'Minecraft' => '/play/minecraft/',
|
'Electronics' => '/circuits/electronics/projects/',
|
||||||
'Music' => '/play/music/',
|
'Gadgets' => '/circuits/gadgets/projects/',
|
||||||
'Nerf' => '/play/nerf/',
|
'Lasers' => '/circuits/lasers/projects/',
|
||||||
'Nintendo' => '/play/nintendo/',
|
'LEDs' => '/circuits/leds/projects/',
|
||||||
'Office Supplies' => '/play/office-supplies/',
|
'Linux' => '/circuits/linux/projects/',
|
||||||
'Paintball' => '/play/paintball/',
|
'Microcontrollers' => '/circuits/microcontrollers/projects/',
|
||||||
'Paper Airplanes' => '/play/paper-airplanes/',
|
'Microsoft' => '/circuits/microsoft/projects/',
|
||||||
'Party Tricks' => '/play/party-tricks/',
|
'Mobile' => '/circuits/mobile/projects/',
|
||||||
'PlayStation' => '/play/playstation/',
|
'Raspberry Pi' => '/circuits/raspberry-pi/projects/',
|
||||||
'Pranks and Humor' => '/play/pranks-and-humor/',
|
'Remote Control' => '/circuits/remote-control/projects/',
|
||||||
'Puzzles' => '/play/puzzles/',
|
'Reuse' => '/circuits/reuse/projects/',
|
||||||
'Siege Engines' => '/play/siege-engines/',
|
'Robots' => '/circuits/robots/projects/',
|
||||||
'Sports' => '/play/sports/',
|
'Sensors' => '/circuits/sensors/projects/',
|
||||||
'Table Top' => '/play/table-top/',
|
'Software' => '/circuits/software/projects/',
|
||||||
'Toys' => '/play/toys/',
|
'Soldering' => '/circuits/soldering/projects/',
|
||||||
'Video Games' => '/play/video-games/',
|
'Speakers' => '/circuits/speakers/projects/',
|
||||||
'Wii' => '/play/wii/',
|
'Tools' => '/circuits/tools/projects/',
|
||||||
'Xbox' => '/play/xbox/',
|
'USB' => '/circuits/usb/projects/',
|
||||||
'Yo-Yo' => '/play/yo-yo/',
|
'Wearables' => '/circuits/wearables/projects/',
|
||||||
),
|
'Websites' => '/circuits/websites/projects/',
|
||||||
'Craft' => array(
|
'Wireless' => '/circuits/wireless/projects/',
|
||||||
'All' => '/craft/',
|
|
||||||
'Art' => '/craft/art/',
|
|
||||||
'Sewing' => '/craft/sewing/',
|
|
||||||
'Paper' => '/craft/paper/',
|
|
||||||
'Jewelry' => '/craft/jewelry/',
|
|
||||||
'Fashion' => '/craft/fashion/',
|
|
||||||
'Books & Journals' => '/craft/books-and-journals/',
|
|
||||||
'Cards' => '/craft/cards/',
|
|
||||||
'Clay' => '/craft/clay/',
|
|
||||||
'Duct Tape' => '/craft/duct-tape/',
|
|
||||||
'Embroidery' => '/craft/embroidery/',
|
|
||||||
'Felt' => '/craft/felt/',
|
|
||||||
'Fiber Arts' => '/craft/fiber-arts/',
|
|
||||||
'Gifts & Wrapping' => '/craft/gifts-and-wrapping/',
|
|
||||||
'Knitting & Crocheting' => '/craft/knitting-and-crocheting/',
|
|
||||||
'Leather' => '/craft/leather/',
|
|
||||||
'Mason Jars' => '/craft/mason-jars/',
|
|
||||||
'No-Sew' => '/craft/no-sew/',
|
|
||||||
'Parties & Weddings' => '/craft/parties-and-weddings/',
|
|
||||||
'Print Making' => '/craft/print-making/',
|
|
||||||
'Soap' => '/craft/soap/',
|
|
||||||
'Wallets' => '/craft/wallets/',
|
|
||||||
),
|
|
||||||
'Technology' => array(
|
|
||||||
'All' => '/technology/',
|
|
||||||
'Electronics' => '/technology/electronics/',
|
|
||||||
'Arduino' => '/technology/arduino/',
|
|
||||||
'Photography' => '/technology/photography/',
|
|
||||||
'Leds' => '/technology/leds/',
|
|
||||||
'Science' => '/technology/science/',
|
|
||||||
'Reuse' => '/technology/reuse/',
|
|
||||||
'Apple' => '/technology/apple/',
|
|
||||||
'Computers' => '/technology/computers/',
|
|
||||||
'3D Printing' => '/technology/3D-Printing/',
|
|
||||||
'Robots' => '/technology/robots/',
|
|
||||||
'Art' => '/technology/art/',
|
|
||||||
'Assistive Tech' => '/technology/assistive-technology/',
|
|
||||||
'Audio' => '/technology/audio/',
|
|
||||||
'Clocks' => '/technology/clocks/',
|
|
||||||
'CNC' => '/technology/cnc/',
|
|
||||||
'Digital Graphics' => '/technology/digital-graphics/',
|
|
||||||
'Gadgets' => '/technology/gadgets/',
|
|
||||||
'Kits' => '/technology/kits/',
|
|
||||||
'Laptops' => '/technology/laptops/',
|
|
||||||
'Lasers' => '/technology/lasers/',
|
|
||||||
'Linux' => '/technology/linux/',
|
|
||||||
'Microcontrollers' => '/technology/microcontrollers/',
|
|
||||||
'Microsoft' => '/technology/microsoft/',
|
|
||||||
'Mobile' => '/technology/mobile/',
|
|
||||||
'Raspberry Pi' => '/technology/raspberry-pi/',
|
|
||||||
'Remote Control' => '/technology/remote-control/',
|
|
||||||
'Sensors' => '/technology/sensors/',
|
|
||||||
'Software' => '/technology/software/',
|
|
||||||
'Soldering' => '/technology/soldering/',
|
|
||||||
'Speakers' => '/technology/speakers/',
|
|
||||||
'Steampunk' => '/technology/steampunk/',
|
|
||||||
'Tools' => '/technology/tools/',
|
|
||||||
'USB' => '/technology/usb/',
|
|
||||||
'Wearables' => '/technology/wearables/',
|
|
||||||
'Websites' => '/technology/websites/',
|
|
||||||
'Wireless' => '/technology/wireless/',
|
|
||||||
),
|
),
|
||||||
'Workshop' => array(
|
'Workshop' => array(
|
||||||
'All' => '/workshop/',
|
'All' => '/workshop/',
|
||||||
'Woodworking' => '/workshop/woodworking/',
|
'3D Printing' => '/workshop/3d-printing/projects/',
|
||||||
'Tools' => '/workshop/tools/',
|
'Cars' => '/workshop/cars/projects/',
|
||||||
'Gardening' => '/workshop/gardening/',
|
'CNC' => '/workshop/cnc/projects/',
|
||||||
'Cars' => '/workshop/cars/',
|
'Electric Vehicles' => '/workshop/electric-vehicles/projects/',
|
||||||
'Metalworking' => '/workshop/metalworking/',
|
'Energy' => '/workshop/energy/projects/',
|
||||||
'Cardboard' => '/workshop/cardboard/',
|
'Furniture' => '/workshop/furniture/projects/',
|
||||||
'Electric Vehicles' => '/workshop/electric-vehicles/',
|
'Home Improvement' => '/workshop/home-improvement/projects/',
|
||||||
'Energy' => '/workshop/energy/',
|
'Home Theater' => '/workshop/home-theater/projects/',
|
||||||
'Furniture' => '/workshop/furniture/',
|
'Hydroponics' => '/workshop/hydroponics/projects/',
|
||||||
'Home Improvement' => '/workshop/home-improvement/',
|
'Knives' => '/workshop/knives/projects/',
|
||||||
'Home Theater' => '/workshop/home-theater/',
|
'Laser Cutting' => '/workshop/laser-cutting/projects/',
|
||||||
'Hydroponics' => '/workshop/hydroponics/',
|
'Lighting' => '/workshop/lighting/projects/',
|
||||||
'Laser Cutting' => '/workshop/laser-cutting/',
|
'Metalworking' => '/workshop/metalworking/projects/',
|
||||||
'Lighting' => '/workshop/lighting/',
|
'Molds & Casting' => '/workshop/molds-and-casting/projects/',
|
||||||
'Molds & Casting' => '/workshop/molds-and-casting/',
|
'Motorcycles' => '/workshop/motorcycles/projects/',
|
||||||
'Motorcycles' => '/workshop/motorcycles/',
|
'Organizing' => '/workshop/organizing/projects/',
|
||||||
'Organizing' => '/workshop/organizing/',
|
'Pallets' => '/workshop/pallets/projects/',
|
||||||
'Pallets' => '/workshop/pallets/',
|
'Repair' => '/workshop/repair/projects/',
|
||||||
'Repair' => '/workshop/repair/',
|
'Science' => '/workshop/science/projects/',
|
||||||
'Shelves' => '/workshop/shelves/',
|
'Shelves' => '/workshop/shelves/projects/',
|
||||||
'Solar' => '/workshop/solar/',
|
'Solar' => '/workshop/solar/projects/',
|
||||||
'Workbenches' => '/workshop/workbenches/',
|
'Tools' => '/workshop/tools/projects/',
|
||||||
|
'Woodworking' => '/workshop/woodworking/projects/',
|
||||||
|
'Workbenches' => '/workshop/workbenches/projects/',
|
||||||
),
|
),
|
||||||
'Home' => array(
|
'Craft' => array(
|
||||||
'All' => '/home/',
|
'All' => '/craft/',
|
||||||
'Halloween' => '/home/halloween/',
|
'Art' => '/craft/art/projects/',
|
||||||
'Decorating' => '/home/decorating/',
|
'Books & Journals' => '/craft/books-and-journals/projects/',
|
||||||
'Organizing' => '/home/organizing/',
|
'Cardboard' => '/craft/cardboard/projects/',
|
||||||
'Pets' => '/home/pets/',
|
'Cards' => '/craft/cards/projects/',
|
||||||
'Life Hacks' => '/home/life-hacks/',
|
'Clay' => '/craft/clay/projects/',
|
||||||
'Beauty' => '/home/beauty/',
|
'Costumes & Cosplay' => '/craft/costumes-and-cosplay/projects/',
|
||||||
'Christmas' => '/home/christmas/',
|
'Digital Graphics' => '/craft/digital-graphics/projects/',
|
||||||
'Cleaning' => '/home/cleaning/',
|
'Duct Tape' => '/craft/duct-tape/projects/',
|
||||||
'Education' => '/home/education/',
|
'Embroidery' => '/craft/embroidery/projects/',
|
||||||
'Finances' => '/home/finances/',
|
'Fashion' => '/craft/fashion/projects/',
|
||||||
'Gardening' => '/home/gardening/',
|
'Felt' => '/craft/felt/projects/',
|
||||||
'Green' => '/home/green/',
|
'Fiber Arts' => '/craft/fiber-arts/projects/',
|
||||||
'Health' => '/home/health/',
|
'Gift Wrapping' => '/craft/gift-wrapping/projects/',
|
||||||
'Hiding Places' => '/home/hiding-places/',
|
'Jewelry' => '/craft/jewelry/projects/',
|
||||||
'Holidays' => '/home/holidays/',
|
'Knitting & Crochet' => '/craft/knitting-and-crochet/projects/',
|
||||||
'Homesteading' => '/home/homesteading/',
|
'Leather' => '/craft/leather/projects/',
|
||||||
'Kids' => '/home/kids/',
|
'Mason Jars' => '/craft/mason-jars/projects/',
|
||||||
'Kitchen' => '/home/kitchen/',
|
'No-Sew' => '/craft/no-sew/projects/',
|
||||||
'Life Skills' => '/home/life-skills/',
|
'Paper' => '/craft/paper/projects/',
|
||||||
'Parenting' => '/home/parenting/',
|
'Parties & Weddings' => '/craft/parties-and-weddings/projects/',
|
||||||
'Pest Control' => '/home/pest-control/',
|
'Photography' => '/craft/photography/projects/',
|
||||||
'Relationships' => '/home/relationships/',
|
'Printmaking' => '/craft/printmaking/projects/',
|
||||||
'Reuse' => '/home/reuse/',
|
'Reuse' => '/craft/reuse/projects/',
|
||||||
'Travel' => '/home/travel/',
|
'Sewing' => '/craft/sewing/projects/',
|
||||||
|
'Soapmaking' => '/craft/soapmaking/projects/',
|
||||||
|
'Wallets' => '/craft/wallets/projects/',
|
||||||
|
),
|
||||||
|
'Cooking' => array(
|
||||||
|
'All' => '/cooking/',
|
||||||
|
'Bacon' => '/cooking/bacon/projects/',
|
||||||
|
'BBQ & Grilling' => '/cooking/bbq-and-grilling/projects/',
|
||||||
|
'Beverages' => '/cooking/beverages/projects/',
|
||||||
|
'Bread' => '/cooking/bread/projects/',
|
||||||
|
'Breakfast' => '/cooking/breakfast/projects/',
|
||||||
|
'Cake' => '/cooking/cake/projects/',
|
||||||
|
'Candy' => '/cooking/candy/projects/',
|
||||||
|
'Canning & Preserving' => '/cooking/canning-and-preserving/projects/',
|
||||||
|
'Cocktails & Mocktails' => '/cooking/cocktails-and-mocktails/projects/',
|
||||||
|
'Coffee' => '/cooking/coffee/projects/',
|
||||||
|
'Cookies' => '/cooking/cookies/projects/',
|
||||||
|
'Cupcakes' => '/cooking/cupcakes/projects/',
|
||||||
|
'Dessert' => '/cooking/dessert/projects/',
|
||||||
|
'Homebrew' => '/cooking/homebrew/projects/',
|
||||||
|
'Main Course' => '/cooking/main-course/projects/',
|
||||||
|
'Pasta' => '/cooking/pasta/projects/',
|
||||||
|
'Pie' => '/cooking/pie/projects/',
|
||||||
|
'Pizza' => '/cooking/pizza/projects/',
|
||||||
|
'Salad' => '/cooking/salad/projects/',
|
||||||
|
'Sandwiches' => '/cooking/sandwiches/projects/',
|
||||||
|
'Snacks & Appetizers' => '/cooking/snacks-and-appetizers/projects/',
|
||||||
|
'Soups & Stews' => '/cooking/soups-and-stews/projects/',
|
||||||
|
'Vegetarian & Vegan' => '/cooking/vegetarian-and-vegan/projects/',
|
||||||
|
),
|
||||||
|
'Living' => array(
|
||||||
|
'All' => '/living/',
|
||||||
|
'Beauty' => '/living/beauty/projects/',
|
||||||
|
'Christmas' => '/living/christmas/projects/',
|
||||||
|
'Cleaning' => '/living/cleaning/projects/',
|
||||||
|
'Decorating' => '/living/decorating/projects/',
|
||||||
|
'Education' => '/living/education/projects/',
|
||||||
|
'Gardening' => '/living/gardening/projects/',
|
||||||
|
'Halloween' => '/living/halloween/projects/',
|
||||||
|
'Health' => '/living/health/projects/',
|
||||||
|
'Hiding Places' => '/living/hiding-places/projects/',
|
||||||
|
'Holidays' => '/living/holidays/projects/',
|
||||||
|
'Homesteading' => '/living/homesteading/projects/',
|
||||||
|
'Kids' => '/living/kids/projects/',
|
||||||
|
'Kitchen' => '/living/kitchen/projects/',
|
||||||
|
'LEGO & KNEX' => '/living/lego-and-knex/projects/',
|
||||||
|
'Life Hacks' => '/living/life-hacks/projects/',
|
||||||
|
'Music' => '/living/music/projects/',
|
||||||
|
'Office Supply Hacks' => '/living/office-supply-hacks/projects/',
|
||||||
|
'Organizing' => '/living/organizing/projects/',
|
||||||
|
'Pest Control' => '/living/pest-control/projects/',
|
||||||
|
'Pets' => '/living/pets/projects/',
|
||||||
|
'Pranks, Tricks, & Humor' => '/living/pranks-tricks-and-humor/projects/',
|
||||||
|
'Relationships' => '/living/relationships/projects/',
|
||||||
|
'Toys & Games' => '/living/toys-and-games/projects/',
|
||||||
|
'Travel' => '/living/travel/projects/',
|
||||||
|
'Video Games' => '/living/video-games/projects/',
|
||||||
),
|
),
|
||||||
'Outside' => array(
|
'Outside' => array(
|
||||||
'All' => '/outside/',
|
'All' => '/outside/',
|
||||||
'Bikes' => '/outside/bikes/',
|
'Backyard' => '/outside/backyard/projects/',
|
||||||
'Survival' => '/outside/survival/',
|
'Beach' => '/outside/beach/projects/',
|
||||||
'Backyard' => '/outside/backyard/',
|
'Bikes' => '/outside/bikes/projects/',
|
||||||
'Beach' => '/outside/beach/',
|
'Birding' => '/outside/birding/projects/',
|
||||||
'Birding' => '/outside/birding/',
|
'Boats' => '/outside/boats/projects/',
|
||||||
'Boats' => '/outside/boats/',
|
'Camping' => '/outside/camping/projects/',
|
||||||
'Camping' => '/outside/camping/',
|
'Climbing' => '/outside/climbing/projects/',
|
||||||
'Climbing' => '/outside/climbing/',
|
'Fire' => '/outside/fire/projects/',
|
||||||
'Fire' => '/outside/fire/',
|
'Fishing' => '/outside/fishing/projects/',
|
||||||
'Fishing' => '/outside/fishing/',
|
'Hunting' => '/outside/hunting/projects/',
|
||||||
'Hunting' => '/outside/hunting/',
|
'Kites' => '/outside/kites/projects/',
|
||||||
'Kites' => '/outside/kites/',
|
'Knots' => '/outside/knots/projects/',
|
||||||
'Knives' => '/outside/knives/',
|
'Launchers' => '/outside/launchers/projects/',
|
||||||
'Knots' => '/outside/knots/',
|
'Paracord' => '/outside/paracord/projects/',
|
||||||
'Paracord' => '/outside/paracord/',
|
'Rockets' => '/outside/rockets/projects/',
|
||||||
'Rockets' => '/outside/rockets/',
|
'Siege Engines' => '/outside/siege-engines/projects/',
|
||||||
'Skateboarding' => '/outside/skateboarding/',
|
'Skateboarding' => '/outside/skateboarding/projects/',
|
||||||
'Snow' => '/outside/snow/',
|
'Snow' => '/outside/snow/projects/',
|
||||||
'Water' => '/outside/water/',
|
'Sports' => '/outside/sports/projects/',
|
||||||
|
'Survival' => '/outside/survival/projects/',
|
||||||
|
'Water' => '/outside/water/projects/',
|
||||||
),
|
),
|
||||||
'Food' => array(
|
'Makeymakey' => array(
|
||||||
'All' => '/food/',
|
'All' => '/makeymakey/',
|
||||||
'Dessert' => '/food/dessert/',
|
'Makey Makey on Instructables' => '/makeymakey/',
|
||||||
'Snacks & Appetizers' => '/food/snacks-and-appetizers/',
|
),
|
||||||
'Bacon' => '/food/bacon/',
|
'Teachers' => array(
|
||||||
'BBQ & Grilling' => '/food/bbq-and-grilling/',
|
'All' => '/teachers/',
|
||||||
'Beverages' => '/food/beverages/',
|
'ELA' => '/teachers/ela/projects/',
|
||||||
'Bread' => '/food/bread/',
|
'Math' => '/teachers/math/projects/',
|
||||||
'Breakfast' => '/food/breakfast/',
|
'Science' => '/teachers/science/projects/',
|
||||||
'Cake' => '/food/cake/',
|
'Social Studies' => '/teachers/social-studies/projects/',
|
||||||
'Candy' => '/food/candy/',
|
'Engineering' => '/teachers/engineering/projects/',
|
||||||
'Canning & Preserves' => '/food/canning-and-preserves/',
|
'Coding' => '/teachers/coding/projects/',
|
||||||
'Cocktails & Mocktails' => '/food/cocktails-and-mocktails/',
|
'Electronics' => '/teachers/electronics/projects/',
|
||||||
'Coffee' => '/food/coffee/',
|
'Robotics' => '/teachers/robotics/projects/',
|
||||||
'Cookies' => '/food/cookies/',
|
'Arduino' => '/teachers/arduino/projects/',
|
||||||
'Cupcakes' => '/food/cupcakes/',
|
'CNC' => '/teachers/cnc/projects/',
|
||||||
'Homebrew' => '/food/homebrew/',
|
'Laser Cutting' => '/teachers/laser-cutting/projects/',
|
||||||
'Main Course' => '/food/main-course/',
|
'3D Printing' => '/teachers/3d-printing/projects/',
|
||||||
'Pasta' => '/food/pasta/',
|
'3D Design' => '/teachers/3d-design/projects/',
|
||||||
'Pie' => '/food/pie/',
|
'Art' => '/teachers/art/projects/',
|
||||||
'Pizza' => '/food/pizza/',
|
'Music' => '/teachers/music/projects/',
|
||||||
'Salad' => '/food/salad/',
|
'Theatre' => '/teachers/theatre/projects/',
|
||||||
'Sandwiches' => '/food/sandwiches/',
|
'Wood Shop' => '/teachers/wood-shop/projects/',
|
||||||
'Soups & Stews' => '/food/soups-and-stews/',
|
'Metal Shop' => '/teachers/metal-shop/projects/',
|
||||||
'Vegetarian & Vegan' => '/food/vegetarian-and-vegan/',
|
'Resources' => '/teachers/resources/projects/',
|
||||||
),
|
),
|
||||||
'Costumes' => array(
|
|
||||||
'All' => '/costumes/',
|
|
||||||
'Props' => '/costumes/props-and-accessories/',
|
|
||||||
'Animals' => '/costumes/animals/',
|
|
||||||
'Comics' => '/costumes/comics/',
|
|
||||||
'Fantasy' => '/costumes/fantasy/',
|
|
||||||
'For Kids' => '/costumes/for-kids/',
|
|
||||||
'For Pets' => '/costumes/for-pets/',
|
|
||||||
'Funny' => '/costumes/funny/',
|
|
||||||
'Games' => '/costumes/games/',
|
|
||||||
'Historic & Futuristic' => '/costumes/historic-and-futuristic/',
|
|
||||||
'Makeup' => '/costumes/makeup/',
|
|
||||||
'Masks' => '/costumes/masks/',
|
|
||||||
'Scary' => '/costumes/scary/',
|
|
||||||
'TV & Movies' => '/costumes/tv-and-movies/',
|
|
||||||
'Weapons & Armor' => '/costumes/weapons-and-armor/',
|
|
||||||
)
|
|
||||||
),
|
),
|
||||||
'title' => 'Select your category (required)',
|
'title' => 'Select your category (required)',
|
||||||
'defaultValue' => 'Technology'
|
'defaultValue' => 'Circuits'
|
||||||
),
|
),
|
||||||
'filter' => array(
|
'filter' => array(
|
||||||
'name' => 'Filter',
|
'name' => 'Filter',
|
||||||
|
@ -252,43 +233,44 @@ class InstructablesBridge extends BridgeAbstract {
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
|
|
||||||
private $uri;
|
|
||||||
|
|
||||||
public function collectData() {
|
public function collectData() {
|
||||||
// Enable the following line to get the category list (dev mode)
|
// Enable the following line to get the category list (dev mode)
|
||||||
// $this->listCategories();
|
// $this->listCategories();
|
||||||
|
|
||||||
$this->uri = static::URI;
|
$html = getSimpleHTMLDOM($this->getURI())
|
||||||
|
or returnServerError('Error loading category ' . $this->getURI());
|
||||||
|
$html = defaultLinkTo($html, $this->getURI());
|
||||||
|
|
||||||
switch($this->queriedContext) {
|
$covers = $html->find('
|
||||||
case 'Category': $this->uri .= $this->getInput('category') . $this->getInput('filter');
|
.category-projects-list > div,
|
||||||
}
|
.category-landing-projects-list > div,
|
||||||
|
');
|
||||||
|
|
||||||
$html = getSimpleHTMLDOM($this->uri)
|
foreach($covers as $cover) {
|
||||||
or returnServerError('Error loading category ' . $this->uri);
|
|
||||||
|
|
||||||
foreach($html->find('ul.explore-covers-list li') as $cover) {
|
|
||||||
$item = array();
|
$item = array();
|
||||||
|
|
||||||
$item['uri'] = static::URI . $cover->find('a.cover-image', 0)->href;
|
$item['uri'] = $cover->find('a.ible-title', 0)->href;
|
||||||
$item['title'] = $cover->find('.title', 0)->innertext;
|
$item['title'] = $cover->find('a.ible-title', 0)->innertext;
|
||||||
$item['author'] = $this->getCategoryAuthor($cover);
|
$item['author'] = $this->getCategoryAuthor($cover);
|
||||||
$item['content'] = '<a href='
|
$item['content'] = '<a href='
|
||||||
. $item['uri']
|
. $item['uri']
|
||||||
. '><img src='
|
. '><img src='
|
||||||
. $cover->find('a.cover-image img', 0)->src
|
. $cover->find('img', 0)->getAttribute('data-src')
|
||||||
. '></a>';
|
. '></a>';
|
||||||
|
|
||||||
$image = str_replace('.RECTANGLE1', '.LARGE', $cover->find('a.cover-image img', 0)->src);
|
$item['enclosures'][] = str_replace(
|
||||||
$item['enclosures'] = [$image];
|
'.RECTANGLE1',
|
||||||
|
'.LARGE',
|
||||||
|
$cover->find('img', 0)->getAttribute('data-src')
|
||||||
|
);
|
||||||
|
|
||||||
$this->items[] = $item;
|
$this->items[] = $item;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getName() {
|
public function getName() {
|
||||||
if(!is_null($this->getInput('category'))
|
switch($this->queriedContext) {
|
||||||
&& !is_null($this->getInput('filter'))) {
|
case 'Category': {
|
||||||
foreach(self::PARAMETERS[$this->queriedContext]['category']['values'] as $key => $value) {
|
foreach(self::PARAMETERS[$this->queriedContext]['category']['values'] as $key => $value) {
|
||||||
$subcategory = array_search($this->getInput('category'), $value);
|
$subcategory = array_search($this->getInput('category'), $value);
|
||||||
|
|
||||||
|
@ -302,15 +284,19 @@ class InstructablesBridge extends BridgeAbstract {
|
||||||
);
|
);
|
||||||
|
|
||||||
return $subcategory . ' (' . $filter . ') - ' . static::NAME;
|
return $subcategory . ' (' . $filter . ') - ' . static::NAME;
|
||||||
|
} break;
|
||||||
}
|
}
|
||||||
|
|
||||||
return parent::getName();
|
return parent::getName();
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getURI() {
|
public function getURI() {
|
||||||
if(!is_null($this->getInput('category'))
|
switch($this->queriedContext) {
|
||||||
&& !is_null($this->getInput('filter'))) {
|
case 'Category': {
|
||||||
return $this->uri;
|
return self::URI
|
||||||
|
. $this->getInput('category')
|
||||||
|
. $this->getInput('filter');
|
||||||
|
} break;
|
||||||
}
|
}
|
||||||
|
|
||||||
return parent::getURI();
|
return parent::getURI();
|
||||||
|
@ -321,11 +307,18 @@ class InstructablesBridge extends BridgeAbstract {
|
||||||
* parameters list)
|
* parameters list)
|
||||||
*/
|
*/
|
||||||
private function listCategories(){
|
private function listCategories(){
|
||||||
// Use arbitrary category to receive full list
|
|
||||||
$html = getSimpleHTMLDOM(self::URI . '/technology/');
|
|
||||||
|
|
||||||
foreach($html->find('.channel a') as $channel) {
|
// Use home page to acquire main categories
|
||||||
$name = html_entity_decode(trim($channel->innertext));
|
$html = getSimpleHTMLDOM(self::URI);
|
||||||
|
$html = defaultLinkTo($html, self::URI);
|
||||||
|
|
||||||
|
foreach($html->find('.home-content-explore-link') as $category) {
|
||||||
|
|
||||||
|
// Use arbitrary category to receive full list
|
||||||
|
$html = getSimpleHTMLDOM($category->href);
|
||||||
|
|
||||||
|
foreach($html->find('.channel-thumbnail a') as $channel) {
|
||||||
|
$name = html_entity_decode(trim($channel->title));
|
||||||
|
|
||||||
// Remove unwanted entities
|
// Remove unwanted entities
|
||||||
$name = str_replace("'", '', $name);
|
$name = str_replace("'", '', $name);
|
||||||
|
@ -333,12 +326,13 @@ class InstructablesBridge extends BridgeAbstract {
|
||||||
|
|
||||||
$uri = $channel->href;
|
$uri = $channel->href;
|
||||||
|
|
||||||
$category = explode('/', $uri)[1];
|
$category_name = explode('/', $uri)[1];
|
||||||
|
|
||||||
if(!isset($categories)
|
if(!isset($categories)
|
||||||
|| !array_key_exists($category, $categories)
|
|| !array_key_exists($category_name, $categories)
|
||||||
|| !in_array($uri, $categories[$category]))
|
|| !in_array($uri, $categories[$category_name]))
|
||||||
$categories[$category][$name] = $uri;
|
$categories[$category_name][$name] = $uri;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Build PHP array manually
|
// Build PHP array manually
|
||||||
|
@ -360,9 +354,9 @@ class InstructablesBridge extends BridgeAbstract {
|
||||||
*/
|
*/
|
||||||
private function getCategoryAuthor($cover) {
|
private function getCategoryAuthor($cover) {
|
||||||
return '<a href='
|
return '<a href='
|
||||||
. static::URI . $cover->find('span.author a', 0)->href
|
. $cover->find('.ible-author a', 0)->href
|
||||||
. '>'
|
. '>'
|
||||||
. $cover->find('span.author a', 0)->innertext
|
. $cover->find('.ible-author a', 0)->innertext
|
||||||
. '</a>';
|
. '</a>';
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
293
bridges/InternetArchiveBridge.php
Normal file
293
bridges/InternetArchiveBridge.php
Normal file
|
@ -0,0 +1,293 @@
|
||||||
|
<?php
|
||||||
|
class InternetArchiveBridge extends BridgeAbstract {
|
||||||
|
const NAME = 'Internet Archive Bridge';
|
||||||
|
const URI = 'https://archive.org';
|
||||||
|
const DESCRIPTION = 'Returns newest uploads, posts and more from an account';
|
||||||
|
const MAINTAINER = 'VerifiedJoseph';
|
||||||
|
const PARAMETERS = array(
|
||||||
|
'Account' => array(
|
||||||
|
'username' => array(
|
||||||
|
'name' => 'Username',
|
||||||
|
'type' => 'text',
|
||||||
|
'required' => true,
|
||||||
|
'exampleValue' => '@verifiedjoseph',
|
||||||
|
),
|
||||||
|
'content' => array(
|
||||||
|
'name' => 'Content',
|
||||||
|
'type' => 'list',
|
||||||
|
'values' => array(
|
||||||
|
'Uploads' => 'uploads',
|
||||||
|
'Posts' => 'posts',
|
||||||
|
'Reviews' => 'reviews',
|
||||||
|
'Collections' => 'collections',
|
||||||
|
'Web Archives' => 'web-archive',
|
||||||
|
),
|
||||||
|
'defaultValue' => 'uploads',
|
||||||
|
)
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
const CACHE_TIMEOUT = 900; // 15 mins
|
||||||
|
|
||||||
|
private $skipClasses = array(
|
||||||
|
'item-ia mobile-header hidden-tiles',
|
||||||
|
'item-ia account-ia'
|
||||||
|
);
|
||||||
|
|
||||||
|
public function collectData() {
|
||||||
|
|
||||||
|
$html = getSimpleHTMLDOM($this->getURI())
|
||||||
|
or returnServerError('Could not request: ' . $this->getURI());
|
||||||
|
|
||||||
|
$html = defaultLinkTo($html, $this->getURI());
|
||||||
|
|
||||||
|
if ($this->getInput('content') !== 'posts') {
|
||||||
|
|
||||||
|
$detailsDivNumber = 0;
|
||||||
|
|
||||||
|
foreach ($html->find('div.results > div[data-id]') as $index => $result) {
|
||||||
|
$item = array();
|
||||||
|
|
||||||
|
if (in_array($result->class, $this->skipClasses)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
switch($result->class) {
|
||||||
|
case 'item-ia':
|
||||||
|
|
||||||
|
switch($this->getInput('content')) {
|
||||||
|
case 'reviews':
|
||||||
|
$item = $this->processReview($result);
|
||||||
|
break;
|
||||||
|
case 'uploads':
|
||||||
|
$item = $this->processUpload($result);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
break;
|
||||||
|
case 'item-ia url-item':
|
||||||
|
$item = $this->processWebArchives($result);
|
||||||
|
break;
|
||||||
|
case 'item-ia collection-ia':
|
||||||
|
$item = $this->processCollection($result);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($this->getInput('content') !== 'reviews') {
|
||||||
|
$hiddenDetails = $this->processHiddenDetails($html, $detailsDivNumber, $item);
|
||||||
|
|
||||||
|
$this->items[] = array_merge($item, $hiddenDetails);
|
||||||
|
} else {
|
||||||
|
|
||||||
|
$this->items[] = $item;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
$detailsDivNumber++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($this->getInput('content') === 'posts') {
|
||||||
|
$this->items = $this->processPosts($html);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getURI() {
|
||||||
|
|
||||||
|
if (!is_null($this->getInput('username')) && !is_null($this->getInput('content'))) {
|
||||||
|
return self::URI . '/details/' . $this->processUsername() . '&tab=' . $this->getInput('content');
|
||||||
|
}
|
||||||
|
|
||||||
|
return parent::getURI();
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getName() {
|
||||||
|
|
||||||
|
if (!is_null($this->getInput('username')) && !is_null($this->getInput('content'))) {
|
||||||
|
|
||||||
|
$contentValues = array_flip(self::PARAMETERS['Account']['content']['values']);
|
||||||
|
|
||||||
|
return $contentValues[$this->getInput('content')] . ' - '
|
||||||
|
. $this->processUsername() . ' - Internet Archive';
|
||||||
|
}
|
||||||
|
|
||||||
|
return parent::getName();
|
||||||
|
}
|
||||||
|
|
||||||
|
private function processUsername() {
|
||||||
|
|
||||||
|
if (substr($this->getInput('username'), 0, 1) !== '@') {
|
||||||
|
return '@' . $this->getInput('username');
|
||||||
|
}
|
||||||
|
|
||||||
|
return $this->getInput('username');
|
||||||
|
}
|
||||||
|
|
||||||
|
private function processUpload($result) {
|
||||||
|
|
||||||
|
$item = array();
|
||||||
|
|
||||||
|
$collection = $result->find('a.stealth', 0);
|
||||||
|
$collectionLink = self::URI . $collection->href;
|
||||||
|
$collectionTitle = $collection->find('div.item-parent-ttl', 0)->plaintext;
|
||||||
|
|
||||||
|
$item['title'] = trim($result->find('div.ttl', 0)->innertext);
|
||||||
|
$item['timestamp'] = strtotime($result->find('div.hidden-tiles.pubdate.C.C3', 0)->children(0)->plaintext);
|
||||||
|
$item['uri'] = $result->find('div.item-ttl.C.C2 > a', 0)->href;
|
||||||
|
|
||||||
|
if ($result->find('div.by.C.C4', 0)->children(2)) {
|
||||||
|
$item['author'] = $result->find('div.by.C.C4', 0)->children(2)->plaintext;
|
||||||
|
}
|
||||||
|
|
||||||
|
$item['content'] = <<<EOD
|
||||||
|
<p>Media Type: {$result->attr['data-mediatype']}<br>
|
||||||
|
Collection: <a href="{$collectionLink}">{$collectionTitle}</a></p>
|
||||||
|
EOD;
|
||||||
|
|
||||||
|
$item['enclosures'][] = self::URI . $result->find('img.item-img', 0)->source;
|
||||||
|
|
||||||
|
return $item;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function processReview($result) {
|
||||||
|
|
||||||
|
$item = array();
|
||||||
|
|
||||||
|
$item['title'] = trim($result->find('div.ttl', 0)->innertext);
|
||||||
|
$item['timestamp'] = strtotime($result->find('div.hidden-tiles.pubdate.C.C3', 0)->children(0)->plaintext);
|
||||||
|
$item['uri'] = $result->find('div.review-title', 0)->children(0)->href;
|
||||||
|
|
||||||
|
if ($result->find('div.by.C.C4', 0)->children(2)) {
|
||||||
|
$item['author'] = $result->find('div.by.C.C4', 0)->children(2)->plaintext;
|
||||||
|
}
|
||||||
|
|
||||||
|
$item['content'] = <<<EOD
|
||||||
|
<p><strong>Subject: {$result->find('div.review-title', 0)->plaintext}</strong></p>
|
||||||
|
<p>{$result->find('div.hidden-lists.review' , 0)->children(1)->plaintext}</p>
|
||||||
|
EOD;
|
||||||
|
|
||||||
|
$item['enclosures'][] = self::URI . $result->find('img.item-img', 0)->source;
|
||||||
|
|
||||||
|
return $item;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function processWebArchives($result) {
|
||||||
|
|
||||||
|
$item = array();
|
||||||
|
|
||||||
|
$item['title'] = trim($result->find('div.ttl', 0)->plaintext);
|
||||||
|
$item['timestamp'] = strtotime($result->find('div.hidden-lists', 0)->children(0)->plaintext);
|
||||||
|
$item['uri'] = $result->find('div.item-ttl.C.C2 > a', 0)->href;
|
||||||
|
|
||||||
|
$item['content'] = <<<EOD
|
||||||
|
{$this->processUsername()} archived <a href="{$item['uri']}">{$result->find('div.ttl', 0)->plaintext}</a>
|
||||||
|
EOD;
|
||||||
|
|
||||||
|
$item['enclosures'][] = $result->find('img.item-img', 0)->source;
|
||||||
|
|
||||||
|
return $item;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function processCollection($result) {
|
||||||
|
|
||||||
|
$item = array();
|
||||||
|
|
||||||
|
$title = trim($result->find('div.collection-title.C.C2', 0)->children(0)->plaintext);
|
||||||
|
$itemCount = strtolower(trim($result->find('div.num-items.topinblock', 0)->plaintext));
|
||||||
|
|
||||||
|
$item['title'] = $title . ' (' . $itemCount . ')';
|
||||||
|
$item['timestamp'] = strtotime($result->find('div.hidden-tiles.pubdate.C.C3', 0)->children(0)->plaintext);
|
||||||
|
$item['uri'] = $result->find('div.collection-title.C.C2 > a', 0)->href;
|
||||||
|
|
||||||
|
$item['content'] = '';
|
||||||
|
|
||||||
|
if ($result->find('img.item-img', 0)) {
|
||||||
|
$item['enclosures'][] = self::URI . $result->find('img.item-img', 0)->source;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $item;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function processHiddenDetails($html, $detailsDivNumber, $item) {
|
||||||
|
|
||||||
|
$description = '';
|
||||||
|
|
||||||
|
if ($html->find('div.details-ia.hidden-tiles', $detailsDivNumber)) {
|
||||||
|
$detailsDiv = $html->find('div.details-ia.hidden-tiles', $detailsDivNumber);
|
||||||
|
|
||||||
|
if ($detailsDiv->find('div.C234', 0)->children(0)) {
|
||||||
|
$description = $detailsDiv->find('div.C234', 0)->children(0)->plaintext;
|
||||||
|
|
||||||
|
$detailsDiv->find('div.C234', 0)->children(0)->innertext = '';
|
||||||
|
}
|
||||||
|
|
||||||
|
$topics = trim($detailsDiv->find('div.C234', 0)->plaintext);
|
||||||
|
|
||||||
|
if (!empty($topics)) {
|
||||||
|
$topics = trim($detailsDiv->find('div.C234', 0)->plaintext);
|
||||||
|
$topics = trim(substr($topics, 7));
|
||||||
|
|
||||||
|
$item['categories'] = explode(',', $topics);
|
||||||
|
}
|
||||||
|
|
||||||
|
$item['content'] = '<p>' . $description . '</p>' . $item['content'];
|
||||||
|
}
|
||||||
|
|
||||||
|
return $item;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function processPosts($html) {
|
||||||
|
|
||||||
|
$items = array();
|
||||||
|
|
||||||
|
foreach ($html->find('table.forumTable > tr') as $index => $tr) {
|
||||||
|
$item = array();
|
||||||
|
|
||||||
|
if ($index === 0) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$item['title'] = $tr->find('td', 0)->plaintext;
|
||||||
|
$item['timestamp'] = strtotime($tr->find('td', 4)->children(0)->plaintext);
|
||||||
|
$item['uri'] = $tr->find('td', 0)->children(0)->href;
|
||||||
|
|
||||||
|
$formLink = <<<EOD
|
||||||
|
<a href="{$tr->find('td', 2)->children(0)->href}">{$tr->find('td', 2)->children(0)->plaintext}</a>
|
||||||
|
EOD;
|
||||||
|
|
||||||
|
$postDate = $tr->find('td', 4)->children(0)->plaintext;
|
||||||
|
|
||||||
|
$postPageHtml = getSimpleHTMLDOMCached($item['uri'], 3600)
|
||||||
|
or returnServerError('Could not request: ' . $item['uri']);
|
||||||
|
|
||||||
|
$postPageHtml = defaultLinkTo($postPageHtml, $this->getURI());
|
||||||
|
|
||||||
|
$post = $postPageHtml->find('div.box.well.well-sm', 0);
|
||||||
|
|
||||||
|
$parentLink = '';
|
||||||
|
$replyLink = <<<EOD
|
||||||
|
<a href="{$post->find('a', 0)->href}">Reply</a>
|
||||||
|
EOD;
|
||||||
|
|
||||||
|
if ($post->find('a', 1)->innertext = 'See parent post') {
|
||||||
|
$parentLink = <<<EOD
|
||||||
|
<a href="{$post->find('a', 1)->href}">View parent post</a>
|
||||||
|
EOD;
|
||||||
|
}
|
||||||
|
|
||||||
|
$post->find('h1', 0)->outertext = '';
|
||||||
|
$post->find('h2', 0)->outertext = '';
|
||||||
|
|
||||||
|
$item['content'] = <<<EOD
|
||||||
|
<p>{$post->innertext}</p>{$replyLink} - {$parentLink} - Posted in {$formLink} on {$postDate}
|
||||||
|
EOD;
|
||||||
|
|
||||||
|
$items[] = $item;
|
||||||
|
|
||||||
|
if (count($items) >= 10) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return $items;
|
||||||
|
}
|
||||||
|
}
|
|
@ -19,28 +19,6 @@ class JapanExpoBridge extends BridgeAbstract {
|
||||||
|
|
||||||
public function collectData(){
|
public function collectData(){
|
||||||
|
|
||||||
function frenchPubDateToTimestamp($date_to_parse) {
|
|
||||||
return strtotime(
|
|
||||||
strtr(
|
|
||||||
strtolower(str_replace('Publié le ', '', $date_to_parse)),
|
|
||||||
array(
|
|
||||||
'janvier' => 'jan',
|
|
||||||
'février' => 'feb',
|
|
||||||
'mars' => 'march',
|
|
||||||
'avril' => 'apr',
|
|
||||||
'mai' => 'may',
|
|
||||||
'juin' => 'jun',
|
|
||||||
'juillet' => 'jul',
|
|
||||||
'août' => 'aug',
|
|
||||||
'septembre' => 'sep',
|
|
||||||
'octobre' => 'oct',
|
|
||||||
'novembre' => 'nov',
|
|
||||||
'décembre' => 'dec'
|
|
||||||
)
|
|
||||||
)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
$convert_article_images = function($matches){
|
$convert_article_images = function($matches){
|
||||||
if(is_array($matches) && count($matches) > 1) {
|
if(is_array($matches) && count($matches) > 1) {
|
||||||
return '<img src="' . $matches[1] . '" />';
|
return '<img src="' . $matches[1] . '" />';
|
||||||
|
@ -82,7 +60,7 @@ class JapanExpoBridge extends BridgeAbstract {
|
||||||
$content = $headings . $article;
|
$content = $headings . $article;
|
||||||
} else {
|
} else {
|
||||||
$date_text = $element->find('span.date', 0)->plaintext;
|
$date_text = $element->find('span.date', 0)->plaintext;
|
||||||
$timestamp = frenchPubDateToTimestamp($date_text);
|
$timestamp = $this->frenchPubDateToTimestamp($date_text);
|
||||||
$title = trim($element->find('span._title', 0)->plaintext);
|
$title = trim($element->find('span._title', 0)->plaintext);
|
||||||
$content = '<img src="'
|
$content = '<img src="'
|
||||||
. $thumbnail
|
. $thumbnail
|
||||||
|
@ -103,4 +81,26 @@ class JapanExpoBridge extends BridgeAbstract {
|
||||||
$count++;
|
$count++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private function frenchPubDateToTimestamp($date_to_parse) {
|
||||||
|
return strtotime(
|
||||||
|
strtr(
|
||||||
|
strtolower(str_replace('Publié le ', '', $date_to_parse)),
|
||||||
|
array(
|
||||||
|
'janvier' => 'jan',
|
||||||
|
'février' => 'feb',
|
||||||
|
'mars' => 'march',
|
||||||
|
'avril' => 'apr',
|
||||||
|
'mai' => 'may',
|
||||||
|
'juin' => 'jun',
|
||||||
|
'juillet' => 'jul',
|
||||||
|
'août' => 'aug',
|
||||||
|
'septembre' => 'sep',
|
||||||
|
'octobre' => 'oct',
|
||||||
|
'novembre' => 'nov',
|
||||||
|
'décembre' => 'dec'
|
||||||
|
)
|
||||||
|
)
|
||||||
|
);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -5,7 +5,7 @@ class KonachanBridge extends MoebooruBridge {
|
||||||
|
|
||||||
const MAINTAINER = 'mitsukarenai';
|
const MAINTAINER = 'mitsukarenai';
|
||||||
const NAME = 'Konachan';
|
const NAME = 'Konachan';
|
||||||
const URI = 'http://konachan.com/';
|
const URI = 'https://konachan.com/';
|
||||||
const DESCRIPTION = 'Returns images from given page';
|
const DESCRIPTION = 'Returns images from given page';
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -24,6 +24,22 @@ class KununuBridge extends BridgeAbstract {
|
||||||
'type' => 'checkbox',
|
'type' => 'checkbox',
|
||||||
'exampleValue' => 'checked',
|
'exampleValue' => 'checked',
|
||||||
'title' => 'Activate to load full article'
|
'title' => 'Activate to load full article'
|
||||||
|
),
|
||||||
|
'include_ratings' => array(
|
||||||
|
'name' => 'Include ratings',
|
||||||
|
'type' => 'checkbox',
|
||||||
|
'title' => 'Activate to include ratings in the feed'
|
||||||
|
),
|
||||||
|
'include_benefits' => array(
|
||||||
|
'name' => 'Include benefits',
|
||||||
|
'type' => 'checkbox',
|
||||||
|
'title' => 'Activate to include benefits in the feed'
|
||||||
|
),
|
||||||
|
'limit' => array(
|
||||||
|
'name' => 'Limit',
|
||||||
|
'type' => 'number',
|
||||||
|
'defaultValue' => 3,
|
||||||
|
'title' => "Maximum number of items to return in the feed.\n0 = unlimited"
|
||||||
)
|
)
|
||||||
),
|
),
|
||||||
array(
|
array(
|
||||||
|
@ -98,6 +114,8 @@ class KununuBridge extends BridgeAbstract {
|
||||||
$articles = $section->find('article')
|
$articles = $section->find('article')
|
||||||
or returnServerError('Unable to find articles!');
|
or returnServerError('Unable to find articles!');
|
||||||
|
|
||||||
|
$limit = $this->getInput('limit') ?: 0;
|
||||||
|
|
||||||
// Go through all articles
|
// Go through all articles
|
||||||
foreach($articles as $article) {
|
foreach($articles as $article) {
|
||||||
|
|
||||||
|
@ -116,7 +134,7 @@ class KununuBridge extends BridgeAbstract {
|
||||||
$item = array();
|
$item = array();
|
||||||
|
|
||||||
$item['author'] = $this->extractArticleAuthorPosition($article);
|
$item['author'] = $this->extractArticleAuthorPosition($article);
|
||||||
$item['timestamp'] = strtotime($date);
|
$item['timestamp'] = strtotime($date->content);
|
||||||
$item['title'] = $rating->getAttribute('aria-label')
|
$item['title'] = $rating->getAttribute('aria-label')
|
||||||
. ' : '
|
. ' : '
|
||||||
. strip_tags($summary->innertext);
|
. strip_tags($summary->innertext);
|
||||||
|
@ -131,6 +149,8 @@ class KununuBridge extends BridgeAbstract {
|
||||||
|
|
||||||
$this->items[] = $item;
|
$this->items[] = $item;
|
||||||
|
|
||||||
|
if ($limit > 0 && count($this->items) >= $limit) break;
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -175,7 +195,32 @@ class KununuBridge extends BridgeAbstract {
|
||||||
$description = $article->find('[itemprop=reviewBody]', 0)
|
$description = $article->find('[itemprop=reviewBody]', 0)
|
||||||
or returnServerError('Cannot find article description!');
|
or returnServerError('Cannot find article description!');
|
||||||
|
|
||||||
return $description->innertext;
|
$retVal = $description->innertext;
|
||||||
|
|
||||||
|
if($this->getInput('include_ratings')
|
||||||
|
&& ($ratings = $article->find('.review-ratings .rating-group'))) {
|
||||||
|
$retVal .= (empty($retVal) ? '' : '<hr>') . '<table>';
|
||||||
|
foreach($ratings as $rating) {
|
||||||
|
$retVal .= <<<EOD
|
||||||
|
<tr>
|
||||||
|
<td>{$rating->find('.rating-title', 0)->plaintext}
|
||||||
|
<td>{$rating->find('.rating-badge', 0)->plaintext}
|
||||||
|
</tr>
|
||||||
|
EOD;
|
||||||
|
}
|
||||||
|
$retVal .= '</table>';
|
||||||
|
}
|
||||||
|
|
||||||
|
if($this->getInput('include_benefits')
|
||||||
|
&& ($benefits = $article->find('benefit'))) {
|
||||||
|
$retVal .= (empty($retVal) ? '' : '<hr>') . '<ul>';
|
||||||
|
foreach($benefits as $benefit) {
|
||||||
|
$retVal .= "<li>{$benefit->plaintext}</li>";
|
||||||
|
}
|
||||||
|
$retVal .= '</ul>';
|
||||||
|
}
|
||||||
|
|
||||||
|
return $retVal;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
477
bridges/LaCentraleBridge.php
Normal file
477
bridges/LaCentraleBridge.php
Normal file
|
@ -0,0 +1,477 @@
|
||||||
|
<?php
|
||||||
|
class LaCentraleBridge extends BridgeAbstract {
|
||||||
|
|
||||||
|
const MAINTAINER = 'jacknumber';
|
||||||
|
const NAME = 'La Centrale';
|
||||||
|
const URI = 'https://www.lacentrale.fr/';
|
||||||
|
const DESCRIPTION = 'Returns most recent vehicules ads from LaCentrale';
|
||||||
|
|
||||||
|
const PARAMETERS = array( array(
|
||||||
|
'type' => array(
|
||||||
|
'name' => 'Type de véhicule',
|
||||||
|
'type' => 'list',
|
||||||
|
'values' => array(
|
||||||
|
'Voiture' => 'car',
|
||||||
|
'Camion/Pickup' => 'truck',
|
||||||
|
'Moto' => 'moto',
|
||||||
|
'Scooter' => 'scooter',
|
||||||
|
'Quad' => 'quad',
|
||||||
|
'Caravane/Camping-car' => 'mobileHome'
|
||||||
|
)
|
||||||
|
),
|
||||||
|
'brand' => array(
|
||||||
|
'name' => 'Marque',
|
||||||
|
'type' => 'list',
|
||||||
|
'values' => array(
|
||||||
|
'' => '',
|
||||||
|
'ABARTH' => 'ABARTH',
|
||||||
|
'AC' => 'AC',
|
||||||
|
'AIXAM' => 'AIXAM',
|
||||||
|
'ALFA ROMEO' => 'ALFA ROMEO',
|
||||||
|
'ALKE' => 'ALKE',
|
||||||
|
'ALPINA' => 'ALPINA',
|
||||||
|
'ALPINE' => 'ALPINE',
|
||||||
|
'AMC' => 'AMC',
|
||||||
|
'ANAIG' => 'ANAIG',
|
||||||
|
'APRILIA' => 'APRILIA',
|
||||||
|
'ARIEL' => 'ARIEL',
|
||||||
|
'ASTON MARTIN' => 'ASTON MARTIN',
|
||||||
|
'AUDI' => 'AUDI',
|
||||||
|
'AUSTIN HEALEY' => 'AUSTIN HEALEY',
|
||||||
|
'AUSTIN' => 'AUSTIN',
|
||||||
|
'AUTOBIANCHI' => 'AUTOBIANCHI',
|
||||||
|
'AVINTON' => 'AVINTON',
|
||||||
|
'BELLIER' => 'BELLIER',
|
||||||
|
'BENELLI' => 'BENELLI',
|
||||||
|
'BENTLEY' => 'BENTLEY',
|
||||||
|
'BETA' => 'BETA',
|
||||||
|
'BMW' => 'BMW',
|
||||||
|
'BOLLORE' => 'BOLLORE',
|
||||||
|
'BRIXTON' => 'BRIXTON',
|
||||||
|
'BUELL' => 'BUELL',
|
||||||
|
'BUGATTI' => 'BUGATTI',
|
||||||
|
'BUICK' => 'BUICK',
|
||||||
|
'BULLIT' => 'BULLIT',
|
||||||
|
'CADILLAC' => 'CADILLAC',
|
||||||
|
'CASALINI' => 'CASALINI',
|
||||||
|
'CATERHAM' => 'CATERHAM',
|
||||||
|
'CHATENET' => 'CHATENET',
|
||||||
|
'CHEVROLET' => 'CHEVROLET',
|
||||||
|
'CHRYSLER' => 'CHRYSLER',
|
||||||
|
'CHUNLAN' => 'CHUNLAN',
|
||||||
|
'CITROEN' => 'CITROEN',
|
||||||
|
'COURB' => 'COURB',
|
||||||
|
'CR&S' => 'CR&S',
|
||||||
|
'CUPRA' => 'CUPRA',
|
||||||
|
'CYCLONE' => 'CYCLONE',
|
||||||
|
'DACIA' => 'DACIA',
|
||||||
|
'DAELIM' => 'DAELIM',
|
||||||
|
'DAEWOO' => 'DAEWOO',
|
||||||
|
'DAF' => 'DAF',
|
||||||
|
'DAIHATSU' => 'DAIHATSU',
|
||||||
|
'DANGEL' => 'DANGEL',
|
||||||
|
'DATSUN' => 'DATSUN',
|
||||||
|
'DE SOTO' => 'DE SOTO',
|
||||||
|
'DE TOMASO' => 'DE TOMASO',
|
||||||
|
'DERBI' => 'DERBI',
|
||||||
|
'DEVINCI' => 'DEVINCI',
|
||||||
|
'DODGE' => 'DODGE',
|
||||||
|
'DONKERVOORT' => 'DONKERVOORT',
|
||||||
|
'DS' => 'DS',
|
||||||
|
'DUCATI' => 'DUCATI',
|
||||||
|
'DUCATY' => 'DUCATY',
|
||||||
|
'DUE' => 'DUE',
|
||||||
|
'ENFIELD' => 'ENFIELD',
|
||||||
|
'EXCALIBUR' => 'EXCALIBUR',
|
||||||
|
'FACEL VEGA' => 'FACEL VEGA',
|
||||||
|
'FANTIC MOTOR' => 'FANTIC MOTOR',
|
||||||
|
'FERRARI' => 'FERRARI',
|
||||||
|
'FIAT' => 'FIAT',
|
||||||
|
'FISKER' => 'FISKER',
|
||||||
|
'FORD' => 'FORD',
|
||||||
|
'FUSO' => 'FUSO',
|
||||||
|
'GAS GAS' => 'GAS GAS',
|
||||||
|
'GILERA' => 'GILERA',
|
||||||
|
'GMC' => 'GMC',
|
||||||
|
'GOWINN' => 'GOWINN',
|
||||||
|
'GRANDIN' => 'GRANDIN',
|
||||||
|
'HARLEY DAVIDSON' => 'HARLEY DAVIDSON',
|
||||||
|
'HOMMELL' => 'HOMMELL',
|
||||||
|
'HONDA' => 'HONDA',
|
||||||
|
'HUMMER' => 'HUMMER',
|
||||||
|
'HUSABERG' => 'HUSABERG',
|
||||||
|
'HUSQVARNA' => 'HUSQVARNA',
|
||||||
|
'HYOSUNG' => 'HYOSUNG',
|
||||||
|
'HYUNDAI' => 'HYUNDAI',
|
||||||
|
'INDIAN' => 'INDIAN',
|
||||||
|
'INFINITI' => 'INFINITI',
|
||||||
|
'INNOCENTI' => 'INNOCENTI',
|
||||||
|
'ISUZU' => 'ISUZU',
|
||||||
|
'IVECO' => 'IVECO',
|
||||||
|
'JAGUAR' => 'JAGUAR',
|
||||||
|
'JDM SIMPA' => 'JDM SIMPA',
|
||||||
|
'JEEP' => 'JEEP',
|
||||||
|
'JENSEN' => 'JENSEN',
|
||||||
|
'JIAYUAN' => 'JIAYUAN',
|
||||||
|
'KAWASAKI' => 'KAWASAKI',
|
||||||
|
'KEEWAY' => 'KEEWAY',
|
||||||
|
'KIA' => 'KIA',
|
||||||
|
'KSR' => 'KSR',
|
||||||
|
'KTM' => 'KTM',
|
||||||
|
'KYMCO' => 'KYMCO',
|
||||||
|
'LADA' => 'LADA',
|
||||||
|
'LAMBORGHINI' => 'LAMBORGHINI',
|
||||||
|
'LANCIA' => 'LANCIA',
|
||||||
|
'LAND ROVER' => 'LAND ROVER',
|
||||||
|
'LEXUS' => 'LEXUS',
|
||||||
|
'LIGIER' => 'LIGIER',
|
||||||
|
'LINCOLN' => 'LINCOLN',
|
||||||
|
'LONDON TAXI COMPANY' => 'LONDON TAXI COMPANY',
|
||||||
|
'LOTUS' => 'LOTUS',
|
||||||
|
'MAGPOWER' => 'MAGPOWER',
|
||||||
|
'MAN' => 'MAN',
|
||||||
|
'MASAI' => 'MASAI',
|
||||||
|
'MASERATI' => 'MASERATI',
|
||||||
|
'MASH' => 'MASH',
|
||||||
|
'MATRA' => 'MATRA',
|
||||||
|
'MAYBACH' => 'MAYBACH',
|
||||||
|
'MAZDA' => 'MAZDA',
|
||||||
|
'MCLAREN' => 'MCLAREN',
|
||||||
|
'MEGA' => 'MEGA',
|
||||||
|
'MERCEDES' => 'MERCEDES',
|
||||||
|
'MERCEDES-AMG' => 'MERCEDES-AMG',
|
||||||
|
'MERCURY' => 'MERCURY',
|
||||||
|
'MEYERS MANX' => 'MEYERS MANX',
|
||||||
|
'MG' => 'MG',
|
||||||
|
'MIA ELECTRIC' => 'MIA ELECTRIC',
|
||||||
|
'MICROCAR' => 'MICROCAR',
|
||||||
|
'MINAUTO' => 'MINAUTO',
|
||||||
|
'MINI' => 'MINI',
|
||||||
|
'MITSUBISHI' => 'MITSUBISHI',
|
||||||
|
'MORGAN' => 'MORGAN',
|
||||||
|
'MORRIS' => 'MORRIS',
|
||||||
|
'MOTO GUZZI' => 'MOTO GUZZI',
|
||||||
|
'MOTO MORINI' => 'MOTO MORINI',
|
||||||
|
'MOTOBECANE' => 'MOTOBECANE',
|
||||||
|
'MPM MOTORS' => 'MPM MOTORS',
|
||||||
|
'MV AGUSTA' => 'MV AGUSTA',
|
||||||
|
'NISSAN' => 'NISSAN',
|
||||||
|
'NORTON' => 'NORTON',
|
||||||
|
'NSU' => 'NSU',
|
||||||
|
'OLDSMOBILE' => 'OLDSMOBILE',
|
||||||
|
'OPEL' => 'OPEL',
|
||||||
|
'ORCAL' => 'ORCAL',
|
||||||
|
'OSSA' => 'OSSA',
|
||||||
|
'PACKARD' => 'PACKARD',
|
||||||
|
'PANTHER' => 'PANTHER',
|
||||||
|
'PEUGEOT' => 'PEUGEOT',
|
||||||
|
'PGO' => 'PGO',
|
||||||
|
'PIAGGIO' => 'PIAGGIO',
|
||||||
|
'PLYMOUTH' => 'PLYMOUTH',
|
||||||
|
'POLARIS' => 'POLARIS',
|
||||||
|
'PONTIAC' => 'PONTIAC',
|
||||||
|
'PORSCHE' => 'PORSCHE',
|
||||||
|
'REALM' => 'REALM',
|
||||||
|
'REGAL RAPTOR' => 'REGAL RAPTOR',
|
||||||
|
'RENAULT' => 'RENAULT',
|
||||||
|
'RIEJU' => 'RIEJU',
|
||||||
|
'ROLLS ROYCE' => 'ROLLS ROYCE',
|
||||||
|
'ROVER' => 'ROVER',
|
||||||
|
'ROYAL ENFIELD' => 'ROYAL ENFIELD',
|
||||||
|
'SAAB' => 'SAAB',
|
||||||
|
'SANTANA' => 'SANTANA',
|
||||||
|
'SCANIA' => 'SCANIA',
|
||||||
|
'SEAT' => 'SEAT',
|
||||||
|
'SECMA' => 'SECMA',
|
||||||
|
'SHELBY' => 'SHELBY',
|
||||||
|
'SHERCO' => 'SHERCO',
|
||||||
|
'SIMCA' => 'SIMCA',
|
||||||
|
'SKODA' => 'SKODA',
|
||||||
|
'SMART' => 'SMART',
|
||||||
|
'SPYKER' => 'SPYKER',
|
||||||
|
'SSANGYONG' => 'SSANGYONG',
|
||||||
|
'STUDEBAKER' => 'STUDEBAKER',
|
||||||
|
'SUBARU' => 'SUBARU',
|
||||||
|
'SUNBEAM' => 'SUNBEAM',
|
||||||
|
'SUZUKI' => 'SUZUKI',
|
||||||
|
'SWM' => 'SWM',
|
||||||
|
'SYM' => 'SYM',
|
||||||
|
'TALBOT SIMCA' => 'TALBOT SIMCA',
|
||||||
|
'TALBOT' => 'TALBOT',
|
||||||
|
'TEILHOL' => 'TEILHOL',
|
||||||
|
'TESLA' => 'TESLA',
|
||||||
|
'TM' => 'TM',
|
||||||
|
'TNT MOTOR' => 'TNT MOTOR',
|
||||||
|
'TOYOTA' => 'TOYOTA',
|
||||||
|
'TRIUMPH' => 'TRIUMPH',
|
||||||
|
'TVR' => 'TVR',
|
||||||
|
'VAUXHALL' => 'VAUXHALL',
|
||||||
|
'VESPA' => 'VESPA',
|
||||||
|
'VICTORY' => 'VICTORY',
|
||||||
|
'VOLKSWAGEN' => 'VOLKSWAGEN',
|
||||||
|
'VOLVO' => 'VOLVO',
|
||||||
|
'VOXAN' => 'VOXAN',
|
||||||
|
'WIESMANN' => 'WIESMANN',
|
||||||
|
'YAMAHA' => 'YAMAHA',
|
||||||
|
'YCF' => 'YCF',
|
||||||
|
'ZERO' => 'ZERO',
|
||||||
|
'ZONGSHEN' => 'ZONGSHEN'
|
||||||
|
)
|
||||||
|
),
|
||||||
|
'model' => array(
|
||||||
|
'name' => 'Modèle',
|
||||||
|
'type' => 'text',
|
||||||
|
'title' => 'Get the exact name on LaCentrale'
|
||||||
|
),
|
||||||
|
'versions' => array(
|
||||||
|
'name' => 'Version(s)',
|
||||||
|
'type' => 'text',
|
||||||
|
'title' => 'Get the exact name(s) on LaCentrale. Separate by comma'
|
||||||
|
),
|
||||||
|
'category' => array(
|
||||||
|
'name' => 'Catégorie',
|
||||||
|
'type' => 'list',
|
||||||
|
'values' => array(
|
||||||
|
'' => '',
|
||||||
|
'Voiture' => array(
|
||||||
|
'4x4, SUV & Crossover' => '47',
|
||||||
|
'Citadine' => '40',
|
||||||
|
'Berline' => '41_42',
|
||||||
|
'Break' => '43',
|
||||||
|
'Cabriolet' => '46',
|
||||||
|
'Coupé' => '45',
|
||||||
|
'Monospace' => '44',
|
||||||
|
'Bus et minibus' => '82',
|
||||||
|
'Fourgonnette' => '85',
|
||||||
|
'Fourgon (< 3,5 tonnes)' => '81',
|
||||||
|
'Pick-up' => '50',
|
||||||
|
'Voiture société, commerciale' => '80',
|
||||||
|
'Sans permis' => '48',
|
||||||
|
'Camion (> 3,5 tonnes)' => '83',
|
||||||
|
),
|
||||||
|
'Camion/Pickup' => array(
|
||||||
|
'Camion (> 3,5 tonnes)' => '83',
|
||||||
|
'Fourgon (< 3,5 tonnes)' => '81',
|
||||||
|
'Bus et minibus' => '82',
|
||||||
|
'Fourgonnette' => '85',
|
||||||
|
'Pick-up' => '50',
|
||||||
|
'Voiture société, commerciale' => '80'
|
||||||
|
),
|
||||||
|
'Moto' => array(
|
||||||
|
'Custom' => '60',
|
||||||
|
'Offroad' => '61',
|
||||||
|
'Roadster' => '62',
|
||||||
|
'GT' => '63',
|
||||||
|
'Mini moto' => '64',
|
||||||
|
'Mobylette' => '65',
|
||||||
|
'Supermotard' => '66',
|
||||||
|
'Trail' => '67',
|
||||||
|
'Side-car' => '69',
|
||||||
|
'Sportive' => '68'
|
||||||
|
),
|
||||||
|
'Caravane/Camping-car' => array(
|
||||||
|
'Caravane' => '423',
|
||||||
|
'Profilé' => '506',
|
||||||
|
'Fourgon aménagé' => '507',
|
||||||
|
'Intégral' => '508',
|
||||||
|
'Capucine' => '510'
|
||||||
|
)
|
||||||
|
)
|
||||||
|
),
|
||||||
|
'pricemin' => array(
|
||||||
|
'name' => 'Prix min',
|
||||||
|
'type' => 'number'
|
||||||
|
),
|
||||||
|
'pricemax' => array(
|
||||||
|
'name' => 'Prix max',
|
||||||
|
'type' => 'number'
|
||||||
|
),
|
||||||
|
'location' => array(
|
||||||
|
'name' => 'CP ou département',
|
||||||
|
'type' => 'number',
|
||||||
|
'title' => 'Only one'
|
||||||
|
),
|
||||||
|
'distance' => array(
|
||||||
|
'name' => 'Rayon de recherche',
|
||||||
|
'type' => 'list',
|
||||||
|
'values' => array(
|
||||||
|
'' => '',
|
||||||
|
'10 km' => '1',
|
||||||
|
'20 km' => '2',
|
||||||
|
'50 km' => '3',
|
||||||
|
'100 km' => '4',
|
||||||
|
'200 km' => '5'
|
||||||
|
)
|
||||||
|
),
|
||||||
|
'region' => array(
|
||||||
|
'name' => 'Région',
|
||||||
|
'type' => 'list',
|
||||||
|
'values' => array(
|
||||||
|
'' => '',
|
||||||
|
'Auvergne-Rhône-Alpes' => 'FR-ARA',
|
||||||
|
'Bourgogne-Franche-Comté' => 'FR-BFC',
|
||||||
|
'Bretagne' => 'FR-BRE',
|
||||||
|
'Centre-Val de Loire' => 'FR-CVL',
|
||||||
|
'Corse' => 'FR-COR',
|
||||||
|
'Grand Est' => 'FR-GES',
|
||||||
|
'Hauts-de-France' => 'FR-HDF',
|
||||||
|
'Île-de-France' => 'FR-IDF',
|
||||||
|
'Normandie' => 'FR-NOR',
|
||||||
|
'Nouvelle-Aquitaine' => 'FR-PAC',
|
||||||
|
'Occitanie' => 'FR-PDL',
|
||||||
|
'Pays de la Loire' => 'FR-OCC',
|
||||||
|
'Provence-Alpes-Côte d\'Azur' => 'FR-NAQ'
|
||||||
|
)
|
||||||
|
),
|
||||||
|
'mileagemin' => array(
|
||||||
|
'name' => 'Kilométrage min',
|
||||||
|
'type' => 'number'
|
||||||
|
),
|
||||||
|
'mileagemax' => array(
|
||||||
|
'name' => 'Kilométrage max',
|
||||||
|
'type' => 'number'
|
||||||
|
),
|
||||||
|
'yearmin' => array(
|
||||||
|
'name' => 'Année min',
|
||||||
|
'type' => 'number'
|
||||||
|
),
|
||||||
|
'yearmax' => array(
|
||||||
|
'name' => 'Année max',
|
||||||
|
'type' => 'number'
|
||||||
|
),
|
||||||
|
'cubiccapacitymin' => array(
|
||||||
|
'name' => 'Cylindrée min',
|
||||||
|
'type' => 'number'
|
||||||
|
),
|
||||||
|
'cubiccapacitymax' => array(
|
||||||
|
'name' => 'Cylindrée max',
|
||||||
|
'type' => 'number'
|
||||||
|
),
|
||||||
|
'fuel' => array(
|
||||||
|
'name' => 'Énergie',
|
||||||
|
'type' => 'list',
|
||||||
|
'values' => array(
|
||||||
|
'' => '',
|
||||||
|
'Diesel' => 'dies',
|
||||||
|
'Essence' => 'ess',
|
||||||
|
'Électrique' => 'elec',
|
||||||
|
'Hybride' => 'hyb',
|
||||||
|
'GPL' => 'gpl',
|
||||||
|
'Bioéthanol' => 'eth',
|
||||||
|
'Autre' => 'alt'
|
||||||
|
)
|
||||||
|
),
|
||||||
|
'gearbox' => array(
|
||||||
|
'name' => 'Boite de vitesse',
|
||||||
|
'type' => 'list',
|
||||||
|
'values' => array(
|
||||||
|
'' => '',
|
||||||
|
'Boite automatique' => 'AUTO',
|
||||||
|
'Boite mécanique' => 'MANUAL'
|
||||||
|
)
|
||||||
|
),
|
||||||
|
'doors' => array(
|
||||||
|
'name' => 'Nombre de portes',
|
||||||
|
'type' => 'list',
|
||||||
|
'values' => array(
|
||||||
|
'' => '',
|
||||||
|
'2 portes' => '2',
|
||||||
|
'3 portes' => '3',
|
||||||
|
'4 portes' => '4',
|
||||||
|
'5 portes' => '5',
|
||||||
|
'6 portes ou plus' => '6'
|
||||||
|
)
|
||||||
|
),
|
||||||
|
'firsthand' => array(
|
||||||
|
'name' => 'Première main',
|
||||||
|
'type' => 'checkbox'
|
||||||
|
),
|
||||||
|
'seller' => array(
|
||||||
|
'name' => 'Vendeur',
|
||||||
|
'type' => 'list',
|
||||||
|
'values' => array(
|
||||||
|
'' => '',
|
||||||
|
'Particulier' => 'PART',
|
||||||
|
'Professionel' => 'PRO'
|
||||||
|
)
|
||||||
|
),
|
||||||
|
'sort' => array(
|
||||||
|
'name' => 'Tri',
|
||||||
|
'type' => 'list',
|
||||||
|
'values' => array(
|
||||||
|
'Prix (croissant)' => 'priceAsc',
|
||||||
|
'Prix (décroissant)' => 'priceDesc',
|
||||||
|
'Marque (croissant)' => 'makeAsc',
|
||||||
|
'Marque (décroissant)' => 'makeDesc',
|
||||||
|
'Kilométrage (croissant)' => 'mileageAsc',
|
||||||
|
'Kilométrage (décroissant)' => 'mileageDesc',
|
||||||
|
'Année (croissant)' => 'yearAsc',
|
||||||
|
'Année (décroissant)' => 'yearDesc',
|
||||||
|
'Département (croissant)' => 'visitPlaceAsc',
|
||||||
|
'Département (décroissant)' => 'visitPlaceDesc'
|
||||||
|
)
|
||||||
|
),
|
||||||
|
));
|
||||||
|
|
||||||
|
public function collectData(){
|
||||||
|
// check data
|
||||||
|
if(!empty($this->getInput('distance'))
|
||||||
|
&& is_null($this->getInput('location'))) {
|
||||||
|
returnClientError('You need a place ("CP ou département") to search arround.');
|
||||||
|
}
|
||||||
|
|
||||||
|
$params = array(
|
||||||
|
'vertical' => $this->getInput('type'),
|
||||||
|
'makesModelsCommercialNames' => $this->getInput('brand') . ':' . $this->getInput('model'),
|
||||||
|
'versions' => $this->getInput('versions'),
|
||||||
|
'categories' => $this->getInput('category'),
|
||||||
|
'priceMin' => $this->getInput('pricemin'),
|
||||||
|
'priceMax' => $this->getInput('pricemax'),
|
||||||
|
'dptCp' => $this->getInput('location'),
|
||||||
|
'distance' => $this->getInput('distance'),
|
||||||
|
'regions' => $this->getInput('region'),
|
||||||
|
'mileageMin' => $this->getInput('mileagemin'),
|
||||||
|
'mileageMax' => $this->getInput('mileagemax'),
|
||||||
|
'yearMin' => $this->getInput('yearmin'),
|
||||||
|
'yearMax' => $this->getInput('yearmax'),
|
||||||
|
'cubicMin' => $this->getInput('cubiccapacitymin'),
|
||||||
|
'cubicMax' => $this->getInput('cubiccapacitymax'),
|
||||||
|
'energies' => $this->getInput('fuel'),
|
||||||
|
'firstHand' => $this->getInput('firsthand') ? 'true' : 'false',
|
||||||
|
'gearbox' => $this->getInput('gearbox'),
|
||||||
|
'doors' => $this->getInput('doors'),
|
||||||
|
'sortBy' => $this->getInput('sort')
|
||||||
|
);
|
||||||
|
$url = self::URI . 'listing?' . http_build_query($params);
|
||||||
|
$html = getSimpleHTMLDOM($url)
|
||||||
|
or returnServerError('Could not request LaCentrale.');
|
||||||
|
|
||||||
|
foreach($html->find('.linkAd') as $element) {
|
||||||
|
|
||||||
|
$item = array();
|
||||||
|
$item['uri'] = trim(self::URI, '/') . $element->href;
|
||||||
|
$item['title'] = $element->find('.brandModel', 0)->plaintext;
|
||||||
|
$item['sellerType'] = $element->find('.typeSeller', 0)->plaintext;
|
||||||
|
$item['author'] = $item['sellerType'];
|
||||||
|
$item['version'] = $element->find('.version', 0)->plaintext;
|
||||||
|
$item['price'] = $element->find('.fieldPrice', 0)->plaintext;
|
||||||
|
$item['year'] = $element->find('.fieldYear', 0)->plaintext;
|
||||||
|
$item['mileage'] = $element->find('.fieldMileage', 0)->plaintext;
|
||||||
|
$item['departement'] = str_replace(',', '', $element->find('.dptCont', 0)->plaintext);
|
||||||
|
$item['thumbnail'] = $element->find('.imgContent img', 0)->src;
|
||||||
|
$item['enclosures'] = array($item['thumbnail']);
|
||||||
|
|
||||||
|
$item['content'] = '
|
||||||
|
<img src="' . $item['thumbnail'] . '">
|
||||||
|
<br>Variation : ' . $item['version']
|
||||||
|
. '<br>Prix : ' . $item['price']
|
||||||
|
. '<br>Année : ' . $item['year']
|
||||||
|
. '<br>Kilométrage : ' . $item['mileage']
|
||||||
|
. '<br>Département : ' . $item['departement']
|
||||||
|
. '<br>Type de vendeur : ' . $item['sellerType'];
|
||||||
|
|
||||||
|
$this->items[] = $item;
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -356,6 +356,7 @@ class LeBonCoinBridge extends BridgeAbstract {
|
||||||
$data = $this->buildRequestJson();
|
$data = $this->buildRequestJson();
|
||||||
|
|
||||||
$header = array(
|
$header = array(
|
||||||
|
'User-Agent: LBC;Android;Null;Null;Null;Null;Null;Null;Null;Null',
|
||||||
'Content-Type: application/json',
|
'Content-Type: application/json',
|
||||||
'Content-Length: ' . strlen($data),
|
'Content-Length: ' . strlen($data),
|
||||||
'api_key: ' . self::$LBC_API_KEY
|
'api_key: ' . self::$LBC_API_KEY
|
||||||
|
@ -430,11 +431,11 @@ class LeBonCoinBridge extends BridgeAbstract {
|
||||||
);
|
);
|
||||||
|
|
||||||
if($this->getInput('region') != '') {
|
if($this->getInput('region') != '') {
|
||||||
$requestJson->filters->location['regions'] = [$this->getInput('region')];
|
$requestJson->filters->location['regions'] = array($this->getInput('region'));
|
||||||
}
|
}
|
||||||
|
|
||||||
if($this->getInput('department') != '') {
|
if($this->getInput('department') != '') {
|
||||||
$requestJson->filters->location['departments'] = [$this->getInput('department')];
|
$requestJson->filters->location['departments'] = array($this->getInput('department'));
|
||||||
}
|
}
|
||||||
|
|
||||||
if($this->getInput('cities') != '') {
|
if($this->getInput('cities') != '') {
|
||||||
|
@ -466,7 +467,7 @@ class LeBonCoinBridge extends BridgeAbstract {
|
||||||
}
|
}
|
||||||
|
|
||||||
if($this->getInput('estate') != '') {
|
if($this->getInput('estate') != '') {
|
||||||
$requestJson->filters->enums['real_estate_type'] = [$this->getInput('estate')];
|
$requestJson->filters->enums['real_estate_type'] = array($this->getInput('estate'));
|
||||||
}
|
}
|
||||||
|
|
||||||
if($this->getInput('roomsmin') != ''
|
if($this->getInput('roomsmin') != ''
|
||||||
|
@ -525,7 +526,7 @@ class LeBonCoinBridge extends BridgeAbstract {
|
||||||
}
|
}
|
||||||
|
|
||||||
if($this->getInput('fuel') != '') {
|
if($this->getInput('fuel') != '') {
|
||||||
$requestJson->filters->enums['fuel'] = [$this->getInput('fuel')];
|
$requestJson->filters->enums['fuel'] = array($this->getInput('fuel'));
|
||||||
}
|
}
|
||||||
|
|
||||||
$requestJson->limit = 30;
|
$requestJson->limit = 30;
|
||||||
|
|
22
bridges/ListverseBridge.php
Normal file
22
bridges/ListverseBridge.php
Normal file
|
@ -0,0 +1,22 @@
|
||||||
|
<?php
|
||||||
|
class ListverseBridge extends FeedExpander {
|
||||||
|
|
||||||
|
const MAINTAINER = 'IceWreck';
|
||||||
|
const NAME = 'Listverse Bridge';
|
||||||
|
const URI = 'https://listverse.com/';
|
||||||
|
const CACHE_TIMEOUT = 3600;
|
||||||
|
const DESCRIPTION = 'RSS feed for Listverse';
|
||||||
|
|
||||||
|
public function collectData(){
|
||||||
|
$this->collectExpandableDatas('https://listverse.com/feed/', 15);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function parseItem($newsItem){
|
||||||
|
$item = parent::parseItem($newsItem);
|
||||||
|
// $articlePage gets the entire page's contents
|
||||||
|
$articlePage = getSimpleHTMLDOM($newsItem->link);
|
||||||
|
$article = $articlePage->find('#articlecontentonly', 0);
|
||||||
|
$item['content'] = $article;
|
||||||
|
return $item;
|
||||||
|
}
|
||||||
|
}
|
|
@ -3,7 +3,7 @@ class MangareaderBridge extends BridgeAbstract {
|
||||||
|
|
||||||
const MAINTAINER = 'logmanoriginal';
|
const MAINTAINER = 'logmanoriginal';
|
||||||
const NAME = 'Mangareader Bridge';
|
const NAME = 'Mangareader Bridge';
|
||||||
const URI = 'http://www.mangareader.net';
|
const URI = 'https://www.mangareader.net';
|
||||||
const CACHE_TIMEOUT = 10800; // 3h
|
const CACHE_TIMEOUT = 10800; // 3h
|
||||||
const DESCRIPTION = 'Returns the latest updates, popular mangas or manga updates (new chapters)';
|
const DESCRIPTION = 'Returns the latest updates, popular mangas or manga updates (new chapters)';
|
||||||
|
|
||||||
|
|
89
bridges/MastodonBridge.php
Normal file
89
bridges/MastodonBridge.php
Normal file
|
@ -0,0 +1,89 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
class MastodonBridge extends FeedExpander {
|
||||||
|
|
||||||
|
const MAINTAINER = 'husim0';
|
||||||
|
const NAME = 'Mastodon Bridge';
|
||||||
|
const CACHE_TIMEOUT = 900; // 15mn
|
||||||
|
const DESCRIPTION = 'Returns toots';
|
||||||
|
const URI = 'https://mastodon.social';
|
||||||
|
|
||||||
|
const PARAMETERS = array(array(
|
||||||
|
'canusername' => array(
|
||||||
|
'name' => 'Canonical username (ex : @sebsauvage@framapiaf.org)',
|
||||||
|
'required' => true,
|
||||||
|
),
|
||||||
|
'norep' => array(
|
||||||
|
'name' => 'Without replies',
|
||||||
|
'type' => 'checkbox',
|
||||||
|
'title' => 'Only return initial toots'
|
||||||
|
),
|
||||||
|
'noboost' => array(
|
||||||
|
'name' => 'Without boosts',
|
||||||
|
'required' => false,
|
||||||
|
'type' => 'checkbox',
|
||||||
|
'title' => 'Hide boosts'
|
||||||
|
)
|
||||||
|
));
|
||||||
|
|
||||||
|
public function getName(){
|
||||||
|
switch($this->queriedContext) {
|
||||||
|
case 'By username':
|
||||||
|
return $this->getInput('canusername');
|
||||||
|
default: return parent::getName();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function parseItem($newItem){
|
||||||
|
$item = parent::parseItem($newItem);
|
||||||
|
|
||||||
|
$content = str_get_html($item['content']);
|
||||||
|
$title = str_get_html($item['title']);
|
||||||
|
|
||||||
|
$item['title'] = $content->plaintext;
|
||||||
|
|
||||||
|
if(strlen($item['title']) > 75) {
|
||||||
|
$item['title'] = substr($item['title'], 0, strpos(wordwrap($item['title'], 75), "\n")) . '...';
|
||||||
|
}
|
||||||
|
|
||||||
|
if(strpos($title, 'shared a status by') !== false) {
|
||||||
|
if($this->getInput('noboost')) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
preg_match('/shared a status by (\S{0,})/', $title, $matches);
|
||||||
|
$item['title'] = 'Boost ' . $matches[1] . ' ' . $item['title'];
|
||||||
|
$item['author'] = $matches[1];
|
||||||
|
} else {
|
||||||
|
$item['author'] = $this->getInput('canusername');
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check if it's a initial toot or a response
|
||||||
|
if($this->getInput('norep') && preg_match('/^@.+/', trim($content->plaintext))) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $item;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function getInstance(){
|
||||||
|
preg_match('/^@[a-zA-Z0-9_]+@(.+)/', $this->getInput('canusername'), $matches);
|
||||||
|
return $matches[1];
|
||||||
|
}
|
||||||
|
|
||||||
|
private function getUsername(){
|
||||||
|
preg_match('/^@([a-zA-Z_0-9_]+)@.+/', $this->getInput('canusername'), $matches);
|
||||||
|
return $matches[1];
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getURI(){
|
||||||
|
if($this->getInput('canusername'))
|
||||||
|
return 'https://' . $this->getInstance() . '/users/' . $this->getUsername() . '.atom';
|
||||||
|
|
||||||
|
return parent::getURI();
|
||||||
|
}
|
||||||
|
|
||||||
|
public function collectData(){
|
||||||
|
return $this->collectExpandableDatas($this->getURI());
|
||||||
|
}
|
||||||
|
}
|
|
@ -30,6 +30,10 @@ class MediapartBridge extends FeedExpander {
|
||||||
protected function parseItem($newsItem) {
|
protected function parseItem($newsItem) {
|
||||||
$item = parent::parseItem($newsItem);
|
$item = parent::parseItem($newsItem);
|
||||||
|
|
||||||
|
// Mediapart provide multiple type of contents.
|
||||||
|
// We only process items relative to the newspaper
|
||||||
|
// See issue #1292 - https://github.com/RSS-Bridge/rss-bridge/issues/1292
|
||||||
|
if (strpos($item['uri'], self::URI . 'journal/') === 0) {
|
||||||
// Enable single page mode?
|
// Enable single page mode?
|
||||||
if ($this->getInput('single_page_mode') === true) {
|
if ($this->getInput('single_page_mode') === true) {
|
||||||
$item['uri'] .= '?onglet=full';
|
$item['uri'] .= '?onglet=full';
|
||||||
|
@ -54,6 +58,7 @@ class MediapartBridge extends FeedExpander {
|
||||||
$content = defaultLinkTo($content, static::URI);
|
$content = defaultLinkTo($content, static::URI);
|
||||||
$item['content'] .= $content;
|
$item['content'] .= $content;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return $item;
|
return $item;
|
||||||
}
|
}
|
||||||
|
|
|
@ -15,11 +15,11 @@ class N26Bridge extends BridgeAbstract
|
||||||
|
|
||||||
public function collectData()
|
public function collectData()
|
||||||
{
|
{
|
||||||
$html = getSimpleHTMLDOM(self::URI . '/en-fr/blog-archive')
|
$html = getSimpleHTMLDOM(self::URI . '/en-eu/blog-archive')
|
||||||
or returnServerError('Error while downloading the website content');
|
or returnServerError('Error while downloading the website content');
|
||||||
|
|
||||||
foreach($html->find('div.ga') as $article) {
|
foreach($html->find('div[class="ag ah ai aj bs bt dx ea fo gx ie if ih ii ij ik s"]') as $article) {
|
||||||
$item = [];
|
$item = array();
|
||||||
|
|
||||||
$item['uri'] = self::URI . $article->find('h2 a', 0)->href;
|
$item['uri'] = self::URI . $article->find('h2 a', 0)->href;
|
||||||
$item['title'] = $article->find('h2 a', 0)->plaintext;
|
$item['title'] = $article->find('h2 a', 0)->plaintext;
|
||||||
|
@ -27,9 +27,9 @@ class N26Bridge extends BridgeAbstract
|
||||||
$fullArticle = getSimpleHTMLDOM($item['uri'])
|
$fullArticle = getSimpleHTMLDOM($item['uri'])
|
||||||
or returnServerError('Error while downloading the full article');
|
or returnServerError('Error while downloading the full article');
|
||||||
|
|
||||||
$dateElement = $fullArticle->find('span[class="fk fl de ch fm by"]', 0);
|
$dateElement = $fullArticle->find('time', 0);
|
||||||
$item['timestamp'] = strtotime($dateElement->plaintext);
|
$item['timestamp'] = strtotime($dateElement->plaintext);
|
||||||
$item['content'] = $fullArticle->find('main article', 0)->innertext;
|
$item['content'] = $fullArticle->find('div[class="af ag ah ai an"]', 1)->innertext;
|
||||||
|
|
||||||
$this->items[] = $item;
|
$this->items[] = $item;
|
||||||
}
|
}
|
||||||
|
|
60
bridges/NFLRUSBridge.php
Normal file
60
bridges/NFLRUSBridge.php
Normal file
|
@ -0,0 +1,60 @@
|
||||||
|
<?php
|
||||||
|
class NFLRUSBridge extends BridgeAbstract {
|
||||||
|
|
||||||
|
const NAME = 'NFLRUS';
|
||||||
|
const URI = 'http://nflrus.ru/';
|
||||||
|
const DESCRIPTION = 'Returns the recent articles published on nflrus.ru';
|
||||||
|
const MAINTAINER = 'Maxim Shpak';
|
||||||
|
|
||||||
|
private function getEnglishMonth($month) {
|
||||||
|
$months = array(
|
||||||
|
'Января' => 'January',
|
||||||
|
'Февраля' => 'February',
|
||||||
|
'Марта' => 'March',
|
||||||
|
'Апреля' => 'April',
|
||||||
|
'Мая' => 'May',
|
||||||
|
'Июня' => 'June',
|
||||||
|
'Июля' => 'July',
|
||||||
|
'Августа' => 'August',
|
||||||
|
'Сентября' => 'September',
|
||||||
|
'Октября' => 'October',
|
||||||
|
'Ноября' => 'November',
|
||||||
|
'Декабря' => 'December',
|
||||||
|
);
|
||||||
|
|
||||||
|
if (isset($months[$month])) {
|
||||||
|
return $months[$month];
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function extractArticleTimestamp($article) {
|
||||||
|
$time = $article->find('time', 0);
|
||||||
|
if($time) {
|
||||||
|
$timestring = trim($time->plaintext);
|
||||||
|
$parts = explode(' ', $timestring);
|
||||||
|
$month = $this->getEnglishMonth($parts[1]);
|
||||||
|
if ($month) {
|
||||||
|
$timestring = $parts[0] . ' ' . $month . ' ' . $parts[2];
|
||||||
|
return strtotime($timestring);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function collectData() {
|
||||||
|
$html = getSimpleHTMLDOM(self::URI)
|
||||||
|
or returnServerError('Unable to get any articles from NFLRUS');
|
||||||
|
$html = defaultLinkTo($html, self::URI);
|
||||||
|
|
||||||
|
foreach($html->find('article') as $article) {
|
||||||
|
$item = array();
|
||||||
|
$item['uri'] = $article->find('.b-article__title a', 0)->href;
|
||||||
|
$item['title'] = $article->find('.b-article__title a', 0)->plaintext;
|
||||||
|
$item['author'] = $article->find('.link-author', 0)->plaintext;
|
||||||
|
$item['timestamp'] = $this->extractArticleTimestamp($article);
|
||||||
|
$item['content'] = $article->find('div', 0)->innertext;
|
||||||
|
$this->items[] = $item;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
26
bridges/NYTBridge.php
Normal file
26
bridges/NYTBridge.php
Normal file
|
@ -0,0 +1,26 @@
|
||||||
|
<?php
|
||||||
|
class NYTBridge extends FeedExpander {
|
||||||
|
|
||||||
|
const MAINTAINER = 'IceWreck';
|
||||||
|
const NAME = 'New York Times Bridge';
|
||||||
|
const URI = 'https://www.nytimes.com/';
|
||||||
|
const CACHE_TIMEOUT = 3600;
|
||||||
|
const DESCRIPTION = 'RSS feed for the New York Times';
|
||||||
|
|
||||||
|
public function collectData(){
|
||||||
|
$this->collectExpandableDatas('https://rss.nytimes.com/services/xml/rss/nyt/HomePage.xml', 15);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function parseItem($newsItem){
|
||||||
|
$item = parent::parseItem($newsItem);
|
||||||
|
// $articlePage gets the entire page's contents
|
||||||
|
$articlePage = getSimpleHTMLDOM($newsItem->link);
|
||||||
|
// figure contain's the main article image
|
||||||
|
$article = $articlePage->find('figure', 0);
|
||||||
|
// p > css-exrw3m has the actual article
|
||||||
|
foreach($articlePage->find('p.css-exrw3m') as $element)
|
||||||
|
$article = $article . $element;
|
||||||
|
$item['content'] = $article;
|
||||||
|
return $item;
|
||||||
|
}
|
||||||
|
}
|
194
bridges/NationalGeographicBridge.php
Normal file
194
bridges/NationalGeographicBridge.php
Normal file
|
@ -0,0 +1,194 @@
|
||||||
|
<?php
|
||||||
|
class NationalGeographicBridge extends BridgeAbstract {
|
||||||
|
|
||||||
|
const CONTEXT_BY_TOPIC = 'By Topic';
|
||||||
|
const PARAMETER_TOPIC = 'topic';
|
||||||
|
const PARAMETER_FULL_ARTICLE = 'full';
|
||||||
|
const TOPIC_MAGAZINE = 'Magazine';
|
||||||
|
const TOPIC_LATEST_STORIES = 'Latest Stories';
|
||||||
|
|
||||||
|
const NAME = 'National Geographic';
|
||||||
|
const URI = 'https://www.nationalgeographic.com/';
|
||||||
|
const DESCRIPTION = 'Fetches the latest articles from the National Geographic Magazine';
|
||||||
|
const MAINTAINER = 'logmanoriginal';
|
||||||
|
const PARAMETERS = array(
|
||||||
|
self::CONTEXT_BY_TOPIC => array(
|
||||||
|
self::PARAMETER_TOPIC => array(
|
||||||
|
'name' => 'Topic',
|
||||||
|
'type' => 'list',
|
||||||
|
'values' => array(
|
||||||
|
self::TOPIC_MAGAZINE => 'magazine',
|
||||||
|
self::TOPIC_LATEST_STORIES => 'latest-stories'
|
||||||
|
),
|
||||||
|
'title' => 'Select your topic',
|
||||||
|
'defaultValue' => 'Magazine'
|
||||||
|
)
|
||||||
|
),
|
||||||
|
'global' => array(
|
||||||
|
self::PARAMETER_FULL_ARTICLE => array(
|
||||||
|
'name' => 'Full Article',
|
||||||
|
'type' => 'checkbox',
|
||||||
|
'title' => 'Enable to load full articles (takes longer)'
|
||||||
|
)
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
private $topicName = '';
|
||||||
|
|
||||||
|
public function getURI() {
|
||||||
|
switch ($this->queriedContext) {
|
||||||
|
case self::CONTEXT_BY_TOPIC: {
|
||||||
|
return self::URI . $this->getInput(self::PARAMETER_TOPIC);
|
||||||
|
} break;
|
||||||
|
default: {
|
||||||
|
return parent::getURI();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public function collectData() {
|
||||||
|
$this->topicName = $this->getTopicName($this->getInput(self::PARAMETER_TOPIC));
|
||||||
|
|
||||||
|
switch($this->topicName) {
|
||||||
|
case self::TOPIC_MAGAZINE: {
|
||||||
|
return $this->collectMagazine();
|
||||||
|
} break;
|
||||||
|
case self::TOPIC_LATEST_STORIES: {
|
||||||
|
return $this->collectLatestStories();
|
||||||
|
} break;
|
||||||
|
default: {
|
||||||
|
returnServerError('Unknown topic: "' . $this->topicName . '"');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getName() {
|
||||||
|
switch ($this->queriedContext) {
|
||||||
|
case self::CONTEXT_BY_TOPIC: {
|
||||||
|
return static::NAME . ': ' . $this->topicName;
|
||||||
|
} break;
|
||||||
|
default: {
|
||||||
|
return parent::getName();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private function getTopicName($topic) {
|
||||||
|
return array_search($topic, static::PARAMETERS[self::CONTEXT_BY_TOPIC][self::PARAMETER_TOPIC]['values']);
|
||||||
|
}
|
||||||
|
|
||||||
|
private function collectMagazine() {
|
||||||
|
$uri = $this->getURI();
|
||||||
|
|
||||||
|
$html = getSimpleHTMLDOM($uri)
|
||||||
|
or returnServerError('Could not request ' . $uri);
|
||||||
|
|
||||||
|
$script = $html->find('#lead-component script')[0];
|
||||||
|
|
||||||
|
$json = json_decode($script->innertext, true);
|
||||||
|
|
||||||
|
// This is probably going to break in the future, fix it then :)
|
||||||
|
foreach($json['body']['0']['multilayout_promo_beta']['stories'] as $story) {
|
||||||
|
$this->addStory($story);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private function collectLatestStories() {
|
||||||
|
$uri = self::URI . 'latest-stories/_jcr_content/content/hubfeed.promo-hub-feed-all-stories.json';
|
||||||
|
|
||||||
|
$json_raw = getContents($uri)
|
||||||
|
or returnServerError('Could not request ' . $uri);
|
||||||
|
|
||||||
|
foreach(json_decode($json_raw, true) as $story) {
|
||||||
|
$this->addStory($story);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private function addStory($story) {
|
||||||
|
$title = 'Unknown title';
|
||||||
|
$content = '';
|
||||||
|
|
||||||
|
foreach($story['components'] as $component) {
|
||||||
|
switch($component['content_type']) {
|
||||||
|
case 'title': {
|
||||||
|
$title = $component['title']['text'];
|
||||||
|
} break;
|
||||||
|
case 'dek': {
|
||||||
|
$content = $component['dek']['text'];
|
||||||
|
} break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$item = array();
|
||||||
|
|
||||||
|
$item['uri'] = $story['uri'];
|
||||||
|
$item['title'] = $title;
|
||||||
|
|
||||||
|
// if full article is requested!
|
||||||
|
if ($this->getInput(self::PARAMETER_FULL_ARTICLE))
|
||||||
|
$item['content'] = $this->getFullArticle($item['uri']);
|
||||||
|
else
|
||||||
|
$item['content'] = $content;
|
||||||
|
|
||||||
|
if (isset($story['promo_image'])) {
|
||||||
|
switch($story['promo_image']['content_type']) {
|
||||||
|
case 'image': {
|
||||||
|
$item['enclosures'][] = $story['promo_image']['image']['uri'];
|
||||||
|
} break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (isset($story['lead_media'])) {
|
||||||
|
$media = $story['lead_media'];
|
||||||
|
switch($media['content_type']) {
|
||||||
|
case 'image': {
|
||||||
|
// Don't add if promo_image was added
|
||||||
|
if (empty($item['enclosures']))
|
||||||
|
$item['enclosures'][] = $media['image']['uri'];
|
||||||
|
} break;
|
||||||
|
case 'image_gallery': {
|
||||||
|
foreach($media['image_gallery']['images'] as $image) {
|
||||||
|
$item['enclosures'][] = $image['uri'];
|
||||||
|
}
|
||||||
|
} break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->items[] = $item;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function getFullArticle($uri) {
|
||||||
|
$html = getSimpleHTMLDOMCached($uri)
|
||||||
|
or returnServerError('Could not load ' . $uri);
|
||||||
|
|
||||||
|
$html = defaultLinkTo($html, $uri);
|
||||||
|
|
||||||
|
$content = '';
|
||||||
|
|
||||||
|
foreach($html->find('
|
||||||
|
.content > .smartbody.text,
|
||||||
|
.content > .section.image script[type="text/json"],
|
||||||
|
.content > .section.image span[itemprop="caption"],
|
||||||
|
.content > .section.inline script[type="text/json"]
|
||||||
|
') as $element) {
|
||||||
|
if ($element->tag === 'script') {
|
||||||
|
$json = json_decode($element->innertext, true);
|
||||||
|
if (isset($json['src'])) {
|
||||||
|
$content .= '<img src="' . $json['src'] . '" width="100%" alt="' . $json['alt'] . '">';
|
||||||
|
} elseif (isset($json['galleryType']) && isset($json['endpoint'])) {
|
||||||
|
$doc = getContents($json['endpoint'])
|
||||||
|
or returnServerError('Could not load ' . $json['endpoint']);
|
||||||
|
$json = json_decode($doc, true);
|
||||||
|
foreach($json['items'] as $item) {
|
||||||
|
$content .= '<p>' . $item['caption'] . '</p>';
|
||||||
|
$content .= '<img src="' . $item['url'] . '" width="100%" alt="' . $item['caption'] . '">';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
$content .= $element->outertext;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $content;
|
||||||
|
}
|
||||||
|
}
|
|
@ -3,7 +3,7 @@ class NiceMatinBridge extends FeedExpander {
|
||||||
|
|
||||||
const MAINTAINER = 'pit-fgfjiudghdf';
|
const MAINTAINER = 'pit-fgfjiudghdf';
|
||||||
const NAME = 'NiceMatin';
|
const NAME = 'NiceMatin';
|
||||||
const URI = 'http://www.nicematin.com/';
|
const URI = 'https://www.nicematin.com/';
|
||||||
const DESCRIPTION = 'Returns the 10 newest posts from NiceMatin (full text)';
|
const DESCRIPTION = 'Returns the 10 newest posts from NiceMatin (full text)';
|
||||||
|
|
||||||
public function collectData(){
|
public function collectData(){
|
||||||
|
|
|
@ -17,6 +17,15 @@ class NineGagBridge extends BridgeAbstract {
|
||||||
'Fresh' => 'fresh',
|
'Fresh' => 'fresh',
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
'video' => array(
|
||||||
|
'name' => 'Filter Video',
|
||||||
|
'type' => 'list',
|
||||||
|
'values' => array(
|
||||||
|
'NotFiltred' => 'none',
|
||||||
|
'VideoFiltred' => 'without',
|
||||||
|
'VideoOnly' => 'only',
|
||||||
|
),
|
||||||
|
),
|
||||||
'p' => array(
|
'p' => array(
|
||||||
'name' => 'Pages',
|
'name' => 'Pages',
|
||||||
'type' => 'number',
|
'type' => 'number',
|
||||||
|
@ -121,6 +130,24 @@ class NineGagBridge extends BridgeAbstract {
|
||||||
}
|
}
|
||||||
|
|
||||||
foreach ($posts as $post) {
|
foreach ($posts as $post) {
|
||||||
|
$AvoidElement = false;
|
||||||
|
switch ($this->getInput('video')) {
|
||||||
|
case 'without':
|
||||||
|
if ($post['type'] === 'Animated') {
|
||||||
|
$AvoidElement = true;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 'only':
|
||||||
|
echo $post['type'];
|
||||||
|
if ($post['type'] !== 'Animated') {
|
||||||
|
$AvoidElement = true;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 'none': default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!$AvoidElement) {
|
||||||
$item['uri'] = $post['url'];
|
$item['uri'] = $post['url'];
|
||||||
$item['title'] = $post['title'];
|
$item['title'] = $post['title'];
|
||||||
$item['content'] = self::getContent($post);
|
$item['content'] = self::getContent($post);
|
||||||
|
@ -130,6 +157,7 @@ class NineGagBridge extends BridgeAbstract {
|
||||||
$this->items[] = $item;
|
$this->items[] = $item;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public function getName() {
|
public function getName() {
|
||||||
if ($this->getInput('d')) {
|
if ($this->getInput('d')) {
|
||||||
|
|
|
@ -3,7 +3,7 @@ class NovelUpdatesBridge extends BridgeAbstract {
|
||||||
|
|
||||||
const MAINTAINER = 'albirew';
|
const MAINTAINER = 'albirew';
|
||||||
const NAME = 'Novel Updates';
|
const NAME = 'Novel Updates';
|
||||||
const URI = 'http://www.novelupdates.com/';
|
const URI = 'https://www.novelupdates.com/';
|
||||||
const CACHE_TIMEOUT = 21600; // 6h
|
const CACHE_TIMEOUT = 21600; // 6h
|
||||||
const DESCRIPTION = 'Returns releases from Novel Updates';
|
const DESCRIPTION = 'Returns releases from Novel Updates';
|
||||||
const PARAMETERS = array( array(
|
const PARAMETERS = array( array(
|
||||||
|
|
|
@ -1,9 +1,9 @@
|
||||||
<?php
|
<?php
|
||||||
class WhydBridge extends BridgeAbstract {
|
class OpenwhydBridge extends BridgeAbstract {
|
||||||
|
|
||||||
const MAINTAINER = 'kranack';
|
const MAINTAINER = 'kranack';
|
||||||
const NAME = 'Whyd Bridge';
|
const NAME = 'Openwhyd Bridge';
|
||||||
const URI = 'http://www.whyd.com/';
|
const URI = 'https://openwhyd.org';
|
||||||
const CACHE_TIMEOUT = 600; // 10min
|
const CACHE_TIMEOUT = 600; // 10min
|
||||||
const DESCRIPTION = 'Returns 10 newest music from user profile';
|
const DESCRIPTION = 'Returns 10 newest music from user profile';
|
||||||
|
|
||||||
|
@ -17,8 +17,7 @@ class WhydBridge extends BridgeAbstract {
|
||||||
private $userName = '';
|
private $userName = '';
|
||||||
|
|
||||||
public function getIcon() {
|
public function getIcon() {
|
||||||
return self::URI . 'assets/favicons/
|
return self::URI . '/images/favicon.ico';
|
||||||
32-6b62a9f14d5e1a9213090d8f00f286bba3a6022381a76390d1d0926493b12593.png?v=6';
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public function collectData(){
|
public function collectData(){
|
||||||
|
@ -26,11 +25,11 @@ class WhydBridge extends BridgeAbstract {
|
||||||
if(strlen(preg_replace('/[^0-9a-f]/', '', $this->getInput('u'))) == 24) {
|
if(strlen(preg_replace('/[^0-9a-f]/', '', $this->getInput('u'))) == 24) {
|
||||||
// is input the userid ?
|
// is input the userid ?
|
||||||
$html = getSimpleHTMLDOM(
|
$html = getSimpleHTMLDOM(
|
||||||
self::URI . 'u/' . preg_replace('/[^0-9a-f]/', '', $this->getInput('u'))
|
self::URI . '/u/' . preg_replace('/[^0-9a-f]/', '', $this->getInput('u'))
|
||||||
) or returnServerError('No results for this query.');
|
) or returnServerError('No results for this query.');
|
||||||
} else { // input may be the username
|
} else { // input may be the username
|
||||||
$html = getSimpleHTMLDOM(
|
$html = getSimpleHTMLDOM(
|
||||||
self::URI . 'search?q=' . urlencode($this->getInput('u'))
|
self::URI . '/search?q=' . urlencode($this->getInput('u'))
|
||||||
) or returnServerError('No results for this query.');
|
) or returnServerError('No results for this query.');
|
||||||
|
|
||||||
for($j = 0; $j < 5; $j++) {
|
for($j = 0; $j < 5; $j++) {
|
||||||
|
@ -57,6 +56,6 @@ class WhydBridge extends BridgeAbstract {
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getName(){
|
public function getName(){
|
||||||
return (!empty($this->userName) ? $this->userName . ' - ' : '') . 'Whyd Bridge';
|
return (!empty($this->userName) ? $this->userName . ' - ' : '') . 'Openwhyd Bridge';
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -3,7 +3,7 @@ class ParuVenduImmoBridge extends BridgeAbstract {
|
||||||
|
|
||||||
const MAINTAINER = 'polo2ro';
|
const MAINTAINER = 'polo2ro';
|
||||||
const NAME = 'Paru Vendu Immobilier';
|
const NAME = 'Paru Vendu Immobilier';
|
||||||
const URI = 'http://www.paruvendu.fr';
|
const URI = 'https://www.paruvendu.fr';
|
||||||
const CACHE_TIMEOUT = 10800; // 3h
|
const CACHE_TIMEOUT = 10800; // 3h
|
||||||
const DESCRIPTION = 'Returns the ads from the first page of search result.';
|
const DESCRIPTION = 'Returns the ads from the first page of search result.';
|
||||||
|
|
||||||
|
|
203
bridges/PatreonBridge.php
Normal file
203
bridges/PatreonBridge.php
Normal file
|
@ -0,0 +1,203 @@
|
||||||
|
<?php
|
||||||
|
class PatreonBridge extends BridgeAbstract {
|
||||||
|
const NAME = 'Patreon Bridge';
|
||||||
|
const URI = 'https://www.patreon.com/';
|
||||||
|
const CACHE_TIMEOUT = 300; // 5min
|
||||||
|
const DESCRIPTION = 'Returns posts by creators on Patreon';
|
||||||
|
const MAINTAINER = 'Roliga';
|
||||||
|
const PARAMETERS = array( array(
|
||||||
|
'creator' => array(
|
||||||
|
'name' => 'Creator',
|
||||||
|
'type' => 'text',
|
||||||
|
'required' => true,
|
||||||
|
'title' => 'Creator name as seen in their page URL'
|
||||||
|
)
|
||||||
|
));
|
||||||
|
|
||||||
|
public function collectData(){
|
||||||
|
$html = getSimpleHTMLDOMCached($this->getURI(), 86400)
|
||||||
|
or returnServerError('Failed to load creator page at ' . $this->getURI());
|
||||||
|
$regex = '#/api/campaigns/([0-9]+)#';
|
||||||
|
if(preg_match($regex, $html->save(), $matches) > 0) {
|
||||||
|
$campaign_id = $matches[1];
|
||||||
|
} else {
|
||||||
|
returnServerError('Could not find campaign ID');
|
||||||
|
}
|
||||||
|
|
||||||
|
$query = array(
|
||||||
|
'include' => implode(',', array(
|
||||||
|
'user',
|
||||||
|
'attachments',
|
||||||
|
'user_defined_tags',
|
||||||
|
//'campaign',
|
||||||
|
//'poll.choices',
|
||||||
|
//'poll.current_user_responses.user',
|
||||||
|
//'poll.current_user_responses.choice',
|
||||||
|
//'poll.current_user_responses.poll',
|
||||||
|
//'access_rules.tier.null',
|
||||||
|
//'images.null',
|
||||||
|
//'audio.null'
|
||||||
|
)),
|
||||||
|
'fields' => array(
|
||||||
|
'post' => implode(',', array(
|
||||||
|
//'change_visibility_at',
|
||||||
|
//'comment_count',
|
||||||
|
'content',
|
||||||
|
//'current_user_can_delete',
|
||||||
|
//'current_user_can_view',
|
||||||
|
//'current_user_has_liked',
|
||||||
|
//'embed',
|
||||||
|
'image',
|
||||||
|
//'is_paid',
|
||||||
|
//'like_count',
|
||||||
|
//'min_cents_pledged_to_view',
|
||||||
|
//'patreon_url',
|
||||||
|
//'patron_count',
|
||||||
|
//'pledge_url',
|
||||||
|
//'post_file',
|
||||||
|
//'post_metadata',
|
||||||
|
//'post_type',
|
||||||
|
'published_at',
|
||||||
|
'teaser_text',
|
||||||
|
//'thumbnail_url',
|
||||||
|
'title',
|
||||||
|
//'upgrade_url',
|
||||||
|
'url',
|
||||||
|
//'was_posted_by_campaign_owner'
|
||||||
|
)),
|
||||||
|
'user' => implode(',', array(
|
||||||
|
//'image_url',
|
||||||
|
'full_name',
|
||||||
|
//'url'
|
||||||
|
))
|
||||||
|
),
|
||||||
|
'filter' => array(
|
||||||
|
'contains_exclusive_posts' => true,
|
||||||
|
'is_draft' => false,
|
||||||
|
'campaign_id' => $campaign_id
|
||||||
|
),
|
||||||
|
'sort' => '-published_at'
|
||||||
|
);
|
||||||
|
$posts = $this->apiGet('posts', $query);
|
||||||
|
|
||||||
|
foreach($posts->data as $post) {
|
||||||
|
$item = array(
|
||||||
|
'uri' => $post->attributes->url,
|
||||||
|
'title' => $post->attributes->title,
|
||||||
|
'timestamp' => $post->attributes->published_at,
|
||||||
|
'content' => '',
|
||||||
|
'uid' => 'patreon.com/' . $post->id
|
||||||
|
);
|
||||||
|
|
||||||
|
$user = $this->findInclude($posts,
|
||||||
|
'user',
|
||||||
|
$post->relationships->user->data->id);
|
||||||
|
$item['author'] = $user->full_name;
|
||||||
|
|
||||||
|
if(isset($post->attributes->image))
|
||||||
|
$item['content'] .= '<p><a href="'
|
||||||
|
. $post->attributes->url
|
||||||
|
. '"><img src="'
|
||||||
|
. $post->attributes->image->thumb_url
|
||||||
|
. '" /></a></p>';
|
||||||
|
|
||||||
|
if(isset($post->attributes->content)) {
|
||||||
|
$item['content'] .= $post->attributes->content;
|
||||||
|
} elseif (isset($post->attributes->teaser_text)) {
|
||||||
|
$item['content'] .= '<p>'
|
||||||
|
. $post->attributes->teaser_text
|
||||||
|
. '</p>';
|
||||||
|
}
|
||||||
|
|
||||||
|
if(isset($post->relationships->user_defined_tags)) {
|
||||||
|
$item['categories'] = array();
|
||||||
|
foreach($post->relationships->user_defined_tags->data as $tag) {
|
||||||
|
$attrs = $this->findInclude($posts, 'post_tag', $tag->id);
|
||||||
|
$item['categories'][] = $attrs->value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if(isset($post->relationships->attachments)) {
|
||||||
|
$item['enclosures'] = array();
|
||||||
|
foreach($post->relationships->attachments->data as $attachment) {
|
||||||
|
$attrs = $this->findInclude($posts, 'attachment', $attachment->id);
|
||||||
|
$item['enclosures'][] = $attrs->url;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->items[] = $item;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Searches the "included" array in an API response and returns attributes
|
||||||
|
* for the first match.
|
||||||
|
*/
|
||||||
|
private function findInclude($data, $type, $id) {
|
||||||
|
foreach($data->included as $include)
|
||||||
|
if($include->type === $type && $include->id === $id)
|
||||||
|
return $include->attributes;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function apiGet($endpoint, $query_data = array()) {
|
||||||
|
$query_data['json-api-version'] = 1.0;
|
||||||
|
$query_data['json-api-use-default-includes'] = 0;
|
||||||
|
|
||||||
|
$url = 'https://www.patreon.com/api/'
|
||||||
|
. $endpoint
|
||||||
|
. '?'
|
||||||
|
. http_build_query($query_data);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Accept-Language header and the CURL cipher list are for bypassing the
|
||||||
|
* Cloudflare anti-bot protection on the Patreon API. If this ever breaks,
|
||||||
|
* here are some other project that also deal with this:
|
||||||
|
* https://github.com/mikf/gallery-dl/issues/342
|
||||||
|
* https://github.com/daemionfox/patreon-feed/issues/7
|
||||||
|
* https://www.patreondevelopers.com/t/api-returning-cloudflare-challenge/2025
|
||||||
|
* https://github.com/splitbrain/patreon-rss/issues/4
|
||||||
|
*/
|
||||||
|
$header = array(
|
||||||
|
'Accept-Language: en-US',
|
||||||
|
'Content-Type: application/json'
|
||||||
|
);
|
||||||
|
$opts = array(
|
||||||
|
CURLOPT_SSL_CIPHER_LIST => implode(':', array(
|
||||||
|
'DEFAULT',
|
||||||
|
'!DHE-RSA-CHACHA20-POLY1305'
|
||||||
|
))
|
||||||
|
);
|
||||||
|
|
||||||
|
$data = json_decode(getContents($url, $header, $opts))
|
||||||
|
or returnServerError('API request to "' . $url . '" failed.');
|
||||||
|
|
||||||
|
return $data;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getName(){
|
||||||
|
if(!is_null($this->getInput('creator')))
|
||||||
|
return $this->getInput('creator') . ' posts';
|
||||||
|
|
||||||
|
return parent::getName();
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getURI(){
|
||||||
|
if(!is_null($this->getInput('creator')))
|
||||||
|
return self::URI . $this->getInput('creator');
|
||||||
|
|
||||||
|
return parent::getURI();
|
||||||
|
}
|
||||||
|
|
||||||
|
public function detectParameters($url){
|
||||||
|
$params = array();
|
||||||
|
|
||||||
|
// Matches e.g. https://www.patreon.com/SomeCreator
|
||||||
|
$regex = '/^(https?:\/\/)?(www\.)?patreon\.com\/([^\/&?\n]+)/';
|
||||||
|
if(preg_match($regex, $url, $matches) > 0) {
|
||||||
|
$params['creator'] = urldecode($matches[3]);
|
||||||
|
return $params;
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
|
@ -3,7 +3,7 @@ class PickyWallpapersBridge extends BridgeAbstract {
|
||||||
|
|
||||||
const MAINTAINER = 'nel50n';
|
const MAINTAINER = 'nel50n';
|
||||||
const NAME = 'PickyWallpapers Bridge';
|
const NAME = 'PickyWallpapers Bridge';
|
||||||
const URI = 'http://www.pickywallpapers.com/';
|
const URI = 'https://www.pickywallpapers.com/';
|
||||||
const CACHE_TIMEOUT = 43200; // 12h
|
const CACHE_TIMEOUT = 43200; // 12h
|
||||||
const DESCRIPTION = 'Returns the latests wallpapers from PickyWallpapers';
|
const DESCRIPTION = 'Returns the latests wallpapers from PickyWallpapers';
|
||||||
|
|
||||||
|
|
|
@ -32,6 +32,13 @@ class PikabuBridge extends BridgeAbstract {
|
||||||
'required' => true
|
'required' => true
|
||||||
),
|
),
|
||||||
'filter' => self::PARAMETERS_FILTER
|
'filter' => self::PARAMETERS_FILTER
|
||||||
|
),
|
||||||
|
'По пользователю' => array(
|
||||||
|
'user' => array(
|
||||||
|
'name' => 'Пользователь',
|
||||||
|
'exampleValue' => 'admin',
|
||||||
|
'required' => true
|
||||||
|
)
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
|
|
||||||
|
@ -40,6 +47,8 @@ class PikabuBridge extends BridgeAbstract {
|
||||||
public function getURI() {
|
public function getURI() {
|
||||||
if ($this->getInput('tag')) {
|
if ($this->getInput('tag')) {
|
||||||
return self::URI . '/tag/' . rawurlencode($this->getInput('tag')) . '/' . rawurlencode($this->getInput('filter'));
|
return self::URI . '/tag/' . rawurlencode($this->getInput('tag')) . '/' . rawurlencode($this->getInput('filter'));
|
||||||
|
} else if ($this->getInput('user')) {
|
||||||
|
return self::URI . '/@' . rawurlencode($this->getInput('user'));
|
||||||
} else if ($this->getInput('community')) {
|
} else if ($this->getInput('community')) {
|
||||||
$uri = self::URI . '/community/' . rawurlencode($this->getInput('community'));
|
$uri = self::URI . '/community/' . rawurlencode($this->getInput('community'));
|
||||||
if ($this->getInput('filter') != 'hot') {
|
if ($this->getInput('filter') != 'hot') {
|
||||||
|
@ -101,6 +110,10 @@ class PikabuBridge extends BridgeAbstract {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
$img->outertext = '<img src="' . $src . '">';
|
$img->outertext = '<img src="' . $src . '">';
|
||||||
|
|
||||||
|
// it is assumed, that img's parents are links to post itself
|
||||||
|
// we don't need them
|
||||||
|
$img->parent()->outertext = $img->outertext;
|
||||||
}
|
}
|
||||||
|
|
||||||
$categories = array();
|
$categories = array();
|
||||||
|
@ -116,7 +129,10 @@ class PikabuBridge extends BridgeAbstract {
|
||||||
$item['categories'] = $categories;
|
$item['categories'] = $categories;
|
||||||
$item['author'] = $post->find('.user__nick', 0)->innertext;
|
$item['author'] = $post->find('.user__nick', 0)->innertext;
|
||||||
$item['title'] = $title->plaintext;
|
$item['title'] = $title->plaintext;
|
||||||
$item['content'] = strip_tags(backgroundToImg($post->find('.story__content-inner', 0)->innertext), '<br><p><img>');
|
$item['content'] = strip_tags(
|
||||||
|
backgroundToImg($post->find('.story__content-inner', 0)->innertext),
|
||||||
|
'<br><p><img><a>
|
||||||
|
');
|
||||||
$item['uri'] = $title->href;
|
$item['uri'] = $title->href;
|
||||||
$item['timestamp'] = strtotime($time->getAttribute('datetime'));
|
$item['timestamp'] = strtotime($time->getAttribute('datetime'));
|
||||||
$this->items[] = $item;
|
$this->items[] = $item;
|
||||||
|
|
|
@ -16,12 +16,6 @@ class PinterestBridge extends FeedExpander {
|
||||||
'name' => 'board',
|
'name' => 'board',
|
||||||
'required' => true
|
'required' => true
|
||||||
)
|
)
|
||||||
),
|
|
||||||
'From search' => array(
|
|
||||||
'q' => array(
|
|
||||||
'name' => 'Keyword',
|
|
||||||
'required' => true
|
|
||||||
)
|
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
|
|
||||||
|
@ -30,21 +24,13 @@ class PinterestBridge extends FeedExpander {
|
||||||
}
|
}
|
||||||
|
|
||||||
public function collectData() {
|
public function collectData() {
|
||||||
switch($this->queriedContext) {
|
|
||||||
case 'By username and board':
|
|
||||||
$this->collectExpandableDatas($this->getURI() . '.rss');
|
$this->collectExpandableDatas($this->getURI() . '.rss');
|
||||||
$this->fixLowRes();
|
$this->fixLowRes();
|
||||||
break;
|
|
||||||
case 'From search':
|
|
||||||
default:
|
|
||||||
$html = getSimpleHTMLDOMCached($this->getURI());
|
|
||||||
$this->getSearchResults($html);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private function fixLowRes() {
|
private function fixLowRes() {
|
||||||
|
|
||||||
$newitems = [];
|
$newitems = array();
|
||||||
$pattern = '/https\:\/\/i\.pinimg\.com\/[a-zA-Z0-9]*x\//';
|
$pattern = '/https\:\/\/i\.pinimg\.com\/[a-zA-Z0-9]*x\//';
|
||||||
foreach($this->items as $item) {
|
foreach($this->items as $item) {
|
||||||
|
|
||||||
|
@ -55,71 +41,21 @@ class PinterestBridge extends FeedExpander {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private function getSearchResults($html){
|
|
||||||
$json = json_decode($html->find('#jsInit1', 0)->innertext, true);
|
|
||||||
$results = $json['resourceDataCache'][0]['data']['results'];
|
|
||||||
|
|
||||||
foreach($results as $result) {
|
|
||||||
$item = array();
|
|
||||||
|
|
||||||
$item['uri'] = self::URI . $result['board']['url'];
|
|
||||||
|
|
||||||
// Some use regular titles, others provide 'advanced' infos, a few
|
|
||||||
// provide even less info. Thus we attempt multiple options.
|
|
||||||
$item['title'] = trim($result['title']);
|
|
||||||
|
|
||||||
if($item['title'] === '')
|
|
||||||
$item['title'] = trim($result['rich_summary']['display_name']);
|
|
||||||
|
|
||||||
if($item['title'] === '')
|
|
||||||
$item['title'] = trim($result['grid_description']);
|
|
||||||
|
|
||||||
$item['timestamp'] = strtotime($result['created_at']);
|
|
||||||
$item['username'] = $result['pinner']['username'];
|
|
||||||
$item['fullname'] = $result['pinner']['full_name'];
|
|
||||||
$item['avatar'] = $result['pinner']['image_small_url'];
|
|
||||||
$item['author'] = $item['username'] . ' (' . $item['fullname'] . ')';
|
|
||||||
$item['content'] = '<img align="left" style="margin: 2px 4px;" src="'
|
|
||||||
. htmlentities($item['avatar'])
|
|
||||||
. '" /><p><strong>'
|
|
||||||
. $item['username']
|
|
||||||
. '</strong><br>'
|
|
||||||
. $item['fullname']
|
|
||||||
. '</p><br><img src="'
|
|
||||||
. $result['images']['736x']['url']
|
|
||||||
. '" alt="" /><br><p>'
|
|
||||||
. $result['description']
|
|
||||||
. '</p>';
|
|
||||||
|
|
||||||
$item['enclosures'] = array($result['images']['orig']['url']);
|
|
||||||
|
|
||||||
$this->items[] = $item;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public function getURI() {
|
public function getURI() {
|
||||||
switch($this->queriedContext) {
|
|
||||||
case 'By username and board':
|
if ($this->queriedContext === 'By username and board') {
|
||||||
$uri = self::URI . '/' . urlencode($this->getInput('u')) . '/' . urlencode($this->getInput('b'));// . '.rss';
|
return self::URI . '/' . urlencode($this->getInput('u')) . '/' . urlencode($this->getInput('b'));
|
||||||
break;
|
|
||||||
case 'From search':
|
|
||||||
$uri = self::URI . '/search/?q=' . urlencode($this->getInput('q'));
|
|
||||||
break;
|
|
||||||
default: return parent::getURI();
|
|
||||||
}
|
}
|
||||||
return $uri;
|
|
||||||
|
return parent::getURI();
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getName() {
|
public function getName() {
|
||||||
switch($this->queriedContext) {
|
|
||||||
case 'By username and board':
|
if ($this->queriedContext === 'By username and board') {
|
||||||
$specific = $this->getInput('u') . ' - ' . $this->getInput('b');
|
return $this->getInput('u') . ' - ' . $this->getInput('b') . ' - ' . self::NAME;
|
||||||
break;
|
|
||||||
case 'From search':
|
|
||||||
$specific = $this->getInput('q');
|
|
||||||
break;
|
|
||||||
default: return parent::getName();
|
|
||||||
}
|
}
|
||||||
return $specific . ' - ' . self::NAME;
|
|
||||||
|
return parent::getName();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
88
bridges/PirateCommunityBridge.php
Normal file
88
bridges/PirateCommunityBridge.php
Normal file
|
@ -0,0 +1,88 @@
|
||||||
|
<?php
|
||||||
|
class PirateCommunityBridge extends BridgeAbstract {
|
||||||
|
const NAME = 'Pirate-Community Bridge';
|
||||||
|
const URI = 'https://raymanpc.com/';
|
||||||
|
const CACHE_TIMEOUT = 300; // 5min
|
||||||
|
const DESCRIPTION = 'Returns replies to topics';
|
||||||
|
const MAINTAINER = 'Roliga';
|
||||||
|
const PARAMETERS = array( array(
|
||||||
|
't' => array(
|
||||||
|
'name' => 'Topic ID',
|
||||||
|
'type' => 'number',
|
||||||
|
'title' => 'Topic ID from topic URL. If the URL contains t=12 the ID is 12.',
|
||||||
|
'required' => true
|
||||||
|
)));
|
||||||
|
|
||||||
|
private $feedName = '';
|
||||||
|
|
||||||
|
public function detectParameters($url){
|
||||||
|
$parsed_url = parse_url($url);
|
||||||
|
|
||||||
|
if($parsed_url['host'] !== 'raymanpc.com')
|
||||||
|
return null;
|
||||||
|
|
||||||
|
parse_str($parsed_url['query'], $parsed_query);
|
||||||
|
|
||||||
|
if($parsed_url['path'] === '/forum/viewtopic.php'
|
||||||
|
&& array_key_exists('t', $parsed_query)) {
|
||||||
|
return array('t' => $parsed_query['t']);
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getName() {
|
||||||
|
if(!empty($this->feedName))
|
||||||
|
return $this->feedName;
|
||||||
|
|
||||||
|
return parent::getName();
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getURI(){
|
||||||
|
if(!is_null($this->getInput('t'))) {
|
||||||
|
return self::URI
|
||||||
|
. 'forum/viewtopic.php?t='
|
||||||
|
. $this->getInput('t')
|
||||||
|
. '&sd=d'; // sort posts decending by ate so first page has latest posts
|
||||||
|
}
|
||||||
|
|
||||||
|
return parent::getURI();
|
||||||
|
}
|
||||||
|
|
||||||
|
public function collectData(){
|
||||||
|
$html = getSimpleHTMLDOM($this->getURI())
|
||||||
|
or returnServerError('Could not retrieve topic page at ' . $this->getURI());
|
||||||
|
|
||||||
|
$this->feedName = $html->find('head title', 0)->plaintext;
|
||||||
|
|
||||||
|
foreach($html->find('.post') as $reply) {
|
||||||
|
$item = array();
|
||||||
|
|
||||||
|
$item['uri'] = $this->getURI()
|
||||||
|
. $reply->find('h3 a', 0)->getAttribute('href');
|
||||||
|
|
||||||
|
$item['title'] = $reply->find('h3 a', 0)->plaintext;
|
||||||
|
|
||||||
|
$author_html = $reply->find('.author', 0);
|
||||||
|
// author_html contains the timestamp as text directly inside it,
|
||||||
|
// so delete all other child elements
|
||||||
|
foreach($author_html->children as $child)
|
||||||
|
$child->outertext = '';
|
||||||
|
// Timestamps are always in UTC+1
|
||||||
|
$item['timestamp'] = trim($author_html->innertext) . ' +01:00';
|
||||||
|
|
||||||
|
$item['author'] = $reply
|
||||||
|
->find('.username, .username-coloured', 0)
|
||||||
|
->plaintext;
|
||||||
|
|
||||||
|
$item['content'] = defaultLinkTo($reply->find('.content', 0)->innertext,
|
||||||
|
$this->getURI());
|
||||||
|
|
||||||
|
$item['enclosures'] = array();
|
||||||
|
foreach($reply->find('.attachbox img.postimage') as $img)
|
||||||
|
$item['enclosures'][] = urljoin($this->getURI(), $img->src);
|
||||||
|
|
||||||
|
$this->items[] = $item;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
67
bridges/PlantUMLReleasesBridge.php
Normal file
67
bridges/PlantUMLReleasesBridge.php
Normal file
|
@ -0,0 +1,67 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* PlantUML releases bridge showing latest releases content
|
||||||
|
* @author nicolas-delsaux
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
class PlantUMLReleasesBridge extends BridgeAbstract
|
||||||
|
{
|
||||||
|
const MAINTAINER = 'Riduidel';
|
||||||
|
|
||||||
|
const NAME = 'PlantUML Releases';
|
||||||
|
|
||||||
|
const AUTHOR = 'PlantUML team';
|
||||||
|
|
||||||
|
// URI is no more valid, since we can address the whole gq galaxy
|
||||||
|
const URI = 'http://plantuml.com/fr/changes';
|
||||||
|
|
||||||
|
const CACHE_TIMEOUT = 7200; // 2h
|
||||||
|
const DESCRIPTION = 'PlantUML releases bridge, showing for each release the changelog';
|
||||||
|
|
||||||
|
const DEFAULT_DOMAIN = 'plantuml.com';
|
||||||
|
|
||||||
|
const PARAMETERS = array( array(
|
||||||
|
));
|
||||||
|
|
||||||
|
const REPLACED_ATTRIBUTES = array(
|
||||||
|
'href' => 'href',
|
||||||
|
'src' => 'src',
|
||||||
|
'data-original' => 'src'
|
||||||
|
);
|
||||||
|
|
||||||
|
private function getDomain() {
|
||||||
|
$domain = $this->getInput('domain');
|
||||||
|
if (empty($domain))
|
||||||
|
$domain = self::DEFAULT_DOMAIN;
|
||||||
|
if (strpos($domain, '://') === false)
|
||||||
|
$domain = 'https://' . $domain;
|
||||||
|
return $domain;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getURI()
|
||||||
|
{
|
||||||
|
return self::URI;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function collectData()
|
||||||
|
{
|
||||||
|
$html = getSimpleHTMLDOM($this->getURI()) or returnServerError('Could not request ' . $this->getURI());
|
||||||
|
|
||||||
|
// Since GQ don't want simple class scrapping, let's do it the hard way and ... discover content !
|
||||||
|
$main = $html->find('div[id=root]', 0);
|
||||||
|
foreach ($main->find('h2') as $release) {
|
||||||
|
$item = array();
|
||||||
|
$item['author'] = self::AUTHOR;
|
||||||
|
$release_text = $release->innertext;
|
||||||
|
if (preg_match('/(.+) \((.*)\)/', $release_text, $matches)) {
|
||||||
|
$item['title'] = $matches[1];
|
||||||
|
// And now, build the date from the date text
|
||||||
|
$item['timestamp'] = strtotime($matches[2]);
|
||||||
|
}
|
||||||
|
$item['uri'] = $this->getURI();
|
||||||
|
$item['content'] = $release->next_sibling ();
|
||||||
|
$this->items[] = $item;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,44 +0,0 @@
|
||||||
<?php
|
|
||||||
class ReadComicsBridge extends BridgeAbstract {
|
|
||||||
|
|
||||||
const MAINTAINER = 'niawag';
|
|
||||||
const NAME = 'Read Comics';
|
|
||||||
const URI = 'http://www.readcomics.tv/';
|
|
||||||
const DESCRIPTION = 'Enter the comics as they appear in the website uri,
|
|
||||||
separated by semicolons, ex: good-comic-1;good-comic-2; ...';
|
|
||||||
|
|
||||||
const PARAMETERS = array( array(
|
|
||||||
'q' => array(
|
|
||||||
'name' => 'keywords, separated by semicolons',
|
|
||||||
'exampleValue' => 'first list;second list;...',
|
|
||||||
'required' => true
|
|
||||||
),
|
|
||||||
));
|
|
||||||
|
|
||||||
public function collectData(){
|
|
||||||
|
|
||||||
function parseDateTimestamp($element){
|
|
||||||
$guessedDate = $element->find('span', 0)->plaintext;
|
|
||||||
$guessedDate = strptime($guessedDate, '%m/%d/%Y');
|
|
||||||
$timestamp = mktime(0, 0, 0, $guessedDate['tm_mon'] + 1, $guessedDate['tm_mday'], date('Y'));
|
|
||||||
|
|
||||||
return $timestamp;
|
|
||||||
}
|
|
||||||
|
|
||||||
$keywordsList = explode(';', $this->getInput('q'));
|
|
||||||
foreach($keywordsList as $keywords) {
|
|
||||||
$html = $this->getSimpleHTMLDOM(self::URI . 'comic/' . rawurlencode($keywords))
|
|
||||||
or $this->returnServerError('Could not request readcomics.tv.');
|
|
||||||
|
|
||||||
foreach($html->find('li') as $element) {
|
|
||||||
$item = array();
|
|
||||||
$item['uri'] = $element->find('a.ch-name', 0)->href;
|
|
||||||
$item['id'] = $item['uri'];
|
|
||||||
$item['timestamp'] = parseDateTimestamp($element);
|
|
||||||
$item['title'] = $element->find('a.ch-name', 0)->plaintext;
|
|
||||||
if(isset($item['title']))
|
|
||||||
$this->items[] = $item;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
40
bridges/RedditBridge.php
Normal file
40
bridges/RedditBridge.php
Normal file
|
@ -0,0 +1,40 @@
|
||||||
|
<?php
|
||||||
|
class RedditBridge extends FeedExpander {
|
||||||
|
|
||||||
|
const MAINTAINER = 'leomaradan';
|
||||||
|
const NAME = 'Reddit Bridge';
|
||||||
|
const URI = 'https://www.reddit.com/';
|
||||||
|
const DESCRIPTION = 'Reddit RSS Feed fixer';
|
||||||
|
|
||||||
|
const PARAMETERS = array(
|
||||||
|
'single' => array(
|
||||||
|
'r' => array(
|
||||||
|
'name' => 'SubReddit',
|
||||||
|
'required' => true,
|
||||||
|
'exampleValue' => 'selfhosted',
|
||||||
|
'title' => 'SubReddit name'
|
||||||
|
)
|
||||||
|
),
|
||||||
|
'multi' => array(
|
||||||
|
'rs' => array(
|
||||||
|
'name' => 'SubReddits',
|
||||||
|
'required' => true,
|
||||||
|
'exampleValue' => 'selfhosted, php',
|
||||||
|
'title' => 'SubReddit names, separated by commas'
|
||||||
|
)
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
public function collectData(){
|
||||||
|
|
||||||
|
switch($this->queriedContext) {
|
||||||
|
case 'single': $subreddits[] = $this->getInput('r'); break;
|
||||||
|
case 'multi': $subreddits = explode(',', $this->getInput('rs')); break;
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach ($subreddits as $subreddit) {
|
||||||
|
$name = trim($subreddit);
|
||||||
|
$this->collectExpandableDatas("https://www.reddit.com/r/$name/.rss");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -9,22 +9,6 @@ class Releases3DSBridge extends BridgeAbstract {
|
||||||
|
|
||||||
public function collectData(){
|
public function collectData(){
|
||||||
|
|
||||||
function typeToString($type){
|
|
||||||
switch($type) {
|
|
||||||
case 1: return '3DS Game';
|
|
||||||
case 4: return 'eShop';
|
|
||||||
default: return '??? (' . $type . ')';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function cardToString($card){
|
|
||||||
switch($card) {
|
|
||||||
case 1: return 'Regular (CARD1)';
|
|
||||||
case 2: return 'NAND (CARD2)';
|
|
||||||
default: return '??? (' . $card . ')';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
$dataUrl = self::URI . 'xml.php';
|
$dataUrl = self::URI . 'xml.php';
|
||||||
$xml = getContents($dataUrl)
|
$xml = getContents($dataUrl)
|
||||||
or returnServerError('Could not request 3dsdb: ' . $dataUrl);
|
or returnServerError('Could not request 3dsdb: ' . $dataUrl);
|
||||||
|
@ -95,8 +79,8 @@ class Releases3DSBridge extends BridgeAbstract {
|
||||||
. '<br /><b>Release Name: </b>' . $releasename
|
. '<br /><b>Release Name: </b>' . $releasename
|
||||||
. '<br /><b>Trimmed size: </b>' . intval(intval($trimmedsize) / 1048576)
|
. '<br /><b>Trimmed size: </b>' . intval(intval($trimmedsize) / 1048576)
|
||||||
. 'MB<br /><b>Firmware: </b>' . $firmware
|
. 'MB<br /><b>Firmware: </b>' . $firmware
|
||||||
. '<br /><b>Type: </b>' . typeToString($type)
|
. '<br /><b>Type: </b>' . $this->typeToString($type)
|
||||||
. '<br /><b>Card: </b>' . cardToString($card)
|
. '<br /><b>Card: </b>' . $this->cardToString($card)
|
||||||
. '<br />';
|
. '<br />';
|
||||||
|
|
||||||
//Build search links section to facilitate release search using search engines
|
//Build search links section to facilitate release search using search engines
|
||||||
|
@ -124,4 +108,20 @@ class Releases3DSBridge extends BridgeAbstract {
|
||||||
$limit++;
|
$limit++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private function typeToString($type){
|
||||||
|
switch($type) {
|
||||||
|
case 1: return '3DS Game';
|
||||||
|
case 4: return 'eShop';
|
||||||
|
default: return '??? (' . $type . ')';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private function cardToString($card){
|
||||||
|
switch($card) {
|
||||||
|
case 1: return 'Regular (CARD1)';
|
||||||
|
case 2: return 'NAND (CARD2)';
|
||||||
|
default: return '??? (' . $card . ')';
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,7 +3,7 @@ class ReporterreBridge extends BridgeAbstract {
|
||||||
|
|
||||||
const MAINTAINER = 'nyutag';
|
const MAINTAINER = 'nyutag';
|
||||||
const NAME = 'Reporterre Bridge';
|
const NAME = 'Reporterre Bridge';
|
||||||
const URI = 'http://www.reporterre.net/';
|
const URI = 'https://www.reporterre.net/';
|
||||||
const DESCRIPTION = 'Returns the newest articles.';
|
const DESCRIPTION = 'Returns the newest articles.';
|
||||||
|
|
||||||
private function extractContent($url){
|
private function extractContent($url){
|
||||||
|
|
|
@ -25,7 +25,7 @@ class RoadAndTrackBridge extends BridgeAbstract {
|
||||||
|
|
||||||
private function fixImages($content) {
|
private function fixImages($content) {
|
||||||
|
|
||||||
$enclosures = [];
|
$enclosures = array();
|
||||||
foreach($content->find('img') as $image) {
|
foreach($content->find('img') as $image) {
|
||||||
$image->src = explode('?', $image->getAttribute('data-src'))[0];
|
$image->src = explode('?', $image->getAttribute('data-src'))[0];
|
||||||
$enclosures[] = $image->src;
|
$enclosures[] = $image->src;
|
||||||
|
|
|
@ -5,7 +5,7 @@ class Rule34Bridge extends GelbooruBridge {
|
||||||
|
|
||||||
const MAINTAINER = 'mitsukarenai';
|
const MAINTAINER = 'mitsukarenai';
|
||||||
const NAME = 'Rule34';
|
const NAME = 'Rule34';
|
||||||
const URI = 'http://rule34.xxx/';
|
const URI = 'https://rule34.xxx/';
|
||||||
const DESCRIPTION = 'Returns images from given page';
|
const DESCRIPTION = 'Returns images from given page';
|
||||||
|
|
||||||
const PIDBYPAGE = 50;
|
const PIDBYPAGE = 50;
|
||||||
|
|
|
@ -5,6 +5,23 @@ class Rule34pahealBridge extends Shimmie2Bridge {
|
||||||
|
|
||||||
const MAINTAINER = 'mitsukarenai';
|
const MAINTAINER = 'mitsukarenai';
|
||||||
const NAME = 'Rule34paheal';
|
const NAME = 'Rule34paheal';
|
||||||
const URI = 'http://rule34.paheal.net/';
|
const URI = 'https://rule34.paheal.net/';
|
||||||
const DESCRIPTION = 'Returns images from given page';
|
const DESCRIPTION = 'Returns images from given page';
|
||||||
|
|
||||||
|
protected function getItemFromElement($element){
|
||||||
|
$item = array();
|
||||||
|
$item['uri'] = $this->getURI() . $element->href;
|
||||||
|
$item['id'] = (int)preg_replace('/[^0-9]/', '', $element->getAttribute(static::IDATTRIBUTE));
|
||||||
|
$item['timestamp'] = time();
|
||||||
|
$thumbnailUri = $element->find('img', 0)->src;
|
||||||
|
$item['tags'] = $element->getAttribute('data-tags');
|
||||||
|
$item['title'] = $this->getName() . ' | ' . $item['id'];
|
||||||
|
$item['content'] = '<a href="'
|
||||||
|
. $item['uri']
|
||||||
|
. '"><img src="'
|
||||||
|
. $thumbnailUri
|
||||||
|
. '" /></a><br>Tags: '
|
||||||
|
. $item['tags'];
|
||||||
|
return $item;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -5,7 +5,7 @@ class SafebooruBridge extends GelbooruBridge {
|
||||||
|
|
||||||
const MAINTAINER = 'mitsukarenai';
|
const MAINTAINER = 'mitsukarenai';
|
||||||
const NAME = 'Safebooru';
|
const NAME = 'Safebooru';
|
||||||
const URI = 'http://safebooru.org/';
|
const URI = 'https://safebooru.org/';
|
||||||
const DESCRIPTION = 'Returns images from given page';
|
const DESCRIPTION = 'Returns images from given page';
|
||||||
|
|
||||||
const PIDBYPAGE = 40;
|
const PIDBYPAGE = 40;
|
||||||
|
|
|
@ -1,11 +0,0 @@
|
||||||
<?php
|
|
||||||
require_once('MoebooruBridge.php');
|
|
||||||
|
|
||||||
class SakugabooruBridge extends MoebooruBridge {
|
|
||||||
|
|
||||||
const MAINTAINER = 'mitsukarenai';
|
|
||||||
const NAME = 'Sakugabooru';
|
|
||||||
const URI = 'http://sakuga.yshi.org/';
|
|
||||||
const DESCRIPTION = 'Returns images from given page';
|
|
||||||
|
|
||||||
}
|
|
|
@ -3,7 +3,7 @@ class ScmbBridge extends BridgeAbstract {
|
||||||
|
|
||||||
const MAINTAINER = 'Astalaseven';
|
const MAINTAINER = 'Astalaseven';
|
||||||
const NAME = 'Se Coucher Moins Bête Bridge';
|
const NAME = 'Se Coucher Moins Bête Bridge';
|
||||||
const URI = 'http://secouchermoinsbete.fr';
|
const URI = 'https://secouchermoinsbete.fr';
|
||||||
const CACHE_TIMEOUT = 21600; // 6h
|
const CACHE_TIMEOUT = 21600; // 6h
|
||||||
const DESCRIPTION = 'Returns the newest anecdotes.';
|
const DESCRIPTION = 'Returns the newest anecdotes.';
|
||||||
|
|
||||||
|
|
Some files were not shown because too many files have changed in this diff Show more
Loading…
Reference in a new issue