-
Notifications
You must be signed in to change notification settings - Fork 365
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Add client low-level file system functionality (#696)
* Add low-level file system functionality * Added examples for the use of ua_file_transfer.py In the following you will find examples for the use of the classes UaFile and UaDirectory and how to handle typical uaerrors. see: ./asyncua/client/ua_file_transfer.py. The OPC UA File Transfer specification can be found here: https://reference.opcfoundation.org/Core/docs/Part5/C.1/ * pylint optimized - ua_file_transfer.py pylint optimization of file ua_file_transfer.py. => Your code has been rated at 10.00/10 (previous run: 10.00/10, +0.00) * pylint optimized - client_ua_file_transfer.py pylint optimization of client_ua_file_transfer.py. => Your code has been rated at 10.00/10 (previous run: 10.00/10, +0.00) Co-authored-by: Curious Crook <[email protected]>
- Loading branch information
1 parent
0f4f68c
commit f57477a
Showing
2 changed files
with
343 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,255 @@ | ||
""" | ||
Low level implementation of OPC UA File Transfer. | ||
This module contains the mandatory functionality specified | ||
by the OPC Foundation. | ||
See also: | ||
OPC 10000-5: OPC Unified Architecture V1.04 | ||
Part 5: Information Model - Annex C (normative) File Transfer | ||
https://reference.opcfoundation.org/Core/docs/Part5/C.1/ | ||
""" | ||
import logging | ||
|
||
from asyncua.ua import OpenFileMode, Variant, VariantType, NodeId | ||
from asyncua.common.node import Node | ||
|
||
_logger = logging.getLogger(__name__) | ||
|
||
|
||
class UaFile: | ||
""" | ||
Provides the functionality to work with "C.2 FileType". | ||
""" | ||
def __init__(self, file_node: Node, open_mode: OpenFileMode = OpenFileMode.Read.value): | ||
self._file_node = file_node | ||
self._open_mode = open_mode | ||
self._file_handle = None | ||
|
||
async def __aenter__(self): | ||
await self.open(self._open_mode) | ||
return self | ||
|
||
async def __aexit__(self, exc_type, exc_value, traceback): | ||
return await self.close() | ||
|
||
async def open(self, open_mode: OpenFileMode = None) -> None: | ||
""" | ||
Open is used to open a file represented by an Object of FileType. | ||
The open mode of OPC UA differs significantly from the | ||
python build-in functionality. | ||
=> See the OPC UA specification for more information. | ||
:param open_mode: Open mode defined in C.2.1. | ||
:return: The file handle. | ||
""" | ||
_logger.debug("Request to open file %s in mode: %s", self._file_node, OpenFileMode) | ||
open_node = await self._file_node.get_child("Open") | ||
arg1_mode = Variant(open_mode or self._open_mode, VariantType.Byte) | ||
self._file_handle = await self._file_node.call_method(open_node, arg1_mode) | ||
|
||
async def close(self) -> None: | ||
""" | ||
Close is used to close a file represented by a FileType. | ||
When a client closes a file the handle becomes invalid. | ||
""" | ||
_logger.debug("Request to close file %s", self._file_node) | ||
read_node = await self._file_node.get_child("Close") | ||
arg1_file_handle = Variant(self._file_handle, VariantType.UInt32) | ||
await self._file_node.call_method(read_node, arg1_file_handle) | ||
|
||
async def read(self) -> bytes: | ||
""" | ||
Read is used to read a part of the file starting from the current file position. | ||
The file position is advanced by the number of bytes read. | ||
:return: Contains the returned data of the file. | ||
If the ByteString is empty it indicates that the end of the file is reached. | ||
""" | ||
_logger.debug("Request to read from file %s", self._file_node) | ||
size = await self.get_size() | ||
read_node = await self._file_node.get_child("Read") | ||
arg1_file_handle = Variant(self._file_handle, VariantType.UInt32) | ||
arg2_length = Variant(size, VariantType.Int32) | ||
return await self._file_node.call_method(read_node, arg1_file_handle, arg2_length) | ||
|
||
async def write(self, data: bytes) -> None: | ||
""" | ||
Write is used to write a part of the file starting from the current file position. | ||
The file position is advanced by the number of bytes written. | ||
:param data: Contains the data to be written at the position of the file. | ||
It is server-dependent whether the written data are persistently | ||
stored if the session is ended without calling the Close Method with the fileHandle. | ||
Writing an empty or null ByteString returns a Good result code without any | ||
affect on the file. | ||
""" | ||
_logger.debug("Request to write to file %s", self._file_node) | ||
write_node = await self._file_node.get_child("Write") | ||
arg1_file_handle = Variant(self._file_handle, VariantType.UInt32) | ||
arg2_data = Variant(data, VariantType.ByteString) | ||
await self._file_node.call_method(write_node, arg1_file_handle, arg2_data) | ||
|
||
async def get_position(self) -> int: | ||
""" | ||
GetPosition is used to provide the current position of the file handle. | ||
:return: The position of the fileHandle in the file. | ||
If a Read or Write is called it starts at that position. | ||
""" | ||
_logger.debug("Request to get position from file %s", self._file_node) | ||
get_position_node = await self._file_node.get_child("GetPosition") | ||
arg1_file_handle = Variant(self._file_handle, VariantType.UInt32) | ||
return await self._file_node.call_method(get_position_node, arg1_file_handle) | ||
|
||
async def set_position(self, position: int) -> None: | ||
""" | ||
SetPosition is used to set the current position of the file handle. | ||
:param position: The position to be set for the fileHandle in the file. | ||
If a Read or Write is called it starts at that position. | ||
If the position is higher than the file size the position is set to the end of the file. | ||
""" | ||
_logger.debug("Request to set position in file %s", self._file_node) | ||
set_position_node = await self._file_node.get_child("SetPosition") | ||
arg1_file_handle = Variant(self._file_handle, VariantType.UInt32) | ||
arg2_position = Variant(position, VariantType.UInt64) | ||
return await self._file_node.call_method(set_position_node, arg1_file_handle, arg2_position) | ||
|
||
async def get_size(self) -> int: | ||
""" | ||
Size defines the size of the file in Bytes. | ||
When a file is opened for write the size might not be accurate. | ||
:return: The size of the file in Bytes. | ||
""" | ||
_logger.debug("Request to get size of file %s", self._file_node) | ||
size_node = await self._file_node.get_child("Size") | ||
return await size_node.read_value() | ||
|
||
async def get_writable(self) -> bool: | ||
""" | ||
Writable indicates whether the file is writable. | ||
It does not take any user access rights into account, i.e. although the file | ||
is writable this may be restricted to a certain user / user group. | ||
The Property does not take into account whether the file is currently | ||
opened for writing by another client and thus currently locked and not writable by others. | ||
:return: | ||
""" | ||
_logger.debug("Request to get writable of file %s", self._file_node) | ||
writable_node = await self._file_node.get_child("Writable") | ||
return await writable_node.read_value() | ||
|
||
async def get_user_writable(self) -> bool: | ||
""" | ||
UserWritable indicates whether the file is writable taking user access rights into account. | ||
The Property does not take into account whether the file is currently opened | ||
for writing by another client and thus currently locked and not writable by others. | ||
:return: Indicates whether the file is writable taking user access rights into account | ||
""" | ||
_logger.debug("Request to get user writable of file %s", self._file_node) | ||
user_writable_node = await self._file_node.get_child("UserWritable") | ||
return await user_writable_node.read_value() | ||
|
||
async def get_open_count(self): | ||
""" | ||
OpenCount indicates the number of currently valid file handles on the file. | ||
:return: Amount of currently valid file handles on the file | ||
""" | ||
_logger.debug("Request to get open count of file %s", self._file_node) | ||
open_count_node = await self._file_node.get_child("OpenCount") | ||
return await open_count_node.read_value() | ||
|
||
|
||
class UaDirectory: | ||
""" | ||
Provides the functionality to work with "C.3 File System". | ||
""" | ||
def __init__(self, directory_node): | ||
self._directory_node = directory_node | ||
|
||
async def create_directory(self, directory_name: str) -> NodeId: | ||
""" | ||
CreateDirectory is used to create a new FileDirectoryType Object organized by this Object. | ||
:param directory_name: The name of the directory to create. | ||
The name is used for the BrowseName and DisplayName of the directory object and also | ||
for the directory in the file system. | ||
For the BrowseName, the directoryName is used for the name part of the QualifiedName. | ||
The namespace index is Server specific. | ||
For the DisplayName, the directoryName is used for the text part of the LocalizedText. | ||
The locale part is Server specific. | ||
:return: The NodeId of the created directory Object. | ||
""" | ||
_logger.debug("Request to create directory %s in %s", directory_name, self._directory_node) | ||
create_directory_node = await self._directory_node.get_child("CreateDirectory") | ||
arg1_directory_name = Variant(directory_name, VariantType.String) | ||
return await self._directory_node.call_method(create_directory_node, arg1_directory_name) | ||
|
||
async def create_file(self, file_name: str, request_file_open: bool) -> [NodeId, int]: | ||
""" | ||
CreateFile is used to create a new FileType Object organized by this Object. | ||
The created file can be written using the Write Method of the FileType. | ||
:param file_name: The name of the file to create. The name is used for the | ||
BrowseName and DisplayName of the file object and also for the file in the | ||
file system. | ||
For the BrowseName, the fileName is used for the name part of the QualifiedName. | ||
The namespace index is Server specific. For the DisplayName, the fileName is | ||
used for the text part of the LocalizedText. The locale part is Server specific. | ||
:param request_file_open: Flag indicating if the new file should be opened | ||
with the Write and Read bits set in the open mode after the creation of the file. | ||
If the flag is set to True, the file is created and opened for writing. | ||
If the flag is set to False, the file is just created. | ||
:return: The NodeId of the created file Object. | ||
The fileHandle is returned if the requestFileOpen is set to True. | ||
The fileNodeId and the fileHandle can be used to access the new file | ||
through the FileType Object representing the new file. | ||
If requestFileOpen is set to False, the returned value shall be 0 | ||
and shall be ignored by the caller. | ||
""" | ||
_logger.debug("Request to create file %s in %s", file_name, self._directory_node) | ||
print(f"Request to create file {file_name} in {self._directory_node}") | ||
create_file_node = await self._directory_node.get_child("CreateFile") | ||
arg1_file_name = Variant(file_name, VariantType.String) | ||
arg2_request_file_open = Variant(request_file_open, VariantType.Boolean) | ||
return await self._directory_node.call_method(create_file_node, | ||
arg1_file_name, | ||
arg2_request_file_open) | ||
|
||
async def delete(self, object_to_delete: NodeId) -> None: | ||
""" | ||
Delete is used to delete a file or directory organized by this Object. | ||
:param object_to_delete: The NodeId of the file or directory to delete. | ||
In the case of a directory, all file and directory Objects below the | ||
directory to delete are deleted recursively. | ||
""" | ||
_logger.debug("Request to delete file %s from %s", object_to_delete, self._directory_node) | ||
delete_node = await self._directory_node.get_child("Delete") | ||
await self._directory_node.call_method(delete_node, object_to_delete) | ||
|
||
async def move_or_copy(self, | ||
object_to_move_or_copy: NodeId, | ||
target_directory: NodeId, | ||
create_copy: bool, | ||
new_name: str) -> NodeId: | ||
""" | ||
MoveOrCopy is used to move or copy a file or directory organized by this Object | ||
to another directory or to rename a file or directory. | ||
:param object_to_move_or_copy: The NodeId of the file or directory to move or copy. | ||
:param target_directory: The NodeId of the target directory of the move or copy command. | ||
If the file or directory is just renamed, the targetDirectory matches the ObjectId | ||
passed to the method call. | ||
:param create_copy: A flag indicating if a copy of the file or directory should be | ||
created at the target directory. | ||
:param new_name: The new name of the file or directory in the new location. | ||
If the string is empty, the name is unchanged. | ||
:return: The NodeId of the moved or copied object. Even if the Object is moved, | ||
the Server may return a new NodeId. | ||
""" | ||
_logger.debug("Request to %s%s file system object %s from %s to %s, new name=%s", | ||
'' if create_copy else 'move', | ||
'copy' if create_copy else '', | ||
object_to_move_or_copy, | ||
self._directory_node, | ||
target_directory, | ||
new_name) | ||
move_or_copy_node = await self._directory_node.get_child("MoveOrCopy") | ||
arg3_create_copy = Variant(create_copy, VariantType.Boolean) | ||
arg4_new_name = Variant(new_name, VariantType.String) | ||
return await self._directory_node.call_method(move_or_copy_node, | ||
object_to_move_or_copy, | ||
target_directory, | ||
arg3_create_copy, | ||
arg4_new_name) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,88 @@ | ||
""" | ||
In the following you will find examples for the use of the | ||
classes UaFile and UaDirectory and how to handle typical uaerrors. | ||
see: ./asyncua/client/ua_file_transfer.py. | ||
The two classes are close to OPC UA File Transfer. This means they | ||
have to be operated "OPC UA typical" and not "Python typical". | ||
The advantage of this case is, that you can use the maximum | ||
range of functions of OPC UA File Transfer. | ||
See: https://reference.opcfoundation.org/Core/docs/Part5/C.1/ | ||
IMPORTANT NOTE: | ||
In order to to test the functions from UaFile and UaDiretory, | ||
you need an OPC UA server that offers the required File Transfer functionality. | ||
However, in this project there is currently no demo server containing | ||
file transfer capabilities. | ||
""" | ||
import asyncio | ||
import logging | ||
|
||
from asyncua import Client | ||
from asyncua.client.ua_file_transfer import UaFile, UaDirectory | ||
from asyncua.ua import OpenFileMode | ||
from asyncua.ua import uaerrors | ||
|
||
_logger = logging.getLogger("asyncua") | ||
|
||
|
||
async def task(): | ||
"""All communication takes place within this task. | ||
For the sake of simplicity, all (OPC UA) calls are executed purely sequentially. | ||
""" | ||
url = "opc.tcp://localhost:4840/freeopcua/server/" | ||
|
||
async with Client(url=url) as client: | ||
uri = "http://examples.freeopcua.github.io" | ||
idx = await client.get_namespace_index(uri) | ||
|
||
remote_file_system = await client.nodes.objects.get_child([f"{idx}:FileSystem"]) | ||
remote_file_name = "test.txt" | ||
remote_file_node = await remote_file_system.get_child([f"{idx}:{remote_file_name}"]) | ||
|
||
# Read file from server | ||
remote_file_content = None | ||
async with UaFile(remote_file_node, OpenFileMode.Read.value) as remote_file: | ||
remote_file_content = await remote_file.read() | ||
print("File content:") | ||
print(remote_file_content, end="\n\n") | ||
|
||
# Create file on server | ||
new_file_name = "new_file.txt" | ||
ua_dir = UaDirectory(remote_file_system) | ||
try: | ||
await ua_dir.create_file(new_file_name, False) | ||
except uaerrors.BadBrowseNameDuplicated: | ||
_logger.warning("=> File '%s' already exists on server.", new_file_name) | ||
|
||
# Write to file on server | ||
file_content = ("I am a random file\n" * 3).encode('utf-8') | ||
# In order to write to a file, it must already exist on the target system. (OPC UA typical) | ||
remote_file_node = await remote_file_system.get_child(f"{uri}:{new_file_name}") | ||
# In order to write to a file, you need the OpenFileModes "Write" | ||
# and one of the following "Append" or "EraseExisting". (OPC UA typical too) | ||
async with UaFile(remote_file_node, | ||
OpenFileMode.Write + OpenFileMode.EraseExisting) as remote_file: | ||
await remote_file.write(file_content) | ||
|
||
# Append to file on server | ||
file_content = ("I am appended text\n" * 3).encode('utf-8') | ||
remote_file_node = await remote_file_system.get_child(f"{uri}:{new_file_name}") | ||
async with UaFile(remote_file_node, | ||
OpenFileMode.Write + OpenFileMode.Append) as remote_file: | ||
await remote_file.write(file_content) | ||
|
||
# Get size of remote file | ||
file_size = await UaFile(remote_file_node).get_size() | ||
print(f"Size of file node '{remote_file_node}' = {file_size} byte") | ||
|
||
# Delete file from server | ||
try: | ||
await ua_dir.delete(remote_file_node.nodeid) | ||
except uaerrors.BadNotFound: | ||
_logger.warning("File %s not found on server.", remote_file_node) | ||
|
||
|
||
if __name__ == "__main__": | ||
logging.basicConfig(level=logging.WARNING) | ||
asyncio.run(task()) |