Skip to content
Snippets Groups Projects
Commit 013486f5 authored by Moul's avatar Moul
Browse files

Fix API.request_url() with Py3.13 (#208)

Since Python 3.13, the IOBase finalizer now logs
any errors raised by the close() method
https://docs.python.org/3/whatsnew/3.13.html#io

According to
https://docs.python.org/3.13/library/io.html#io.IOBase.close
if the file (descriptor) is accessed after being close,
`ValueError` is raised.

The `fd` is copied with `copy.copy()`.
Both gets automatically closed once the function ends,
the second can’t close a second time the same fd,
that’s why we get this error.

Reorganise complex function: don’t use `with` since the function
is designed to allow returning the fd
Drop `copy.copy()` used previously to return the fd, causing the issue
Read once the content.
Cleanly separate three return types and function’s returns
parent 58540690
No related branches found
No related tags found
No related merge requests found
Pipeline #38527 passed
......@@ -13,10 +13,8 @@
# You should have received a copy of the GNU General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
import copy
import json
import logging
from http.client import HTTPResponse
from typing import Any, Callable, Dict, Optional, Union
from urllib import parse, request
......@@ -252,39 +250,31 @@ class API:
self.connection_handler.proxy, self.connection_handler.http_scheme
)
with request.urlopen(
response = request.urlopen( # pylint: disable=consider-using-with
duniter_request, timeout=15
) as response: # type: HTTPResponse
if response.status != 200:
content = response.read().decode("utf-8")
if bma_errors:
try:
error_data = parse_error(content)
raise DuniterError(error_data)
except (TypeError, jsonschema.ValidationError) as exception:
raise ValueError(
f"status code != 200 => {response.status} ({content})"
) from exception
raise ValueError(f"status code != 200 => {response.status} ({content})")
# get response content
return_response = copy.copy(response)
content = response.read().decode("utf-8")
# if schema supplied...
if schema is not None:
)
content = response.read().decode("utf-8")
if response.status != 200:
if bma_errors:
try:
error_data = parse_error(content)
raise DuniterError(error_data)
except (TypeError, jsonschema.ValidationError) as exception:
raise ValueError(
f"status code != 200 => {response.status} ({content})"
) from exception
raise ValueError(f"status code != 200 => {response.status} ({content})")
if rtype == RESPONSE_HTTP:
return response
# if schema supplied…
if schema:
# validate response
parse_response(content, schema)
# return the chosen type
result = return_response # type: Any
if rtype == RESPONSE_TEXT:
result = content
elif rtype == RESPONSE_JSON:
result = json.loads(content)
return result
return content
return json.loads(content)
def connect_ws(self, path: str) -> WSConnection:
"""
......
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment