mirror of
https://github.com/bellingcat/vk-url-scraper.git
synced 2026-06-12 21:38:36 +03:00
Compare commits
30 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
798684a334 | ||
|
|
a556b237e9 | ||
|
|
283bc35658 | ||
|
|
cef70fb80d | ||
|
|
e66ef4f477 | ||
|
|
1f6a8368fd | ||
|
|
9a046fd1cb | ||
|
|
aae2bb5999 | ||
|
|
9e30b81d16 | ||
|
|
72bc355606 | ||
|
|
7f59eefb73 | ||
|
|
30003c524e | ||
|
|
d1b27bef1d | ||
|
|
e5e9e08ee6 | ||
|
|
3a8a3f54c0 | ||
|
|
4d73864dbb | ||
|
|
ceaa8e45f3 | ||
|
|
007c8e07a8 | ||
|
|
a515b2c3de | ||
|
|
54540cd132 | ||
|
|
cfb13e5d82 | ||
|
|
926c3cb8a4 | ||
|
|
15ebe2e66c | ||
|
|
eaff88b2d9 | ||
|
|
a6d066a192 | ||
|
|
9078a17400 | ||
|
|
17b516bd7f | ||
|
|
8bd182b041 | ||
|
|
0b8abfb5cb | ||
|
|
cf5fb91c84 |
5
.github/actions/setup-venv/action.yml
vendored
5
.github/actions/setup-venv/action.yml
vendored
@@ -16,6 +16,11 @@ runs:
|
|||||||
with:
|
with:
|
||||||
python-version: ${{ inputs.python-version }}
|
python-version: ${{ inputs.python-version }}
|
||||||
|
|
||||||
|
- shell: bash
|
||||||
|
run: |
|
||||||
|
# install ffmpeg
|
||||||
|
sudo apt install ffmpeg
|
||||||
|
|
||||||
- shell: bash
|
- shell: bash
|
||||||
run: |
|
run: |
|
||||||
# Install prerequisites.
|
# Install prerequisites.
|
||||||
|
|||||||
3
.github/workflows/main.yml
vendored
3
.github/workflows/main.yml
vendored
@@ -30,8 +30,7 @@ jobs:
|
|||||||
strategy:
|
strategy:
|
||||||
fail-fast: false
|
fail-fast: false
|
||||||
matrix:
|
matrix:
|
||||||
# python: ['3.7', '3.10']
|
python: ['3.7', '3.10']
|
||||||
python: ['3.10']
|
|
||||||
task: # --show-capture=no on purpose
|
task: # --show-capture=no on purpose
|
||||||
- name: Test
|
- name: Test
|
||||||
run: |
|
run: |
|
||||||
|
|||||||
1
.gitignore
vendored
1
.gitignore
vendored
@@ -1,6 +1,7 @@
|
|||||||
.env
|
.env
|
||||||
vk_config.v2.json
|
vk_config.v2.json
|
||||||
output/
|
output/
|
||||||
|
tmp*/
|
||||||
# build artifacts
|
# build artifacts
|
||||||
|
|
||||||
.eggs/
|
.eggs/
|
||||||
|
|||||||
5
Pipfile
5
Pipfile
@@ -18,7 +18,7 @@ pytest-sphinx = "*"
|
|||||||
pytest-cov = "*"
|
pytest-cov = "*"
|
||||||
twine = ">=1.11.0"
|
twine = ">=1.11.0"
|
||||||
sphinx = ">=4.3.0,<5.1.0"
|
sphinx = ">=4.3.0,<5.1.0"
|
||||||
furo = "==2022.6.4.1"
|
furo = "==2022.6.21"
|
||||||
myst-parser = ">=0.15.2,<0.19.0"
|
myst-parser = ">=0.15.2,<0.19.0"
|
||||||
sphinx-autobuild = "==2021.3.14"
|
sphinx-autobuild = "==2021.3.14"
|
||||||
sphinx-autodoc-typehints = "*"
|
sphinx-autodoc-typehints = "*"
|
||||||
@@ -26,3 +26,6 @@ python-dotenv = "*"
|
|||||||
|
|
||||||
[requires]
|
[requires]
|
||||||
python_version = "3.9"
|
python_version = "3.9"
|
||||||
|
|
||||||
|
[pipenv]
|
||||||
|
allow_prereleases = true
|
||||||
|
|||||||
1211
Pipfile.lock
generated
1211
Pipfile.lock
generated
File diff suppressed because it is too large
Load Diff
16
README.md
16
README.md
@@ -1,7 +1,13 @@
|
|||||||
# vk-url-scraper
|
# vk-url-scraper
|
||||||
Library to scrape data and especially media links (videos and photos) from vk.com URLs.
|
Python library to scrape data, and especially media links like videos and photos, from vk.com URLs.
|
||||||
|
|
||||||
You can use it via the [command line](#command-line-usage) or as a [python library](#python-library-usage).
|
|
||||||
|
[](https://badge.fury.io/py/vk-url-scraper)
|
||||||
|
[](https://pypi.python.org/pypi/vk-url-scraper/)
|
||||||
|
[](https://vk-url-scraper.readthedocs.io/en/latest/?badge=latest)
|
||||||
|
|
||||||
|
|
||||||
|
You can use it via the [command line](#command-line-usage) or as a [python library](#python-library-usage), check the **[documentation](https://vk-url-scraper.readthedocs.io/en/latest/)**.
|
||||||
|
|
||||||
## Installation
|
## Installation
|
||||||
You can install the most recent release from [pypi](https://pypi.org/project/vk-url-scraper/) via `pip install vk-url-scraper`.
|
You can install the most recent release from [pypi](https://pypi.org/project/vk-url-scraper/) via `pip install vk-url-scraper`.
|
||||||
@@ -21,7 +27,8 @@ vk_url_scraper -u "username here" -p "password here" --urls https://vk.com/wall1
|
|||||||
vk_url_scraper -u "username here" -p "password here" --urls https://vk.com/wall12345_6789 https://vk.com/photo-12345_6789 https://vk.com/video12345_6789
|
vk_url_scraper -u "username here" -p "password here" --urls https://vk.com/wall12345_6789 https://vk.com/photo-12345_6789 https://vk.com/video12345_6789
|
||||||
|
|
||||||
# you can pass a token as well to avoid always authenticating
|
# you can pass a token as well to avoid always authenticating
|
||||||
# and possibly getting captch prompts
|
# and possibly getting captcha prompts
|
||||||
|
# you can fetch the token from the bk_config.v2.json file generated under by searching for "access_token"
|
||||||
vk_url_scraper -u "username" -p "password" -t "vktoken goes here" --urls https://vk.com/wall12345_6789
|
vk_url_scraper -u "username" -p "password" -t "vktoken goes here" --urls https://vk.com/wall12345_6789
|
||||||
|
|
||||||
# save the JSON output into a file
|
# save the JSON output into a file
|
||||||
@@ -48,7 +55,7 @@ res = vks.scrape("https://vk.com/wall-1_398461")
|
|||||||
|
|
||||||
# scrape any "video" URL
|
# scrape any "video" URL
|
||||||
res = vks.scrape("https://vk.com/video-6596301_145810025")
|
res = vks.scrape("https://vk.com/video-6596301_145810025")
|
||||||
print(res[0]["text]) # eg: -> to get the text from code
|
print(res[0]["text"]) # eg: -> to get the text from code
|
||||||
```
|
```
|
||||||
|
|
||||||
```python
|
```python
|
||||||
@@ -93,6 +100,7 @@ To test the command line interface available in [__main__.py](__vk_url_scraper/_
|
|||||||
2. run `./scripts/release.sh` to create a tag and push, alternatively
|
2. run `./scripts/release.sh` to create a tag and push, alternatively
|
||||||
1. `git tag vx.y.z` to tag version
|
1. `git tag vx.y.z` to tag version
|
||||||
2. `git push origin vx.y.z` -> this will trigger workflow and put project on [pypi](https://pypi.org/project/vk-url-scraper/)
|
2. `git push origin vx.y.z` -> this will trigger workflow and put project on [pypi](https://pypi.org/project/vk-url-scraper/)
|
||||||
|
3. go to https://readthedocs.org/ to deploy new docs version (if webhook is not setup)
|
||||||
|
|
||||||
### Fixing a failed release
|
### Fixing a failed release
|
||||||
|
|
||||||
|
|||||||
@@ -2,14 +2,94 @@
|
|||||||
# These requirements were autogenerated by pipenv
|
# These requirements were autogenerated by pipenv
|
||||||
# To regenerate from the project's Pipfile, run:
|
# To regenerate from the project's Pipfile, run:
|
||||||
#
|
#
|
||||||
# pipenv lock --requirements
|
# pipenv lock --requirements --dev
|
||||||
#
|
#
|
||||||
|
|
||||||
certifi==2022.6.15
|
# Note: in pipenv 2020.x, "--dev" changed to emit both default and development
|
||||||
charset-normalizer==2.0.12
|
# requirements. To emit only development requirements, pass "--dev-only".
|
||||||
idna==3.3
|
|
||||||
requests==2.28.0
|
# -i https://pypi.org/simple
|
||||||
urllib3==1.26.9
|
alabaster==0.7.13; python_version >= '3.6'
|
||||||
vk-api==11.9.8
|
attrs==22.2.0; python_version >= '3.6'
|
||||||
python-dotenv==0.20.0
|
babel==2.11.0; python_version >= '3.6'
|
||||||
yt-dlp==2022.5.18
|
beautifulsoup4==4.11.2; python_version >= '3.6'
|
||||||
|
black==22.3.0
|
||||||
|
bleach==6.0.0; python_version >= '3.7'
|
||||||
|
brotli==1.0.9; platform_python_implementation == 'CPython'
|
||||||
|
certifi==2022.12.7; python_version >= '3.6'
|
||||||
|
cffi==1.15.1
|
||||||
|
charset-normalizer==3.0.1; python_version >= '3.6'
|
||||||
|
click==8.1.3; python_version >= '3.7'
|
||||||
|
colorama==0.4.6; python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3, 3.4, 3.5, 3.6'
|
||||||
|
coverage[toml]==7.2.0; python_version >= '3.7'
|
||||||
|
cryptography==39.0.1; python_version >= '3.6'
|
||||||
|
docutils==0.18.1; python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3, 3.4'
|
||||||
|
exceptiongroup==1.1.0; python_version < '3.11'
|
||||||
|
flake8==6.0.0
|
||||||
|
furo==2022.6.21
|
||||||
|
idna==3.4; python_version >= '3.5'
|
||||||
|
imagesize==1.4.1; python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3'
|
||||||
|
importlib-metadata==6.0.0; python_version >= '3.7'
|
||||||
|
iniconfig==2.0.0; python_version >= '3.7'
|
||||||
|
isort==5.10.1
|
||||||
|
jaraco.classes==3.2.3; python_version >= '3.7'
|
||||||
|
jeepney==0.8.0; sys_platform == 'linux'
|
||||||
|
jinja2==3.1.2; python_version >= '3.7'
|
||||||
|
keyring==23.13.1; python_version >= '3.7'
|
||||||
|
livereload==2.6.3
|
||||||
|
markdown-it-py==2.2.0; python_version >= '3.7'
|
||||||
|
markupsafe==2.1.2; python_version >= '3.7'
|
||||||
|
mccabe==0.7.0; python_version >= '3.6'
|
||||||
|
mdit-py-plugins==0.3.4; python_version >= '3.7'
|
||||||
|
mdurl==0.1.2; python_version >= '3.7'
|
||||||
|
more-itertools==9.0.0; python_version >= '3.7'
|
||||||
|
mutagen==1.46.0; python_version >= '3.7'
|
||||||
|
mypy-extensions==1.0.0; python_version >= '3.5'
|
||||||
|
mypy==0.961
|
||||||
|
myst-parser==0.18.1
|
||||||
|
packaging==23.0; python_version >= '3.7'
|
||||||
|
pathspec==0.11.0; python_version >= '3.7'
|
||||||
|
pkginfo==1.9.6; python_version >= '3.6'
|
||||||
|
platformdirs==3.0.0; python_version >= '3.7'
|
||||||
|
pluggy==1.0.0; python_version >= '3.6'
|
||||||
|
pycodestyle==2.10.0; python_version >= '3.6'
|
||||||
|
pycparser==2.21
|
||||||
|
pycryptodomex==3.17; python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3, 3.4'
|
||||||
|
pyflakes==3.0.1; python_version >= '3.6'
|
||||||
|
pygments==2.14.0; python_version >= '3.6'
|
||||||
|
pytest-cov==4.0.0
|
||||||
|
pytest-sphinx==0.5.0
|
||||||
|
pytest==7.2.1
|
||||||
|
python-dotenv==0.21.1
|
||||||
|
pytz==2022.7.1
|
||||||
|
pyyaml==6.0; python_version >= '3.6'
|
||||||
|
readme-renderer==37.3; python_version >= '3.7'
|
||||||
|
requests-toolbelt==0.10.1; python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3'
|
||||||
|
requests==2.28.2; python_version >= '3.7' and python_version < '4'
|
||||||
|
rfc3986==2.0.0; python_version >= '3.7'
|
||||||
|
rich==13.3.1; python_version >= '3.7'
|
||||||
|
secretstorage==3.3.3; sys_platform == 'linux'
|
||||||
|
six==1.16.0; python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3'
|
||||||
|
snowballstemmer==2.2.0
|
||||||
|
soupsieve==2.4; python_version >= '3.7'
|
||||||
|
sphinx-autobuild==2021.3.14
|
||||||
|
sphinx-autodoc-typehints==1.19.1
|
||||||
|
sphinx-basic-ng==1.0.0b1; python_version >= '3.7'
|
||||||
|
sphinx-copybutton==0.5.0
|
||||||
|
sphinx==5.0.2
|
||||||
|
sphinxcontrib-applehelp==1.0.4; python_version >= '3.8'
|
||||||
|
sphinxcontrib-devhelp==1.0.2; python_version >= '3.5'
|
||||||
|
sphinxcontrib-htmlhelp==2.0.1; python_version >= '3.8'
|
||||||
|
sphinxcontrib-jsmath==1.0.1; python_version >= '3.5'
|
||||||
|
sphinxcontrib-qthelp==1.0.3; python_version >= '3.5'
|
||||||
|
sphinxcontrib-serializinghtml==1.1.5; python_version >= '3.5'
|
||||||
|
tomli==2.0.1; python_version < '3.11'
|
||||||
|
tornado==6.2; python_version > '2.7'
|
||||||
|
twine==4.0.2
|
||||||
|
typing-extensions==4.5.0; python_version >= '3.7'
|
||||||
|
urllib3==1.26.14; python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3, 3.4, 3.5'
|
||||||
|
vk-api==11.9.9
|
||||||
|
webencodings==0.5.1
|
||||||
|
websockets==10.4; python_version >= '3.7'
|
||||||
|
yt-dlp==2023.2.17
|
||||||
|
zipp==3.14.0; python_version >= '3.7'
|
||||||
|
|||||||
5
setup.py
5
setup.py
@@ -44,7 +44,10 @@ setup(
|
|||||||
"Programming Language :: Python :: 3",
|
"Programming Language :: Python :: 3",
|
||||||
],
|
],
|
||||||
keywords=["scraper", "vk", "vkontakte", "vk-api", "media-downloader"],
|
keywords=["scraper", "vk", "vkontakte", "vk-api", "media-downloader"],
|
||||||
url="https://github.com/bellingcat/vk-url-scraper",
|
project_urls={
|
||||||
|
"Code": "https://github.com/bellingcat/vk-url-scraper",
|
||||||
|
"Documentation": "https://vk-url-scraper.readthedocs.io/en/latest/",
|
||||||
|
},
|
||||||
author="Bellingcat",
|
author="Bellingcat",
|
||||||
author_email="tech@bellingcat.com",
|
author_email="tech@bellingcat.com",
|
||||||
license="MIT",
|
license="MIT",
|
||||||
|
|||||||
@@ -14,6 +14,17 @@ def test_login_fail():
|
|||||||
VkScraper("invalid", "combination")
|
VkScraper("invalid", "combination")
|
||||||
|
|
||||||
|
|
||||||
|
def test_login_custom_file():
|
||||||
|
session_filename = "test-session.json"
|
||||||
|
VkScraper(
|
||||||
|
os.environ["VK_USERNAME"],
|
||||||
|
os.environ["VK_PASSWORD"],
|
||||||
|
session_file=session_filename,
|
||||||
|
)
|
||||||
|
assert os.path.isfile(session_filename)
|
||||||
|
os.unlink(session_filename)
|
||||||
|
|
||||||
|
|
||||||
def test_login_success():
|
def test_login_success():
|
||||||
global vks
|
global vks
|
||||||
vks = VkScraper(
|
vks = VkScraper(
|
||||||
@@ -69,7 +80,7 @@ def test_scrape_wall_url_with_photos():
|
|||||||
== "Хабаровск\nАллея героев\nПомолимся об укокоении воинов:\nАлександра, Игоря, Эдуарда, \nДионисия, Евгения, Александра, Артемия, Иннокентия, Андрея."
|
== "Хабаровск\nАллея героев\nПомолимся об укокоении воинов:\nАлександра, Игоря, Эдуарда, \nДионисия, Евгения, Александра, Артемия, Иннокентия, Андрея."
|
||||||
)
|
)
|
||||||
assert str(res[0]["datetime"]) == str(datetime.datetime(2022, 6, 15, 10, 37, 24))
|
assert str(res[0]["datetime"]) == str(datetime.datetime(2022, 6, 15, 10, 37, 24))
|
||||||
assert len(res[0]["payload"]) == 16
|
assert len(res[0]["payload"]) == 17
|
||||||
assert len(res[0]["attachments"].keys()) == 1
|
assert len(res[0]["attachments"].keys()) == 1
|
||||||
assert list(res[0]["attachments"].keys()) == ["photo"]
|
assert list(res[0]["attachments"].keys()) == ["photo"]
|
||||||
assert len(res[0]["attachments"]["photo"]) == 9
|
assert len(res[0]["attachments"]["photo"]) == 9
|
||||||
@@ -81,7 +92,7 @@ def test_scrape_wall_url_with_photos_inner_videos_and_links_with_inner_photos():
|
|||||||
assert res[0]["id"] == "wall-17315087_74182"
|
assert res[0]["id"] == "wall-17315087_74182"
|
||||||
assert res[0]["text"] == ""
|
assert res[0]["text"] == ""
|
||||||
assert str(res[0]["datetime"]) == str(datetime.datetime(2022, 3, 24, 11, 1, 9))
|
assert str(res[0]["datetime"]) == str(datetime.datetime(2022, 3, 24, 11, 1, 9))
|
||||||
assert len(res[0]["payload"]) == 15
|
assert len(res[0]["payload"]) == 17
|
||||||
assert len(res[0]["attachments"].keys()) == 3
|
assert len(res[0]["attachments"].keys()) == 3
|
||||||
for k in ["photo", "link", "video"]:
|
for k in ["photo", "link", "video"]:
|
||||||
assert k in list(res[0]["attachments"].keys())
|
assert k in list(res[0]["attachments"].keys())
|
||||||
|
|||||||
@@ -40,7 +40,12 @@ class VkScraper:
|
|||||||
VIDEO_PATTERN = re.compile(r"(video.{0,1}\d+_\d+)")
|
VIDEO_PATTERN = re.compile(r"(video.{0,1}\d+_\d+)")
|
||||||
|
|
||||||
def __init__(
|
def __init__(
|
||||||
self, username: str, password: str, token: str = None, captcha_handler=captcha_handler
|
self,
|
||||||
|
username: str,
|
||||||
|
password: str,
|
||||||
|
token: str = None,
|
||||||
|
session_file="vk_config.v2.json",
|
||||||
|
captcha_handler=captcha_handler,
|
||||||
) -> None:
|
) -> None:
|
||||||
"""Initializes the scraper.
|
"""Initializes the scraper.
|
||||||
|
|
||||||
@@ -55,9 +60,17 @@ class VkScraper:
|
|||||||
Matching password on vk.com
|
Matching password on vk.com
|
||||||
token : str
|
token : str
|
||||||
Access token received after authenticating, can be found in the vl_config.v2.json file
|
Access token received after authenticating, can be found in the vl_config.v2.json file
|
||||||
|
session_file : str
|
||||||
|
File name where the VK session is saved so future logins are easier, this will not be created if token is passed
|
||||||
|
captcha_handler : func
|
||||||
|
Function that can receive a vk_api captcha instance and help the user solve it, default is a complete CLI handler
|
||||||
"""
|
"""
|
||||||
self.session = vk_api.VkApi(
|
self.session = vk_api.VkApi(
|
||||||
username, password, token=token, captcha_handler=captcha_handler
|
username,
|
||||||
|
password,
|
||||||
|
token=token,
|
||||||
|
config_filename=session_file,
|
||||||
|
captcha_handler=captcha_handler,
|
||||||
)
|
)
|
||||||
if token is None or len(token) == 0:
|
if token is None or len(token) == 0:
|
||||||
self.session.auth(token_only=True)
|
self.session.auth(token_only=True)
|
||||||
@@ -339,11 +352,9 @@ class VkScraper:
|
|||||||
info = ydl.extract_info(url, download=True)
|
info = ydl.extract_info(url, download=True)
|
||||||
filename = ydl.prepare_filename(info)
|
filename = ydl.prepare_filename(info)
|
||||||
if "unknown_video" in filename:
|
if "unknown_video" in filename:
|
||||||
print(f"before {filename=}")
|
|
||||||
filename = shutil.copy(
|
filename = shutil.copy(
|
||||||
filename, filename.replace("unknown_video", "mkv")
|
filename, filename.replace("unknown_video", "mkv")
|
||||||
)
|
)
|
||||||
print(f"after {filename=}")
|
|
||||||
os.remove(filename)
|
os.remove(filename)
|
||||||
downloaded.append(filename)
|
downloaded.append(filename)
|
||||||
return downloaded
|
return downloaded
|
||||||
|
|||||||
@@ -15,7 +15,7 @@ class DateTimeEncoder(json.JSONEncoder):
|
|||||||
|
|
||||||
def captcha_handler(captcha):
|
def captcha_handler(captcha):
|
||||||
key = input(
|
key = input(
|
||||||
f"CAPTCHA DETECTED, please solve it and input the solution. {captcha.sid=} {captcha.get_url()=}:"
|
f"CAPTCHA DETECTED, please solve it and input the solution. url={captcha.get_url()}:"
|
||||||
).strip()
|
).strip()
|
||||||
return captcha.try_again(key)
|
return captcha.try_again(key)
|
||||||
|
|
||||||
|
|||||||
@@ -2,7 +2,7 @@ _MAJOR = "0"
|
|||||||
_MINOR = "3"
|
_MINOR = "3"
|
||||||
# On main and in a nightly release the patch should be one ahead of the last
|
# On main and in a nightly release the patch should be one ahead of the last
|
||||||
# released build.
|
# released build.
|
||||||
_PATCH = "3"
|
_PATCH = "13"
|
||||||
# This is mainly for nightly builds which have the suffix ".dev$DATE". See
|
# This is mainly for nightly builds which have the suffix ".dev$DATE". See
|
||||||
# https://semver.org/#is-v123-a-semantic-version for the semantics.
|
# https://semver.org/#is-v123-a-semantic-version for the semantics.
|
||||||
_SUFFIX = ""
|
_SUFFIX = ""
|
||||||
|
|||||||
Reference in New Issue
Block a user