Compare commits

...

48 commits
5.2.0 ... main

Author SHA1 Message Date
Hirad 88bb8ea789 Update README.md
Some checks failed
Release drafter / update_release_draft (push) Has been cancelled
Deploy / Build & verify package (push) Has been cancelled
Deploy / Publish in-dev package to test.pypi.org (push) Has been cancelled
Deploy / Publish released package to pypi.org (push) Has been cancelled
Lint / lint (push) Has been cancelled
Test / test (ubuntu-latest, 3.10) (push) Has been cancelled
Test / test (ubuntu-latest, 3.11) (push) Has been cancelled
Test / test (ubuntu-latest, 3.12) (push) Has been cancelled
Test / test (ubuntu-latest, 3.13) (push) Has been cancelled
Test / test (ubuntu-latest, 3.8) (push) Has been cancelled
Test / test (ubuntu-latest, 3.9) (push) Has been cancelled
Test / test (ubuntu-latest, pypy3.10) (push) Has been cancelled
Test / Test successful (push) Has been cancelled
2024-07-07 09:16:54 +03:30
Hirad 6ae051157f Update README.md
Some checks are pending
Deploy / Build & verify package (push) Waiting to run
Deploy / Publish in-dev package to test.pypi.org (push) Blocked by required conditions
Deploy / Publish released package to pypi.org (push) Blocked by required conditions
Lint / lint (push) Waiting to run
Release drafter / update_release_draft (push) Waiting to run
Test / test (ubuntu-latest, 3.10) (push) Waiting to run
Test / test (ubuntu-latest, 3.11) (push) Waiting to run
Test / test (ubuntu-latest, 3.12) (push) Waiting to run
Test / test (ubuntu-latest, 3.13) (push) Waiting to run
Test / test (ubuntu-latest, 3.8) (push) Waiting to run
Test / test (ubuntu-latest, 3.9) (push) Waiting to run
Test / test (ubuntu-latest, pypy3.10) (push) Waiting to run
Test / Test successful (push) Blocked by required conditions
2024-07-07 09:16:25 +03:30
Hirad c260d7b83f change libre.fm to music.lonestar.it
Some checks are pending
Deploy / Build & verify package (push) Waiting to run
Deploy / Publish in-dev package to test.pypi.org (push) Blocked by required conditions
Deploy / Publish released package to pypi.org (push) Blocked by required conditions
Lint / lint (push) Waiting to run
Release drafter / update_release_draft (push) Waiting to run
Test / test (ubuntu-latest, 3.10) (push) Waiting to run
Test / test (ubuntu-latest, 3.11) (push) Waiting to run
Test / test (ubuntu-latest, 3.12) (push) Waiting to run
Test / test (ubuntu-latest, 3.13) (push) Waiting to run
Test / test (ubuntu-latest, 3.8) (push) Waiting to run
Test / test (ubuntu-latest, 3.9) (push) Waiting to run
Test / test (ubuntu-latest, pypy3.10) (push) Waiting to run
Test / Test successful (push) Blocked by required conditions
2024-07-07 09:03:18 +03:30
Hugo van Kemenade 25904371de
[pre-commit.ci] pre-commit autoupdate (#458) 2024-07-02 05:34:35 -06:00
Hugo van Kemenade 184d0328a9 Configure pyproject-fmt to add Python 3.13 classifier 2024-07-02 14:29:20 +03:00
pre-commit-ci[bot] aceaa69c9a
[pre-commit.ci] pre-commit autoupdate
updates:
- [github.com/astral-sh/ruff-pre-commit: v0.4.4 → v0.5.0](https://github.com/astral-sh/ruff-pre-commit/compare/v0.4.4...v0.5.0)
- [github.com/asottile/blacken-docs: 1.16.0 → 1.18.0](https://github.com/asottile/blacken-docs/compare/1.16.0...1.18.0)
- [github.com/python-jsonschema/check-jsonschema: 0.28.4 → 0.28.6](https://github.com/python-jsonschema/check-jsonschema/compare/0.28.4...0.28.6)
- [github.com/rhysd/actionlint: v1.7.0 → v1.7.1](https://github.com/rhysd/actionlint/compare/v1.7.0...v1.7.1)
- [github.com/tox-dev/pyproject-fmt: 1.7.0 → 2.1.3](https://github.com/tox-dev/pyproject-fmt/compare/1.7.0...2.1.3)
2024-07-01 17:24:06 +00:00
Hugo van Kemenade de737fb4ee
Update config (#457) 2024-05-22 22:50:30 +03:00
Hugo van Kemenade 6f97f93dcc Update config 2024-05-22 22:46:47 +03:00
Hugo van Kemenade a23e1c5181
Fix expected result in test and refactor (#456) 2024-05-22 22:38:49 +03:00
Hugo van Kemenade 35b264bae4 Refactor 2024-05-22 22:29:05 +03:00
Hugo van Kemenade 353e32bd6b Fix expected result in test 2024-05-22 22:26:42 +03:00
Hugo van Kemenade 0fa96932a4
Update example_test_pylast.yaml Link in README.md (#455) 2024-05-22 22:08:13 +03:00
pre-commit-ci[bot] 90c3614d6a [pre-commit.ci] auto fixes from pre-commit.com hooks
for more information, see https://pre-commit.ci
2024-05-22 05:10:35 +00:00
Christian McKinnon fa68aa4ae8 Update example_test_pylast.yaml Link in README.md 2024-05-22 11:48:48 +07:00
Hugo van Kemenade 8a26cf88d8
[pre-commit.ci] pre-commit autoupdate (#453) 2024-04-01 20:41:01 +03:00
pre-commit-ci[bot] 82cb504871
[pre-commit.ci] pre-commit autoupdate
updates:
- [github.com/astral-sh/ruff-pre-commit: v0.2.0 → v0.3.4](https://github.com/astral-sh/ruff-pre-commit/compare/v0.2.0...v0.3.4)
- [github.com/psf/black-pre-commit-mirror: 24.1.1 → 24.3.0](https://github.com/psf/black-pre-commit-mirror/compare/24.1.1...24.3.0)
2024-04-01 17:16:48 +00:00
Hugo van Kemenade 9d4283a924
Update github-actions (#450) 2024-03-01 08:06:49 +02:00
Hugo van Kemenade e4d2ebc4a0
Update test.yml 2024-03-01 08:02:36 +02:00
renovate[bot] d5f1c3d3ac
Update github-actions 2024-03-01 01:13:50 +00:00
Hugo van Kemenade e737ae2f34
Add support for Python 3.13 (#448) 2024-02-05 21:35:25 +02:00
Hugo van Kemenade f4547a5821 Add Prettier to pre-commit 2024-02-05 21:27:22 +02:00
Hugo van Kemenade a14a50a333 Scrutinizer was removed in 2019 2024-02-05 21:27:22 +02:00
Hugo van Kemenade 77d1b0009c Add support for Python 3.13 2024-02-05 21:27:22 +02:00
Hugo van Kemenade d505d57fc4
Replace Flake8 with Ruff (#447) 2024-02-05 20:53:51 +02:00
Hugo van Kemenade 3890cb4c04 Add {envpython} and --cov-report html, multiline for clarity 2024-02-04 22:09:24 +02:00
Hugo van Kemenade 5bccda1102 Update config 2024-02-04 22:06:20 +02:00
Hugo van Kemenade 36d89a69e8 Replace Flake8 with Ruff 2024-02-04 22:06:20 +02:00
Hugo van Kemenade a28dea1158
Update github-actions (#446) 2024-02-04 21:10:29 +02:00
Hugo van Kemenade 6c888343c8
Pin codecov/codecov-action to v3.1.5 2024-02-04 12:00:33 -07:00
renovate[bot] ffebde28e2
Update github-actions 2024-02-04 18:59:28 +00:00
Hugo van Kemenade befb5aeceb
Double read timeout to fix 'The read operation timed out' (#445) 2024-02-04 20:58:44 +02:00
Hugo van Kemenade 370ff77f21 Double read timeout to fix 'The read operation timed out'
5 seconds is the default
2024-01-22 09:23:41 +02:00
Hugo van Kemenade cdfc23b5e4
Update github-actions (#443)
Co-authored-by: renovate[bot] <29139614+renovate[bot]@users.noreply.github.com>
2024-01-15 19:24:32 +02:00
pre-commit-ci[bot] d5fe263c23
[pre-commit.ci] pre-commit autoupdate (#444)
Co-authored-by: pre-commit-ci[bot] <66853113+pre-commit-ci[bot]@users.noreply.github.com>
Co-authored-by: Hugo van Kemenade <hugovk@users.noreply.github.com>
2024-01-01 10:31:29 -07:00
renovate[bot] e90d717b66
Update github-actions 2024-01-01 01:18:15 +00:00
Eugene Simonov b4c8dc7282
Add type annotations to methods that take timestamp parameter (#442)
Co-authored-by: Eugene Simonov <eugene.simonov@evergen.com.au>
Co-authored-by: pre-commit-ci[bot] <66853113+pre-commit-ci[bot]@users.noreply.github.com>
Co-authored-by: Hugo van Kemenade <hugovk@users.noreply.github.com>
2023-12-29 12:00:10 -07:00
Hugo van Kemenade 6f30559a3a
Fix incorrect docstrings (#439)
Changes proposed in this pull request:
- fix docstrings with inaccurate descriptions

While going through the pyLast documentation, I noticed that a few
functions had inaccurate descriptions. The proper use of these functions
should be intuitive from their names, but I thought it still might be
useful to fix their docstrings. Let me know if there are any issues with
these changes. Thanks!
2023-10-27 07:00:57 +03:00
Mia Bilka a91bac007d Fixed incorrect docstrings 2023-10-26 18:02:01 -07:00
pre-commit-ci[bot] c0f9f4222a
[pre-commit.ci] pre-commit autoupdate (#437)
Co-authored-by: pre-commit-ci[bot] <66853113+pre-commit-ci[bot]@users.noreply.github.com>
2023-10-02 14:16:48 -06:00
renovate[bot] 47872dbb32
Update actions/checkout action to v4 (#436)
[![Mend
Renovate](https://app.renovatebot.com/images/banner.svg)](https://renovatebot.com)

This PR contains the following updates:

| Package | Type | Update | Change |
|---|---|---|---|
| [actions/checkout](https://togithub.com/actions/checkout) | action |
major | `v3` -> `v4` |

---

### Release Notes

<details>
<summary>actions/checkout (actions/checkout)</summary>

###
[`v4`](https://togithub.com/actions/checkout/blob/HEAD/CHANGELOG.md#v400)

[Compare Source](https://togithub.com/actions/checkout/compare/v3...v4)

- [Support fetching without the --progress
option](https://togithub.com/actions/checkout/pull/1067)
-   [Update to node20](https://togithub.com/actions/checkout/pull/1436)

</details>

---

### Configuration

📅 **Schedule**: Branch creation - "on the first day of the month" (UTC),
Automerge - At any time (no schedule defined).

🚦 **Automerge**: Disabled by config. Please merge this manually once you
are satisfied.

♻ **Rebasing**: Whenever PR becomes conflicted, or you tick the
rebase/retry checkbox.

🔕 **Ignore**: Close this PR and you won't be reminded about this update
again.

---

- [ ] <!-- rebase-check -->If you want to rebase/retry this PR, check
this box

---

This PR has been generated by [Mend
Renovate](https://www.mend.io/free-developer-tools/renovate/). View
repository job log
[here](https://developer.mend.io/github/pylast/pylast).

<!--renovate-debug:eyJjcmVhdGVkSW5WZXIiOiIzNy4wLjMiLCJ1cGRhdGVkSW5WZXIiOiIzNy4wLjMiLCJ0YXJnZXRCcmFuY2giOiJtYWluIn0=-->

Co-authored-by: renovate[bot] <29139614+renovate[bot]@users.noreply.github.com>
2023-10-01 08:37:46 +03:00
pre-commit-ci[bot] 74392c4d71
[pre-commit.ci] pre-commit autoupdate (#434)
Co-authored-by: pre-commit-ci[bot] <66853113+pre-commit-ci[bot]@users.noreply.github.com>
Co-authored-by: Hugo van Kemenade <hugovk@users.noreply.github.com>
2023-08-22 14:22:24 +03:00
Hugo van Kemenade 97eab1719f
Remove default 'cache-dependency-path: pyproject.toml' (#435) 2023-07-24 17:36:32 +03:00
Hugo van Kemenade e4b7af41f9 Remove default 'cache-dependency-path: pyproject.toml'
Committed via https://github.com/asottile/all-repos
2023-07-24 17:33:42 +03:00
Hugo van Kemenade 7f1c90cfea
CI: Replace pypy3.9 with pypy3.10 (#433) 2023-06-18 14:45:26 +03:00
Hugo van Kemenade 68c0197028 CI: Replace pypy3.9 with pypy3.10
Committed via https://github.com/asottile/all-repos
2023-06-18 14:42:29 +03:00
Hugo van Kemenade 9e62e37b1e
Update mheap/github-action-required-labels action to v5 (#432) 2023-06-08 21:24:55 +03:00
Hugo van Kemenade c26c5f86aa Update on the first day of the month 2023-06-08 21:22:05 +03:00
renovate[bot] f7a73aa62f
Update mheap/github-action-required-labels action to v5 2023-06-08 17:29:12 +00:00
29 changed files with 296 additions and 223 deletions

View file

@ -1,2 +0,0 @@
[flake8]
max-line-length = 88

13
.github/renovate.json vendored
View file

@ -1,16 +1,13 @@
{ {
"extends": [ "$schema": "https://docs.renovatebot.com/renovate-schema.json",
"config:base" "extends": ["config:base"],
], "labels": ["changelog: skip", "dependencies"],
"labels": [
"changelog: skip",
"dependencies"
],
"packageRules": [ "packageRules": [
{ {
"groupName": "github-actions", "groupName": "github-actions",
"matchManagers": ["github-actions"], "matchManagers": ["github-actions"],
"separateMajorMinor": "false" "separateMajorMinor": "false"
} }
] ],
"schedule": ["on the first day of the month"]
} }

View file

@ -21,26 +21,28 @@ jobs:
runs-on: ubuntu-latest runs-on: ubuntu-latest
steps: steps:
- uses: actions/checkout@v3 - uses: actions/checkout@v4
with: with:
fetch-depth: 0 fetch-depth: 0
- uses: hynek/build-and-inspect-python-package@v1 - uses: hynek/build-and-inspect-python-package@v2
# Upload to Test PyPI on every commit on main. # Upload to Test PyPI on every commit on main.
release-test-pypi: release-test-pypi:
name: Publish in-dev package to test.pypi.org name: Publish in-dev package to test.pypi.org
if: github.event_name == 'push' && github.ref == 'refs/heads/main' if: |
github.repository_owner == 'pylast'
&& github.event_name == 'push'
&& github.ref == 'refs/heads/main'
runs-on: ubuntu-latest runs-on: ubuntu-latest
needs: build-package needs: build-package
permissions: permissions:
# IMPORTANT: this permission is mandatory for trusted publishing
id-token: write id-token: write
steps: steps:
- name: Download packages built by build-and-inspect-python-package - name: Download packages built by build-and-inspect-python-package
uses: actions/download-artifact@v3 uses: actions/download-artifact@v4
with: with:
name: Packages name: Packages
path: dist path: dist
@ -53,17 +55,18 @@ jobs:
# Upload to real PyPI on GitHub Releases. # Upload to real PyPI on GitHub Releases.
release-pypi: release-pypi:
name: Publish released package to pypi.org name: Publish released package to pypi.org
if: github.event.action == 'published' if: |
github.repository_owner == 'pylast'
&& github.event.action == 'published'
runs-on: ubuntu-latest runs-on: ubuntu-latest
needs: build-package needs: build-package
permissions: permissions:
# IMPORTANT: this permission is mandatory for trusted publishing
id-token: write id-token: write
steps: steps:
- name: Download packages built by build-and-inspect-python-package - name: Download packages built by build-and-inspect-python-package
uses: actions/download-artifact@v3 uses: actions/download-artifact@v4
with: with:
name: Packages name: Packages
path: dist path: dist

View file

@ -1,5 +1,8 @@
name: Sync labels name: Sync labels
permissions:
pull-requests: write
on: on:
push: push:
branches: branches:
@ -12,7 +15,7 @@ jobs:
sync: sync:
runs-on: ubuntu-latest runs-on: ubuntu-latest
steps: steps:
- uses: actions/checkout@v3 - uses: actions/checkout@v4
- uses: micnncim/action-label-syncer@v1 - uses: micnncim/action-label-syncer@v1
with: with:
prune: false prune: false

View file

@ -2,6 +2,10 @@ name: Lint
on: [push, pull_request, workflow_dispatch] on: [push, pull_request, workflow_dispatch]
env:
FORCE_COLOR: 1
PIP_DISABLE_PIP_VERSION_CHECK: 1
permissions: permissions:
contents: read contents: read
@ -10,8 +14,9 @@ jobs:
runs-on: ubuntu-latest runs-on: ubuntu-latest
steps: steps:
- uses: actions/checkout@v3 - uses: actions/checkout@v4
- uses: actions/setup-python@v4 - uses: actions/setup-python@v5
with: with:
python-version: "3.x" python-version: "3.x"
- uses: pre-commit/action@v3.0.0 cache: pip
- uses: pre-commit/action@v3.0.1

View file

@ -29,6 +29,6 @@ jobs:
runs-on: ubuntu-latest runs-on: ubuntu-latest
steps: steps:
# Drafts your next release notes as pull requests are merged into "main" # Drafts your next release notes as pull requests are merged into "main"
- uses: release-drafter/release-drafter@v5 - uses: release-drafter/release-drafter@v6
env: env:
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}

View file

@ -10,9 +10,10 @@ jobs:
permissions: permissions:
issues: write issues: write
pull-requests: write
steps: steps:
- uses: mheap/github-action-required-labels@v4 - uses: mheap/github-action-required-labels@v5
with: with:
mode: minimum mode: minimum
count: 1 count: 1

View file

@ -11,19 +11,18 @@ jobs:
strategy: strategy:
fail-fast: false fail-fast: false
matrix: matrix:
python-version: ["pypy3.9", "3.8", "3.9", "3.10", "3.11", "3.12"] python-version: ["pypy3.10", "3.8", "3.9", "3.10", "3.11", "3.12", "3.13"]
os: [ubuntu-latest] os: [ubuntu-latest]
steps: steps:
- uses: actions/checkout@v3 - uses: actions/checkout@v4
- name: Set up Python ${{ matrix.python-version }} - name: Set up Python ${{ matrix.python-version }}
uses: actions/setup-python@v4 uses: actions/setup-python@v5
with: with:
python-version: ${{ matrix.python-version }} python-version: ${{ matrix.python-version }}
allow-prereleases: true allow-prereleases: true
cache: pip cache: pip
cache-dependency-path: pyproject.toml
- name: Install dependencies - name: Install dependencies
run: | run: |
@ -41,7 +40,7 @@ jobs:
PYLAST_USERNAME: ${{ secrets.PYLAST_USERNAME }} PYLAST_USERNAME: ${{ secrets.PYLAST_USERNAME }}
- name: Upload coverage - name: Upload coverage
uses: codecov/codecov-action@v3 uses: codecov/codecov-action@v3.1.5
with: with:
flags: ${{ matrix.os }} flags: ${{ matrix.os }}
name: ${{ matrix.os }} Python ${{ matrix.python-version }} name: ${{ matrix.os }} Python ${{ matrix.python-version }}

View file

@ -1,64 +1,74 @@
repos: repos:
- repo: https://github.com/asottile/pyupgrade - repo: https://github.com/astral-sh/ruff-pre-commit
rev: v3.4.0 rev: v0.5.0
hooks: hooks:
- id: pyupgrade - id: ruff
args: [--py38-plus] args: [--exit-non-zero-on-fix]
- repo: https://github.com/psf/black - repo: https://github.com/psf/black-pre-commit-mirror
rev: 23.3.0 rev: 24.4.2
hooks: hooks:
- id: black - id: black
- repo: https://github.com/asottile/blacken-docs - repo: https://github.com/asottile/blacken-docs
rev: 1.13.0 rev: 1.18.0
hooks: hooks:
- id: blacken-docs - id: blacken-docs
args: [--target-version=py38] args: [--target-version=py38]
additional_dependencies: [black==23.3.0] additional_dependencies: [black]
- repo: https://github.com/PyCQA/isort
rev: 5.12.0
hooks:
- id: isort
- repo: https://github.com/PyCQA/flake8
rev: 6.0.0
hooks:
- id: flake8
additional_dependencies: [flake8-2020, flake8-implicit-str-concat]
- repo: https://github.com/pre-commit/pygrep-hooks
rev: v1.10.0
hooks:
- id: python-check-blanket-noqa
- id: python-no-log-warn
- repo: https://github.com/pre-commit/pre-commit-hooks - repo: https://github.com/pre-commit/pre-commit-hooks
rev: v4.4.0 rev: v4.6.0
hooks: hooks:
- id: check-added-large-files
- id: check-case-conflict - id: check-case-conflict
- id: check-merge-conflict - id: check-merge-conflict
- id: check-json - id: check-json
- id: check-toml - id: check-toml
- id: check-yaml - id: check-yaml
- id: debug-statements
- id: end-of-file-fixer - id: end-of-file-fixer
- id: requirements-txt-fixer - id: forbid-submodules
- id: trailing-whitespace
exclude: .github/(ISSUE_TEMPLATE|PULL_REQUEST_TEMPLATE).md
- repo: https://github.com/python-jsonschema/check-jsonschema
rev: 0.28.6
hooks:
- id: check-github-workflows
- id: check-renovate
- repo: https://github.com/rhysd/actionlint
rev: v1.7.1
hooks:
- id: actionlint
- repo: https://github.com/tox-dev/pyproject-fmt - repo: https://github.com/tox-dev/pyproject-fmt
rev: 0.10.0 rev: 2.1.3
hooks: hooks:
- id: pyproject-fmt - id: pyproject-fmt
- repo: https://github.com/abravalheri/validate-pyproject - repo: https://github.com/abravalheri/validate-pyproject
rev: v0.13 rev: v0.18
hooks: hooks:
- id: validate-pyproject - id: validate-pyproject
- repo: https://github.com/tox-dev/tox-ini-fmt - repo: https://github.com/tox-dev/tox-ini-fmt
rev: 1.3.0 rev: 1.3.1
hooks: hooks:
- id: tox-ini-fmt - id: tox-ini-fmt
- repo: https://github.com/pre-commit/mirrors-prettier
rev: v4.0.0-alpha.8
hooks:
- id: prettier
args: [--prose-wrap=always, --print-width=88]
exclude: .github/(ISSUE_TEMPLATE|PULL_REQUEST_TEMPLATE).md
- repo: meta
hooks:
- id: check-hooks-apply
- id: check-useless-excludes
ci: ci:
autoupdate_schedule: quarterly autoupdate_schedule: quarterly

View file

@ -1,9 +0,0 @@
checks:
python:
code_rating: true
duplicate_code: true
filter:
excluded_paths:
- '*/test/*'
tools:
external_code_coverage: true

View file

@ -12,117 +12,125 @@ See GitHub Releases:
## Changed ## Changed
* Fix unsafe creation of temp file for caching, and improve exception raising (#356) @kvanzuijlen - Fix unsafe creation of temp file for caching, and improve exception raising (#356)
* [pre-commit.ci] pre-commit autoupdate (#362) @pre-commit-ci @kvanzuijlen
- [pre-commit.ci] pre-commit autoupdate (#362) @pre-commit-ci
## [4.1.0] - 2021-01-04 ## [4.1.0] - 2021-01-04
## Added ## Added
* Add support for streaming (#336) @kvanzuijlen - Add support for streaming (#336) @kvanzuijlen
* Add Python 3.9 final to Travis CI (#350) @sheetalsingala - Add Python 3.9 final to Travis CI (#350) @sheetalsingala
## Changed ## Changed
* Update copyright year (#360) @hugovk - Update copyright year (#360) @hugovk
* Replace Travis CI with GitHub Actions (#352) @hugovk - Replace Travis CI with GitHub Actions (#352) @hugovk
* [pre-commit.ci] pre-commit autoupdate (#359) @pre-commit-ci - [pre-commit.ci] pre-commit autoupdate (#359) @pre-commit-ci
## Fixed ## Fixed
* Set limit to 50 by default, not 1 (#355) @hugovk - Set limit to 50 by default, not 1 (#355) @hugovk
## [4.0.0] - 2020-10-07 ## [4.0.0] - 2020-10-07
## Added ## Added
* Add support for Python 3.9 (#347) @hugovk - Add support for Python 3.9 (#347) @hugovk
## Removed ## Removed
* Remove deprecated `Artist.get_cover_image`, `User.get_artist_tracks` and `STATUS_TOKEN_ERROR` (#348) @hugovk - Remove deprecated `Artist.get_cover_image`, `User.get_artist_tracks` and
* Drop support for EOL Python 3.5 (#346) @hugovk `STATUS_TOKEN_ERROR` (#348) @hugovk
- Drop support for EOL Python 3.5 (#346) @hugovk
## [3.3.0] - 2020-06-25 ## [3.3.0] - 2020-06-25
### Added ### Added
* `User.get_now_playing`: Add album and cover image to info (#330) @hugovk - `User.get_now_playing`: Add album and cover image to info (#330) @hugovk
### Changed ### Changed
* Improve handling of error responses from the API (#327) @spiritualized - Improve handling of error responses from the API (#327) @spiritualized
### Deprecated ### Deprecated
* Deprecate `Artist.get_cover_image`, they're no longer available from Last.fm (#332) @hugovk - Deprecate `Artist.get_cover_image`, they're no longer available from Last.fm (#332)
@hugovk
### Fixed ### Fixed
* Fix `artist.get_bio_content()` to return `None` if bio is empty (#326) @hugovk - Fix `artist.get_bio_content()` to return `None` if bio is empty (#326) @hugovk
## [3.2.1] - 2020-03-05 ## [3.2.1] - 2020-03-05
### Fixed ### Fixed
* Only Python 3 is supported: don't create universal wheel (#318) @hugovk - Only Python 3 is supported: don't create universal wheel (#318) @hugovk
* Fix regression calling `get_recent_tracks` with `limit=None` (#320) @hugovk - Fix regression calling `get_recent_tracks` with `limit=None` (#320) @hugovk
* Fix `DeprecationWarning`: Please use `assertRegex` instead (#323) @hugovk - Fix `DeprecationWarning`: Please use `assertRegex` instead (#323) @hugovk
## [3.2.0] - 2020-01-03 ## [3.2.0] - 2020-01-03
### Added ### Added
* Support for Python 3.8 - Support for Python 3.8
* Store album art URLs when you call `GetTopAlbums` ([#307]) - Store album art URLs when you call `GetTopAlbums` ([#307])
* Retry paging through results on exception ([#297]) - Retry paging through results on exception ([#297])
* More error status codes from https://last.fm/api/errorcodes ([#297]) - More error status codes from https://last.fm/api/errorcodes ([#297])
### Changed ### Changed
* Respect `get_recent_tracks`' limit when there's a now playing track ([#310]) - Respect `get_recent_tracks`' limit when there's a now playing track ([#310])
* Move installable code to `src/` ([#301]) - Move installable code to `src/` ([#301])
* Update `get_weekly_artist_charts` docstring: only for `User` ([#311]) - Update `get_weekly_artist_charts` docstring: only for `User` ([#311])
* Remove Python 2 warnings, `python_requires` should be enough ([#312]) - Remove Python 2 warnings, `python_requires` should be enough ([#312])
* Use setuptools_scm to simplify versioning during release ([#316]) - Use setuptools_scm to simplify versioning during release ([#316])
* Various lint and test updates - Various lint and test updates
### Deprecated ### Deprecated
* Last.fm's `user.getArtistTracks` has now been deprecated by Last.fm and is no longer - Last.fm's `user.getArtistTracks` has now been deprecated by Last.fm and is no longer
available. Last.fm returns a "Deprecated - This type of request is no longer available. Last.fm returns a "Deprecated - This type of request is no longer
supported" error when calling it. A future version of pylast will remove its supported" error when calling it. A future version of pylast will remove its
`User.get_artist_tracks` altogether. ([#305]) `User.get_artist_tracks` altogether. ([#305])
* `STATUS_TOKEN_ERROR` is deprecated and will be removed in a future version. - `STATUS_TOKEN_ERROR` is deprecated and will be removed in a future version. Use
Use `STATUS_OPERATION_FAILED` instead. `STATUS_OPERATION_FAILED` instead.
## [3.1.0] - 2019-03-07 ## [3.1.0] - 2019-03-07
### Added ### Added
* Extract username from session via new - Extract username from session via new
`SessionKeyGenerator.get_web_auth_session_key_username` ([#290]) `SessionKeyGenerator.get_web_auth_session_key_username` ([#290])
* `User.get_track_scrobbles` ([#298]) - `User.get_track_scrobbles` ([#298])
### Deprecated ### Deprecated
* `User.get_artist_tracks`. Use `User.get_track_scrobbles` as a partial replacement. - `User.get_artist_tracks`. Use `User.get_track_scrobbles` as a partial replacement.
([#298]) ([#298])
## [3.0.0] - 2019-01-01 ## [3.0.0] - 2019-01-01
### Added ### Added
* This changelog file ([#273])
- This changelog file ([#273])
### Removed ### Removed
* Support for Python 2.7 ([#265]) - Support for Python 2.7 ([#265])
* Constants `COVER_SMALL`, `COVER_MEDIUM`, `COVER_LARGE`, `COVER_EXTRA_LARGE` - Constants `COVER_SMALL`, `COVER_MEDIUM`, `COVER_LARGE`, `COVER_EXTRA_LARGE` and
and `COVER_MEGA`. Use `SIZE_SMALL` etc. instead. ([#282]) `COVER_MEGA`. Use `SIZE_SMALL` etc. instead. ([#282])
## [2.4.0] - 2018-08-08 ## [2.4.0] - 2018-08-08
### Deprecated ### Deprecated
* Support for Python 2.7 ([#265]) - Support for Python 2.7 ([#265])
[4.2.0]: https://github.com/pylast/pylast/compare/4.1.0...4.2.0 [4.2.0]: https://github.com/pylast/pylast/compare/4.1.0...4.2.0
[4.1.0]: https://github.com/pylast/pylast/compare/4.0.0...4.1.0 [4.1.0]: https://github.com/pylast/pylast/compare/4.0.0...4.1.0

View file

@ -15,50 +15,44 @@ Use the pydoc utility for help on usage or see [tests/](tests/) for examples.
## Installation ## Installation
Install via pip:
```sh
python3 -m pip install pylast
```
Install latest development version: Install latest development version:
```sh ```sh
python3 -m pip install -U git+https://github.com/pylast/pylast python3 -m pip install -U git+https://git.hirad.it/Hirad/pylast
``` ```
Or from requirements.txt: Or from requirements.txt:
```txt ```txt
-e https://github.com/pylast/pylast.git#egg=pylast -e https://git.hirad.it/Hirad/pylast#egg=pylast
``` ```
Note: Note:
* pyLast 5.2+ supports Python 3.8-3.12. - pyLast 5.3+ supports Python 3.8-3.13.
* pyLast 5.1 supports Python 3.7-3.11. - pyLast 5.2+ supports Python 3.8-3.12.
* pyLast 5.0 supports Python 3.7-3.10. - pyLast 5.1 supports Python 3.7-3.11.
* pyLast 4.3 - 4.5 supports Python 3.6-3.10. - pyLast 5.0 supports Python 3.7-3.10.
* pyLast 4.0 - 4.2 supports Python 3.6-3.9. - pyLast 4.3 - 4.5 supports Python 3.6-3.10.
* pyLast 3.2 - 3.3 supports Python 3.5-3.8. - pyLast 4.0 - 4.2 supports Python 3.6-3.9.
* pyLast 3.0 - 3.1 supports Python 3.5-3.7. - pyLast 3.2 - 3.3 supports Python 3.5-3.8.
* pyLast 2.2 - 2.4 supports Python 2.7.10+, 3.4-3.7. - pyLast 3.0 - 3.1 supports Python 3.5-3.7.
* pyLast 2.0 - 2.1 supports Python 2.7.10+, 3.4-3.6. - pyLast 2.2 - 2.4 supports Python 2.7.10+, 3.4-3.7.
* pyLast 1.7 - 1.9 supports Python 2.7, 3.3-3.6. - pyLast 2.0 - 2.1 supports Python 2.7.10+, 3.4-3.6.
* pyLast 1.0 - 1.6 supports Python 2.7, 3.3-3.4. - pyLast 1.7 - 1.9 supports Python 2.7, 3.3-3.6.
* pyLast 0.5 supports Python 2, 3. - pyLast 1.0 - 1.6 supports Python 2.7, 3.3-3.4.
* pyLast < 0.5 supports Python 2. - pyLast 0.5 supports Python 2, 3.
- pyLast < 0.5 supports Python 2.
## Features ## Features
* Simple public interface. - Simple public interface.
* Access to all the data exposed by the Last.fm web services. - Access to all the data exposed by the Last.fm web services.
* Scrobbling support. - Scrobbling support.
* Full object-oriented design. - Full object-oriented design.
* Proxy support. - Proxy support.
* Internal caching support for some web services calls (disabled by default). - Internal caching support for some web services calls (disabled by default).
* Support for other API-compatible networks like Libre.fm. - Support for other API-compatible networks like Libre.fm.
## Getting started ## Getting started
@ -87,8 +81,8 @@ network = pylast.LastFMNetwork(
) )
``` ```
Alternatively, instead of creating `network` with a username and password, Alternatively, instead of creating `network` with a username and password, you can
you can authenticate with a session key: authenticate with a session key:
```python ```python
import pylast import pylast
@ -131,7 +125,6 @@ track.add_tags(("awesome", "favorite"))
# to get more help about anything and see examples of how it works # to get more help about anything and see examples of how it works
``` ```
More examples in More examples in
<a href="https://github.com/hugovk/lastfm-tools">hugovk/lastfm-tools</a> and <a href="https://github.com/hugovk/lastfm-tools">hugovk/lastfm-tools</a> and
[tests/](https://github.com/pylast/pylast/tree/main/tests). [tests/](https://github.com/pylast/pylast/tree/main/tests).
@ -143,8 +136,9 @@ integration and unit tests with Last.fm, and plenty of code examples.
For integration tests you need a test account at Last.fm that will become cluttered with For integration tests you need a test account at Last.fm that will become cluttered with
test data, and an API key and secret. Either copy test data, and an API key and secret. Either copy
[example_test_pylast.yaml](example_test_pylast.yaml) to test_pylast.yaml and fill out [example_test_pylast.yaml](https://github.com/pylast/pylast/blob/main/example_test_pylast.yaml)
the credentials, or set them as environment variables like: to test_pylast.yaml and fill out the credentials, or set them as environment variables
like:
```sh ```sh
export PYLAST_USERNAME=TODO_ENTER_YOURS_HERE export PYLAST_USERNAME=TODO_ENTER_YOURS_HERE

View file

@ -1,8 +1,8 @@
# Release Checklist # Release Checklist
- [ ] Get `main` to the appropriate code release state. - [ ] Get `main` to the appropriate code release state.
[GitHub Actions](https://github.com/pylast/pylast/actions) should be running cleanly for [GitHub Actions](https://github.com/pylast/pylast/actions) should be running
all merges to `main`. cleanly for all merges to `main`.
[![Test](https://github.com/pylast/pylast/workflows/Test/badge.svg)](https://github.com/pylast/pylast/actions) [![Test](https://github.com/pylast/pylast/workflows/Test/badge.svg)](https://github.com/pylast/pylast/actions)
- [ ] Edit release draft, adjust text if needed: - [ ] Edit release draft, adjust text if needed:
@ -12,7 +12,8 @@
- [ ] Publish release - [ ] Publish release
- [ ] Check the tagged [GitHub Actions build](https://github.com/pylast/pylast/actions/workflows/deploy.yml) - [ ] Check the tagged
[GitHub Actions build](https://github.com/pylast/pylast/actions/workflows/deploy.yml)
has deployed to [PyPI](https://pypi.org/project/pylast/#history) has deployed to [PyPI](https://pypi.org/project/pylast/#history)
- [ ] Check installation: - [ ] Check installation:

View file

@ -16,19 +16,23 @@ keywords = [
"scrobbling", "scrobbling",
] ]
license = { text = "Apache-2.0" } license = { text = "Apache-2.0" }
maintainers = [{name = "Hugo van Kemenade"}] maintainers = [
authors = [{name = "Amr Hassan <amr.hassan@gmail.com> and Contributors", email = "amr.hassan@gmail.com"}] { name = "Hugo van Kemenade" },
]
authors = [
{ name = "Amr Hassan <amr.hassan@gmail.com> and Contributors", email = "amr.hassan@gmail.com" },
]
requires-python = ">=3.8" requires-python = ">=3.8"
classifiers = [ classifiers = [
"Development Status :: 5 - Production/Stable", "Development Status :: 5 - Production/Stable",
"License :: OSI Approved :: Apache Software License", "License :: OSI Approved :: Apache Software License",
"Programming Language :: Python :: 3",
"Programming Language :: Python :: 3 :: Only", "Programming Language :: Python :: 3 :: Only",
"Programming Language :: Python :: 3.8", "Programming Language :: Python :: 3.8",
"Programming Language :: Python :: 3.9", "Programming Language :: Python :: 3.9",
"Programming Language :: Python :: 3.10", "Programming Language :: Python :: 3.10",
"Programming Language :: Python :: 3.11", "Programming Language :: Python :: 3.11",
"Programming Language :: Python :: 3.12", "Programming Language :: Python :: 3.12",
"Programming Language :: Python :: 3.13",
"Programming Language :: Python :: Implementation :: CPython", "Programming Language :: Python :: Implementation :: CPython",
"Programming Language :: Python :: Implementation :: PyPy", "Programming Language :: Python :: Implementation :: PyPy",
"Topic :: Internet", "Topic :: Internet",
@ -41,18 +45,16 @@ dynamic = [
dependencies = [ dependencies = [
"httpx", "httpx",
] ]
[project.optional-dependencies] optional-dependencies.tests = [
tests = [
"flaky", "flaky",
"pytest", "pytest",
"pytest-cov", "pytest-cov",
"pytest-random-order", "pytest-random-order",
"pyyaml", "pyyaml",
] ]
[project.urls] urls.Changelog = "https://github.com/pylast/pylast/releases"
Changelog = "https://github.com/pylast/pylast/releases" urls.Homepage = "https://github.com/pylast/pylast"
Homepage = "https://github.com/pylast/pylast" urls.Source = "https://github.com/pylast/pylast"
Source = "https://github.com/pylast/pylast"
[tool.hatch] [tool.hatch]
version.source = "vcs" version.source = "vcs"
@ -60,5 +62,36 @@ version.source = "vcs"
[tool.hatch.version.raw-options] [tool.hatch.version.raw-options]
local_scheme = "no-local-version" local_scheme = "no-local-version"
[tool.isort] [tool.ruff]
profile = "black" fix = true
lint.select = [
"C4", # flake8-comprehensions
"E", # pycodestyle errors
"EM", # flake8-errmsg
"F", # pyflakes errors
"I", # isort
"ISC", # flake8-implicit-str-concat
"LOG", # flake8-logging
"PGH", # pygrep-hooks
"RUF022", # unsorted-dunder-all
"RUF100", # unused noqa (yesqa)
"UP", # pyupgrade
"W", # pycodestyle warnings
"YTT", # flake8-2020
]
lint.extend-ignore = [
"E203", # Whitespace before ':'
"E221", # Multiple spaces before operator
"E226", # Missing whitespace around arithmetic operator
"E241", # Multiple spaces after ','
]
lint.isort.known-first-party = [
"pylast",
]
lint.isort.required-imports = [
"from __future__ import annotations",
]
[tool.pyproject-fmt]
max_supported_python = "3.13"

View file

@ -1,6 +1,6 @@
# #
# pylast - # pylast -
# A Python interface to Last.fm and Libre.fm # A Python interface to Last.fm and music.lonestar.it
# #
# Copyright 2008-2010 Amr Hassan # Copyright 2008-2010 Amr Hassan
# Copyright 2013-2021 hugovk # Copyright 2013-2021 hugovk
@ -529,25 +529,25 @@ class _Network:
def scrobble( def scrobble(
self, self,
artist, artist: str,
title, title: str,
timestamp, timestamp: int,
album=None, album: str | None = None,
album_artist=None, album_artist: str | None = None,
track_number=None, track_number: int | None = None,
duration=None, duration: int | None = None,
stream_id=None, stream_id: str | None = None,
context=None, context: str | None = None,
mbid=None, mbid: str | None = None,
): ):
"""Used to add a track-play to a user's profile. """Used to add a track-play to a user's profile.
Parameters: Parameters:
artist (Required) : The artist name. artist (Required) : The artist name.
title (Required) : The track name. title (Required) : The track name.
timestamp (Required) : The time the track started playing, in UNIX timestamp (Required) : The time the track started playing, in Unix
timestamp format (integer number of seconds since 00:00:00, timestamp format (integer number of seconds since 00:00:00,
January 1st 1970 UTC). This must be in the UTC time zone. January 1st 1970 UTC).
album (Optional) : The album name. album (Optional) : The album name.
album_artist (Optional) : The album artist - if this differs from album_artist (Optional) : The album artist - if this differs from
the track artist. the track artist.
@ -628,7 +628,6 @@ class _Network:
class LastFMNetwork(_Network): class LastFMNetwork(_Network):
"""A Last.fm network object """A Last.fm network object
api_key: a provided API_KEY api_key: a provided API_KEY
@ -706,7 +705,7 @@ class LastFMNetwork(_Network):
class LibreFMNetwork(_Network): class LibreFMNetwork(_Network):
""" """
A preconfigured _Network object for Libre.fm A preconfigured _Network object for music.lonestar.it
api_key: a provided API_KEY api_key: a provided API_KEY
api_secret: a provided API_SECRET api_secret: a provided API_SECRET
@ -728,27 +727,27 @@ class LibreFMNetwork(_Network):
password_hash: str = "", password_hash: str = "",
) -> None: ) -> None:
super().__init__( super().__init__(
name="Libre.fm", name="music.lonestar.it",
homepage="https://libre.fm", homepage="https://music.lonestar.it",
ws_server=("libre.fm", "/2.0/"), ws_server=("music.lonestar.it", "/2.0/"),
api_key=api_key, api_key=api_key,
api_secret=api_secret, api_secret=api_secret,
session_key=session_key, session_key=session_key,
username=username, username=username,
password_hash=password_hash, password_hash=password_hash,
domain_names={ domain_names={
DOMAIN_ENGLISH: "libre.fm", DOMAIN_ENGLISH: "music.lonestar.it",
DOMAIN_GERMAN: "libre.fm", DOMAIN_GERMAN: "music.lonestar.it",
DOMAIN_SPANISH: "libre.fm", DOMAIN_SPANISH: "music.lonestar.it",
DOMAIN_FRENCH: "libre.fm", DOMAIN_FRENCH: "music.lonestar.it",
DOMAIN_ITALIAN: "libre.fm", DOMAIN_ITALIAN: "music.lonestar.it",
DOMAIN_POLISH: "libre.fm", DOMAIN_POLISH: "music.lonestar.it",
DOMAIN_PORTUGUESE: "libre.fm", DOMAIN_PORTUGUESE: "music.lonestar.it",
DOMAIN_SWEDISH: "libre.fm", DOMAIN_SWEDISH: "music.lonestar.it",
DOMAIN_TURKISH: "libre.fm", DOMAIN_TURKISH: "music.lonestar.it",
DOMAIN_RUSSIAN: "libre.fm", DOMAIN_RUSSIAN: "music.lonestar.it",
DOMAIN_JAPANESE: "libre.fm", DOMAIN_JAPANESE: "music.lonestar.it",
DOMAIN_CHINESE: "libre.fm", DOMAIN_CHINESE: "music.lonestar.it",
}, },
urls={ urls={
"album": "artist/%(artist)s/album/%(album)s", "album": "artist/%(artist)s/album/%(album)s",
@ -894,6 +893,7 @@ class _Request:
username = "" if username is None else f"?username={username}" username = "" if username is None else f"?username={username}"
(host_name, host_subdir) = self.network.ws_server (host_name, host_subdir) = self.network.ws_server
timeout = httpx.Timeout(5, read=10)
if self.network.is_proxy_enabled(): if self.network.is_proxy_enabled():
client = httpx.Client( client = httpx.Client(
@ -901,12 +901,14 @@ class _Request:
base_url=f"https://{host_name}", base_url=f"https://{host_name}",
headers=HEADERS, headers=HEADERS,
proxies=self.network.proxy, proxies=self.network.proxy,
timeout=timeout,
) )
else: else:
client = httpx.Client( client = httpx.Client(
verify=SSL_CONTEXT, verify=SSL_CONTEXT,
base_url=f"https://{host_name}", base_url=f"https://{host_name}",
headers=HEADERS, headers=HEADERS,
timeout=timeout,
) )
try: try:
@ -1156,7 +1158,7 @@ class _BaseObject:
def get_wiki_published_date(self): def get_wiki_published_date(self):
""" """
Returns the summary of the wiki. Returns the date on which the wiki was published.
Only for Album/Track. Only for Album/Track.
""" """
return self.get_wiki("published") return self.get_wiki("published")
@ -1170,7 +1172,7 @@ class _BaseObject:
def get_wiki_content(self): def get_wiki_content(self):
""" """
Returns the summary of the wiki. Returns the content of the wiki.
Only for Album/Track. Only for Album/Track.
""" """
return self.get_wiki("content") return self.get_wiki("content")
@ -1240,8 +1242,10 @@ class _Chartable(_BaseObject):
from_date value to the to_date value. from_date value to the to_date value.
chart_kind should be one of "album", "artist" or "track" chart_kind should be one of "album", "artist" or "track"
""" """
import sys
method = ".getWeekly" + chart_kind.title() + "Chart" method = ".getWeekly" + chart_kind.title() + "Chart"
chart_type = eval(chart_kind.title()) # string to type chart_type = getattr(sys.modules[__name__], chart_kind.title())
params = self._get_params() params = self._get_params()
if from_date and to_date: if from_date and to_date:
@ -1353,11 +1357,11 @@ class _Taggable(_BaseObject):
new_tags.append(tag) new_tags.append(tag)
for i in range(0, len(old_tags)): for i in range(0, len(old_tags)):
if not c_old_tags[i] in c_new_tags: if c_old_tags[i] not in c_new_tags:
to_remove.append(old_tags[i]) to_remove.append(old_tags[i])
for i in range(0, len(new_tags)): for i in range(0, len(new_tags)):
if not c_new_tags[i] in c_old_tags: if c_new_tags[i] not in c_old_tags:
to_add.append(new_tags[i]) to_add.append(new_tags[i])
self.remove_tags(to_remove) self.remove_tags(to_remove)
@ -1505,7 +1509,7 @@ class _Opus(_Taggable):
return f"{self.get_artist().get_name()} - {self.get_title()}" return f"{self.get_artist().get_name()} - {self.get_title()}"
def __eq__(self, other): def __eq__(self, other):
if type(self) != type(other): if type(self) is not type(other):
return False return False
a = self.get_title().lower() a = self.get_title().lower()
b = other.get_title().lower() b = other.get_title().lower()
@ -1543,7 +1547,7 @@ class _Opus(_Taggable):
return self.info["image"][size] return self.info["image"][size]
def get_title(self, properly_capitalized: bool = False): def get_title(self, properly_capitalized: bool = False):
"""Returns the artist or track title.""" """Returns the album or track title."""
if properly_capitalized: if properly_capitalized:
self.title = _extract( self.title = _extract(
self._request(self.ws_prefix + ".getInfo", True), "name" self._request(self.ws_prefix + ".getInfo", True), "name"
@ -2295,8 +2299,8 @@ class User(_Chartable):
self, self,
limit: int = 10, limit: int = 10,
cacheable: bool = True, cacheable: bool = True,
time_from=None, time_from: int | None = None,
time_to=None, time_to: int | None = None,
stream: bool = False, stream: bool = False,
now_playing: bool = False, now_playing: bool = False,
): ):
@ -2307,13 +2311,11 @@ class User(_Chartable):
Parameters: Parameters:
limit : If None, it will try to pull all the available data. limit : If None, it will try to pull all the available data.
from (Optional) : Beginning timestamp of a range - only display from (Optional) : Beginning timestamp of a range - only display
scrobbles after this time, in UNIX timestamp format (integer scrobbles after this time, in Unix timestamp format (integer
number of seconds since 00:00:00, January 1st 1970 UTC). This number of seconds since 00:00:00, January 1st 1970 UTC).
must be in the UTC time zone.
to (Optional) : End timestamp of a range - only display scrobbles to (Optional) : End timestamp of a range - only display scrobbles
before this time, in UNIX timestamp format (integer number of before this time, in Unix timestamp format (integer number of
seconds since 00:00:00, January 1st 1970 UTC). This must be in seconds since 00:00:00, January 1st 1970 UTC).
the UTC time zone.
stream: If True, it will yield tracks as soon as a page has been retrieved. stream: If True, it will yield tracks as soon as a page has been retrieved.
This method uses caching. Enable caching only if you're pulling a This method uses caching. Enable caching only if you're pulling a
@ -2382,7 +2384,7 @@ class User(_Chartable):
return _extract(doc, "registered") return _extract(doc, "registered")
def get_unixtime_registered(self): def get_unixtime_registered(self):
"""Returns the user's registration date as a UNIX timestamp.""" """Returns the user's registration date as a Unix timestamp."""
doc = self._request(self.ws_prefix + ".getInfo", True) doc = self._request(self.ws_prefix + ".getInfo", True)
@ -2777,7 +2779,8 @@ def _collect_nodes(
main.getAttribute("totalPages") or main.getAttribute("totalpages") main.getAttribute("totalPages") or main.getAttribute("totalpages")
) )
else: else:
raise PyLastError("No total pages attribute") msg = "No total pages attribute"
raise PyLastError(msg)
for node in main.childNodes: for node in main.childNodes:
if not node.nodeType == xml.dom.Node.TEXT_NODE and ( if not node.nodeType == xml.dom.Node.TEXT_NODE and (

View file

@ -2,6 +2,8 @@
""" """
Integration (not unit) tests for pylast.py Integration (not unit) tests for pylast.py
""" """
from __future__ import annotations
import pylast import pylast
from .test_pylast import TestPyLastWithLastFm from .test_pylast import TestPyLastWithLastFm

View file

@ -2,6 +2,8 @@
""" """
Integration (not unit) tests for pylast.py Integration (not unit) tests for pylast.py
""" """
from __future__ import annotations
import pytest import pytest
import pylast import pylast

View file

@ -2,6 +2,8 @@
""" """
Integration (not unit) tests for pylast.py Integration (not unit) tests for pylast.py
""" """
from __future__ import annotations
import pylast import pylast
from .test_pylast import TestPyLastWithLastFm from .test_pylast import TestPyLastWithLastFm

View file

@ -2,6 +2,8 @@
""" """
Integration (not unit) tests for pylast.py Integration (not unit) tests for pylast.py
""" """
from __future__ import annotations
import pylast import pylast
from .test_pylast import TestPyLastWithLastFm from .test_pylast import TestPyLastWithLastFm

View file

@ -2,6 +2,8 @@
""" """
Integration (not unit) tests for pylast.py Integration (not unit) tests for pylast.py
""" """
from __future__ import annotations
from flaky import flaky from flaky import flaky
import pylast import pylast

View file

@ -1,6 +1,9 @@
""" """
Integration (not unit) tests for pylast.py Integration (not unit) tests for pylast.py
""" """
from __future__ import annotations
import re import re
import time import time

View file

@ -2,6 +2,8 @@
""" """
Integration (not unit) tests for pylast.py Integration (not unit) tests for pylast.py
""" """
from __future__ import annotations
import os import os
import time import time

View file

@ -2,6 +2,8 @@
""" """
Integration (not unit) tests for pylast.py Integration (not unit) tests for pylast.py
""" """
from __future__ import annotations
import pylast import pylast
from .test_pylast import TestPyLastWithLastFm from .test_pylast import TestPyLastWithLastFm

View file

@ -1,6 +1,9 @@
""" """
Integration (not unit) tests for pylast.py Integration (not unit) tests for pylast.py
""" """
from __future__ import annotations
import time import time
import pytest import pytest
@ -135,11 +138,7 @@ class TestPyLastTrack(TestPyLastWithLastFm):
similar = track.get_similar() similar = track.get_similar()
# Assert # Assert
found = False found = any(str(track.item) == "Cher - Strong Enough" for track in similar)
for track in similar:
if str(track.item) == "Madonna - Vogue":
found = True
break
assert found assert found
def test_track_get_similar_limits(self) -> None: def test_track_get_similar_limits(self) -> None:

View file

@ -2,6 +2,8 @@
""" """
Integration (not unit) tests for pylast.py Integration (not unit) tests for pylast.py
""" """
from __future__ import annotations
import calendar import calendar
import datetime as dt import datetime as dt
import inspect import inspect

View file

@ -1,3 +1,5 @@
from __future__ import annotations
from unittest import mock from unittest import mock
import pytest import pytest
@ -25,7 +27,7 @@ def test_get_cache_key(artist) -> None:
@pytest.mark.parametrize("obj", [pylast.Artist("B\xe9l", mock_network())]) @pytest.mark.parametrize("obj", [pylast.Artist("B\xe9l", mock_network())])
def test_cast_and_hash(obj) -> None: def test_cast_and_hash(obj) -> None:
assert type(str(obj)) is str assert isinstance(str(obj), str)
assert isinstance(hash(obj), int) assert isinstance(hash(obj), int)

11
tox.ini
View file

@ -3,7 +3,7 @@ requires =
tox>=4.2 tox>=4.2
env_list = env_list =
lint lint
py{py3, 312, 311, 310, 39, 38} py{py3, 313, 312, 311, 310, 39, 38}
[testenv] [testenv]
extras = extras =
@ -15,7 +15,14 @@ pass_env =
PYLAST_PASSWORD_HASH PYLAST_PASSWORD_HASH
PYLAST_USERNAME PYLAST_USERNAME
commands = commands =
pytest -v -s -W all --cov pylast --cov tests --cov-report term-missing --cov-report xml --random-order {posargs} {envpython} -m pytest -v -s -W all \
--cov pylast \
--cov tests \
--cov-report html \
--cov-report term-missing \
--cov-report xml \
--random-order \
{posargs}
[testenv:lint] [testenv:lint]
skip_install = true skip_install = true