chenrj
2 years ago
committed by
GitHub
18 changed files with 720 additions and 5 deletions
@ -0,0 +1,46 @@ |
|||||||
|
.. Licensed to the Apache Software Foundation (ASF) under one |
||||||
|
or more contributor license agreements. See the NOTICE file |
||||||
|
distributed with this work for additional information |
||||||
|
regarding copyright ownership. The ASF licenses this file |
||||||
|
to you under the Apache License, Version 2.0 (the |
||||||
|
"License"); you may not use this file except in compliance |
||||||
|
with the License. You may obtain a copy of the License at |
||||||
|
|
||||||
|
.. http://www.apache.org/licenses/LICENSE-2.0 |
||||||
|
|
||||||
|
.. Unless required by applicable law or agreed to in writing, |
||||||
|
software distributed under the License is distributed on an |
||||||
|
"AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
||||||
|
KIND, either express or implied. See the License for the |
||||||
|
specific language governing permissions and limitations |
||||||
|
under the License. |
||||||
|
|
||||||
|
How to develop |
||||||
|
============== |
||||||
|
|
||||||
|
When you want to create a new resource plugin, you need to add a new class in the module `resources_plugin`. |
||||||
|
|
||||||
|
The resource plug-in class needs to inherit the abstract class `ResourcePlugin` and implement its abstract method `read_file` function. |
||||||
|
|
||||||
|
The parameter of the `__init__` function of `ResourcePlugin` is the prefix of STR type. You can override this function when necessary. |
||||||
|
|
||||||
|
The `read_file` function parameter of `ResourcePlugin` is the file suffix of STR type, and its return value is the file content, if it exists and is readable. |
||||||
|
|
||||||
|
|
||||||
|
Example |
||||||
|
------- |
||||||
|
- Method `__init__`: Initiation method with `param`:`prefix` |
||||||
|
|
||||||
|
.. literalinclude:: ../../../src/pydolphinscheduler/resources_plugin/local.py |
||||||
|
:start-after: [start init_method] |
||||||
|
:end-before: [end init_method] |
||||||
|
|
||||||
|
- Method `read_file`: Get content from the given URI, The function parameter is the suffix of the file path. |
||||||
|
|
||||||
|
The file prefix has been initialized in init of the resource plug-in. |
||||||
|
|
||||||
|
The prefix plus suffix is the absolute path of the file in this resource. |
||||||
|
|
||||||
|
.. literalinclude:: ../../../src/pydolphinscheduler/resources_plugin/local.py |
||||||
|
:start-after: [start read_file_method] |
||||||
|
:end-before: [end read_file_method] |
@ -0,0 +1,28 @@ |
|||||||
|
.. Licensed to the Apache Software Foundation (ASF) under one |
||||||
|
or more contributor license agreements. See the NOTICE file |
||||||
|
distributed with this work for additional information |
||||||
|
regarding copyright ownership. The ASF licenses this file |
||||||
|
to you under the Apache License, Version 2.0 (the |
||||||
|
"License"); you may not use this file except in compliance |
||||||
|
with the License. You may obtain a copy of the License at |
||||||
|
|
||||||
|
.. http://www.apache.org/licenses/LICENSE-2.0 |
||||||
|
|
||||||
|
.. Unless required by applicable law or agreed to in writing, |
||||||
|
software distributed under the License is distributed on an |
||||||
|
"AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
||||||
|
KIND, either express or implied. See the License for the |
||||||
|
specific language governing permissions and limitations |
||||||
|
under the License. |
||||||
|
|
||||||
|
Resources_plugin |
||||||
|
================ |
||||||
|
|
||||||
|
In this section |
||||||
|
|
||||||
|
.. toctree:: |
||||||
|
:maxdepth: 1 |
||||||
|
|
||||||
|
develop |
||||||
|
resource-plugin |
||||||
|
local |
@ -0,0 +1,32 @@ |
|||||||
|
.. Licensed to the Apache Software Foundation (ASF) under one |
||||||
|
or more contributor license agreements. See the NOTICE file |
||||||
|
distributed with this work for additional information |
||||||
|
regarding copyright ownership. The ASF licenses this file |
||||||
|
to you under the Apache License, Version 2.0 (the |
||||||
|
"License"); you may not use this file except in compliance |
||||||
|
with the License. You may obtain a copy of the License at |
||||||
|
|
||||||
|
.. http://www.apache.org/licenses/LICENSE-2.0 |
||||||
|
|
||||||
|
.. Unless required by applicable law or agreed to in writing, |
||||||
|
software distributed under the License is distributed on an |
||||||
|
"AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
||||||
|
KIND, either express or implied. See the License for the |
||||||
|
specific language governing permissions and limitations |
||||||
|
under the License. |
||||||
|
|
||||||
|
Local |
||||||
|
===== |
||||||
|
|
||||||
|
`Local` is a local resource plugin for pydolphinscheduler. |
||||||
|
|
||||||
|
When using a local resource plugin, you only need to add the `resource_plugin` parameter in the task subclass or workflow definition, |
||||||
|
such as `resource_plugin=Local("/tmp")`. |
||||||
|
|
||||||
|
|
||||||
|
For the specific use of resource plugins, you can see `How to use` in :doc:`./resource-plugin` |
||||||
|
|
||||||
|
Dive Into |
||||||
|
--------- |
||||||
|
|
||||||
|
.. automodule:: pydolphinscheduler.resources_plugin.local |
@ -0,0 +1,75 @@ |
|||||||
|
.. Licensed to the Apache Software Foundation (ASF) under one |
||||||
|
or more contributor license agreements. See the NOTICE file |
||||||
|
distributed with this work for additional information |
||||||
|
regarding copyright ownership. The ASF licenses this file |
||||||
|
to you under the Apache License, Version 2.0 (the |
||||||
|
"License"); you may not use this file except in compliance |
||||||
|
with the License. You may obtain a copy of the License at |
||||||
|
|
||||||
|
.. http://www.apache.org/licenses/LICENSE-2.0 |
||||||
|
|
||||||
|
.. Unless required by applicable law or agreed to in writing, |
||||||
|
software distributed under the License is distributed on an |
||||||
|
"AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
||||||
|
KIND, either express or implied. See the License for the |
||||||
|
specific language governing permissions and limitations |
||||||
|
under the License. |
||||||
|
|
||||||
|
ResourcePlugin |
||||||
|
============== |
||||||
|
|
||||||
|
`ResourcePlugin` is an abstract class of resource plug-in parameters of task subclass and workflow. |
||||||
|
All resource plugins need to inherit and override its abstract methods. |
||||||
|
|
||||||
|
Code |
||||||
|
---- |
||||||
|
.. literalinclude:: ../../../src/pydolphinscheduler/core/resource_plugin.py |
||||||
|
:start-after: [start resource_plugin_definition] |
||||||
|
:end-before: [end resource_plugin_definition] |
||||||
|
|
||||||
|
Dive Into |
||||||
|
--------- |
||||||
|
It has the following key functions. |
||||||
|
|
||||||
|
- Method `__init__`: The `__init__` function has STR type parameter `prefix`, which means the prefix of the resource. |
||||||
|
|
||||||
|
You can rewrite this function if necessary. |
||||||
|
|
||||||
|
.. literalinclude:: ../../../src/pydolphinscheduler/core/resource_plugin.py |
||||||
|
:start-after: [start init_method] |
||||||
|
:end-before: [end init_method] |
||||||
|
|
||||||
|
- Method `read_file`: Get content from the given URI, The function parameter is the suffix of the file path. |
||||||
|
|
||||||
|
The file prefix has been initialized in init of the resource plug-in. |
||||||
|
|
||||||
|
The prefix plus suffix is the absolute path of the file in this resource. |
||||||
|
|
||||||
|
It is an abstract function. You must rewrite it |
||||||
|
|
||||||
|
.. literalinclude:: ../../../src/pydolphinscheduler/core/resource_plugin.py |
||||||
|
:start-after: [start abstractmethod read_file] |
||||||
|
:end-before: [end abstractmethod read_file] |
||||||
|
|
||||||
|
.. automodule:: pydolphinscheduler.core.resource_plugin |
||||||
|
|
||||||
|
How to use |
||||||
|
---------- |
||||||
|
Resource plug-ins can be used in task subclasses and workflows. You can use the resource plug-ins by adding the `resource_plugin` parameter when they are initialized. |
||||||
|
For example, local resource plug-ins, add `resource_plugin = Local("/tmp")`. |
||||||
|
|
||||||
|
The resource plug-ins we currently support is `local`. |
||||||
|
|
||||||
|
Here is an example. |
||||||
|
|
||||||
|
.. literalinclude:: ../../../src/pydolphinscheduler/examples/tutorial_resource_plugin.py |
||||||
|
:start-after: [start workflow_declare] |
||||||
|
:end-before: [end task_declare] |
||||||
|
|
||||||
|
When the resource_plugin parameter is defined in both the task subclass and the workflow, the resource_plugin defined in the task subclass is used first. |
||||||
|
|
||||||
|
If the task subclass does not define resource_plugin, but the resource_plugin is defined in the workflow, the resource_plugin in the workflow is used. |
||||||
|
|
||||||
|
Of course, if neither the task subclass nor the workflow specifies resource_plugin, the command at this time will be executed as a script, |
||||||
|
|
||||||
|
in other words, we are forward compatible. |
@ -0,0 +1,49 @@ |
|||||||
|
# Licensed to the Apache Software Foundation (ASF) under one |
||||||
|
# or more contributor license agreements. See the NOTICE file |
||||||
|
# distributed with this work for additional information |
||||||
|
# regarding copyright ownership. The ASF licenses this file |
||||||
|
# to you under the Apache License, Version 2.0 (the |
||||||
|
# "License"); you may not use this file except in compliance |
||||||
|
# with the License. You may obtain a copy of the License at |
||||||
|
# |
||||||
|
# http://www.apache.org/licenses/LICENSE-2.0 |
||||||
|
# |
||||||
|
# Unless required by applicable law or agreed to in writing, |
||||||
|
# software distributed under the License is distributed on an |
||||||
|
# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
||||||
|
# KIND, either express or implied. See the License for the |
||||||
|
# specific language governing permissions and limitations |
||||||
|
# under the License. |
||||||
|
|
||||||
|
"""DolphinScheduler ResourcePlugin object.""" |
||||||
|
|
||||||
|
|
||||||
|
from abc import ABCMeta, abstractmethod |
||||||
|
|
||||||
|
|
||||||
|
# [start resource_plugin_definition] |
||||||
|
class ResourcePlugin(object, metaclass=ABCMeta): |
||||||
|
"""ResourcePlugin object, declare resource plugin for task and workflow to dolphinscheduler. |
||||||
|
|
||||||
|
:param prefix: A string representing the prefix of ResourcePlugin. |
||||||
|
|
||||||
|
""" |
||||||
|
|
||||||
|
# [start init_method] |
||||||
|
def __init__(self, prefix: str, *args, **kwargs): |
||||||
|
self.prefix = prefix |
||||||
|
|
||||||
|
# [end init_method] |
||||||
|
|
||||||
|
# [start abstractmethod read_file] |
||||||
|
@abstractmethod |
||||||
|
def read_file(self, suf: str): |
||||||
|
"""Get the content of the file. |
||||||
|
|
||||||
|
The address of the file is the prefix of the resource plugin plus the parameter suf. |
||||||
|
""" |
||||||
|
|
||||||
|
# [end abstractmethod read_file] |
||||||
|
|
||||||
|
|
||||||
|
# [end resource_plugin_definition] |
@ -0,0 +1,64 @@ |
|||||||
|
# Licensed to the Apache Software Foundation (ASF) under one |
||||||
|
# or more contributor license agreements. See the NOTICE file |
||||||
|
# distributed with this work for additional information |
||||||
|
# regarding copyright ownership. The ASF licenses this file |
||||||
|
# to you under the Apache License, Version 2.0 (the |
||||||
|
# "License"); you may not use this file except in compliance |
||||||
|
# with the License. You may obtain a copy of the License at |
||||||
|
# |
||||||
|
# http://www.apache.org/licenses/LICENSE-2.0 |
||||||
|
# |
||||||
|
# Unless required by applicable law or agreed to in writing, |
||||||
|
# software distributed under the License is distributed on an |
||||||
|
# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
||||||
|
# KIND, either express or implied. See the License for the |
||||||
|
# specific language governing permissions and limitations |
||||||
|
# under the License. |
||||||
|
|
||||||
|
r""" |
||||||
|
A tutorial example take you to experience pydolphinscheduler resource plugin. |
||||||
|
|
||||||
|
Resource plug-ins can be defined in workflows and tasks |
||||||
|
|
||||||
|
it will instantiate and run all the task it have. |
||||||
|
""" |
||||||
|
import os |
||||||
|
from pathlib import Path |
||||||
|
|
||||||
|
# [start tutorial_resource_plugin] |
||||||
|
# [start package_import] |
||||||
|
# Import ProcessDefinition object to define your workflow attributes |
||||||
|
from pydolphinscheduler.core.process_definition import ProcessDefinition |
||||||
|
|
||||||
|
# Import task Shell object cause we would create some shell tasks later |
||||||
|
from pydolphinscheduler.resources_plugin.local import Local |
||||||
|
from pydolphinscheduler.tasks.shell import Shell |
||||||
|
|
||||||
|
# [end package_import] |
||||||
|
|
||||||
|
# [start workflow_declare] |
||||||
|
with ProcessDefinition( |
||||||
|
name="tutorial_resource_plugin", |
||||||
|
schedule="0 0 0 * * ? *", |
||||||
|
start_time="2021-01-01", |
||||||
|
tenant="tenant_exists", |
||||||
|
resource_plugin=Local("/tmp"), |
||||||
|
) as process_definition: |
||||||
|
# [end workflow_declare] |
||||||
|
# [start task_declare] |
||||||
|
file = "resource.sh" |
||||||
|
path = Path("/tmp").joinpath(file) |
||||||
|
with open(str(path), "w") as f: |
||||||
|
f.write("echo tutorial resource plugin") |
||||||
|
task_parent = Shell( |
||||||
|
name="local-resource-example", |
||||||
|
command=file, |
||||||
|
) |
||||||
|
print(task_parent.task_params) |
||||||
|
os.remove(path) |
||||||
|
# [end task_declare] |
||||||
|
|
||||||
|
# [start submit_or_run] |
||||||
|
process_definition.run() |
||||||
|
# [end submit_or_run] |
||||||
|
# [end tutorial_resource_plugin] |
@ -0,0 +1,23 @@ |
|||||||
|
# Licensed to the Apache Software Foundation (ASF) under one |
||||||
|
# or more contributor license agreements. See the NOTICE file |
||||||
|
# distributed with this work for additional information |
||||||
|
# regarding copyright ownership. The ASF licenses this file |
||||||
|
# to you under the Apache License, Version 2.0 (the |
||||||
|
# "License"); you may not use this file except in compliance |
||||||
|
# with the License. You may obtain a copy of the License at |
||||||
|
# |
||||||
|
# http://www.apache.org/licenses/LICENSE-2.0 |
||||||
|
# |
||||||
|
# Unless required by applicable law or agreed to in writing, |
||||||
|
# software distributed under the License is distributed on an |
||||||
|
# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
||||||
|
# KIND, either express or implied. See the License for the |
||||||
|
# specific language governing permissions and limitations |
||||||
|
# under the License. |
||||||
|
|
||||||
|
"""Init resources_plugin package.""" |
||||||
|
from pydolphinscheduler.resources_plugin.local import Local |
||||||
|
|
||||||
|
__all__ = [ |
||||||
|
"Local", |
||||||
|
] |
@ -0,0 +1,57 @@ |
|||||||
|
# Licensed to the Apache Software Foundation (ASF) under one |
||||||
|
# or more contributor license agreements. See the NOTICE file |
||||||
|
# distributed with this work for additional information |
||||||
|
# regarding copyright ownership. The ASF licenses this file |
||||||
|
# to you under the Apache License, Version 2.0 (the |
||||||
|
# "License"); you may not use this file except in compliance |
||||||
|
# with the License. You may obtain a copy of the License at |
||||||
|
# |
||||||
|
# http://www.apache.org/licenses/LICENSE-2.0 |
||||||
|
# |
||||||
|
# Unless required by applicable law or agreed to in writing, |
||||||
|
# software distributed under the License is distributed on an |
||||||
|
# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
||||||
|
# KIND, either express or implied. See the License for the |
||||||
|
# specific language governing permissions and limitations |
||||||
|
# under the License. |
||||||
|
|
||||||
|
"""DolphinScheduler local resource plugin.""" |
||||||
|
|
||||||
|
import os |
||||||
|
from pathlib import Path |
||||||
|
|
||||||
|
from pydolphinscheduler.core.resource_plugin import ResourcePlugin |
||||||
|
from pydolphinscheduler.exceptions import PyResPluginException |
||||||
|
|
||||||
|
|
||||||
|
class Local(ResourcePlugin): |
||||||
|
"""Local object, declare local resource plugin for task and workflow to dolphinscheduler. |
||||||
|
|
||||||
|
:param prefix: A string representing the prefix of Local. |
||||||
|
|
||||||
|
""" |
||||||
|
|
||||||
|
# [start init_method] |
||||||
|
def __init__(self, prefix: str, *args, **kwargs): |
||||||
|
super().__init__(prefix, *args, **kwargs) |
||||||
|
|
||||||
|
# [end init_method] |
||||||
|
|
||||||
|
# [start read_file_method] |
||||||
|
def read_file(self, suf: str): |
||||||
|
"""Get the content of the file. |
||||||
|
|
||||||
|
The address of the file is the prefix of the resource plugin plus the parameter suf. |
||||||
|
""" |
||||||
|
path = Path(self.prefix).joinpath(suf) |
||||||
|
if not path.exists(): |
||||||
|
raise PyResPluginException("{} is not found".format(str(path))) |
||||||
|
if not os.access(str(path), os.R_OK): |
||||||
|
raise PyResPluginException( |
||||||
|
"You don't have permission to access {}".format(self.prefix + suf) |
||||||
|
) |
||||||
|
with open(path, "r") as f: |
||||||
|
content = f.read() |
||||||
|
return content |
||||||
|
|
||||||
|
# [end read_file_method] |
@ -0,0 +1,18 @@ |
|||||||
|
# Licensed to the Apache Software Foundation (ASF) under one |
||||||
|
# or more contributor license agreements. See the NOTICE file |
||||||
|
# distributed with this work for additional information |
||||||
|
# regarding copyright ownership. The ASF licenses this file |
||||||
|
# to you under the Apache License, Version 2.0 (the |
||||||
|
# "License"); you may not use this file except in compliance |
||||||
|
# with the License. You may obtain a copy of the License at |
||||||
|
# |
||||||
|
# http://www.apache.org/licenses/LICENSE-2.0 |
||||||
|
# |
||||||
|
# Unless required by applicable law or agreed to in writing, |
||||||
|
# software distributed under the License is distributed on an |
||||||
|
# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
||||||
|
# KIND, either express or implied. See the License for the |
||||||
|
# specific language governing permissions and limitations |
||||||
|
# under the License. |
||||||
|
|
||||||
|
"""Init resources_plugin package tests.""" |
@ -0,0 +1,108 @@ |
|||||||
|
# Licensed to the Apache Software Foundation (ASF) under one |
||||||
|
# or more contributor license agreements. See the NOTICE file |
||||||
|
# distributed with this work for additional information |
||||||
|
# regarding copyright ownership. The ASF licenses this file |
||||||
|
# to you under the Apache License, Version 2.0 (the |
||||||
|
# "License"); you may not use this file except in compliance |
||||||
|
# with the License. You may obtain a copy of the License at |
||||||
|
# |
||||||
|
# http://www.apache.org/licenses/LICENSE-2.0 |
||||||
|
# |
||||||
|
# Unless required by applicable law or agreed to in writing, |
||||||
|
# software distributed under the License is distributed on an |
||||||
|
# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
||||||
|
# KIND, either express or implied. See the License for the |
||||||
|
# specific language governing permissions and limitations |
||||||
|
# under the License. |
||||||
|
|
||||||
|
"""Test local resource plugin.""" |
||||||
|
from pathlib import Path |
||||||
|
from unittest.mock import PropertyMock, patch |
||||||
|
|
||||||
|
import pytest |
||||||
|
|
||||||
|
from pydolphinscheduler.core import Task |
||||||
|
from pydolphinscheduler.exceptions import PyResPluginException |
||||||
|
from pydolphinscheduler.resources_plugin.local import Local |
||||||
|
from pydolphinscheduler.utils import file |
||||||
|
from tests.testing.file import delete_file |
||||||
|
|
||||||
|
file_name = "local_res.sh" |
||||||
|
file_content = "echo Test local res plugin" |
||||||
|
res_plugin_prefix = Path(__file__).parent |
||||||
|
file_path = res_plugin_prefix.joinpath(file_name) |
||||||
|
|
||||||
|
|
||||||
|
@pytest.fixture() |
||||||
|
def setup_crt_first(): |
||||||
|
"""Set up and teardown about create file first and then delete it.""" |
||||||
|
file.write(content=file_content, to_path=file_path) |
||||||
|
yield |
||||||
|
delete_file(file_path) |
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.parametrize( |
||||||
|
"val, expected", |
||||||
|
[ |
||||||
|
(file_name, file_content), |
||||||
|
], |
||||||
|
) |
||||||
|
@patch( |
||||||
|
"pydolphinscheduler.core.task.Task.gen_code_and_version", |
||||||
|
return_value=(123, 1), |
||||||
|
) |
||||||
|
@patch( |
||||||
|
"pydolphinscheduler.core.task.Task.ext", |
||||||
|
new_callable=PropertyMock, |
||||||
|
return_value={ |
||||||
|
".sh", |
||||||
|
}, |
||||||
|
) |
||||||
|
@patch( |
||||||
|
"pydolphinscheduler.core.task.Task.ext_attr", |
||||||
|
new_callable=PropertyMock, |
||||||
|
return_value="_raw_script", |
||||||
|
) |
||||||
|
@patch( |
||||||
|
"pydolphinscheduler.core.task.Task._raw_script", |
||||||
|
create=True, |
||||||
|
new_callable=PropertyMock, |
||||||
|
) |
||||||
|
def test_task_obtain_res_plugin( |
||||||
|
m_raw_script, m_ext_attr, m_ext, m_code_version, val, expected, setup_crt_first |
||||||
|
): |
||||||
|
"""Test task obtaining resource plug-in.""" |
||||||
|
m_raw_script.return_value = val |
||||||
|
task = Task( |
||||||
|
name="test_task_ext_attr", |
||||||
|
task_type="type", |
||||||
|
resource_plugin=Local(str(res_plugin_prefix)), |
||||||
|
) |
||||||
|
assert expected == getattr(task, "raw_script") |
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.parametrize( |
||||||
|
"attr, expected", |
||||||
|
[({"prefix": res_plugin_prefix, "file_name": file_name}, file_content)], |
||||||
|
) |
||||||
|
def test_local_res_read_file(attr, expected, setup_crt_first): |
||||||
|
"""Test the read_file function of the local resource plug-in.""" |
||||||
|
local = Local(str(attr.get("prefix"))) |
||||||
|
local.read_file(attr.get("file_name")) |
||||||
|
assert expected == local.read_file(file_name) |
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.parametrize( |
||||||
|
"attr", |
||||||
|
[ |
||||||
|
{"prefix": res_plugin_prefix, "file_name": file_name}, |
||||||
|
], |
||||||
|
) |
||||||
|
def test_local_res_file_not_found(attr): |
||||||
|
"""Test local resource plugin file does not exist.""" |
||||||
|
with pytest.raises( |
||||||
|
PyResPluginException, |
||||||
|
match=".* is not found", |
||||||
|
): |
||||||
|
local = Local(str(attr.get("prefix"))) |
||||||
|
local.read_file(attr.get("file_name")) |
Loading…
Reference in new issue