Skip to main content

File Transfer

File transfer in DNP3 is defining using objects from Group 70. The Master API allows you to do several operations related to file transfer:

  • Get file info - Retrieve information on an individual file or directory
  • Read file - Read a remote file from the outstation asynchronously
  • Read directory listing - Read a directory information from the outstation

Get File Info

Information about a single file or directory may be retrieved from outstation using a single method invocation.

match association.get_file_info(".").await {
Err(err) => {
tracing::warn!("Unable to get file info: {err}");
}
Ok(info) => {
print_file_info(info);
}
}

Read a File

File data may be asynchronously read from the outstation using a callback interface. Reading a file consists of multiple DNP3 requests. Obtaining an authentication key from the outstation and is optional depending on outstation requirements.

sequenceDiagram Master-->>Outstation: authenticate_file (optional); Outstation-->>Master: response w/ key; Master->>Outstation: read file block (1 of N); Outstation->>Master: response (1 of N); Master->>Outstation: close file; Outstation->>Master: response;

The FileReader interface is used to receive data from the outstation as it is read. Your code is responsible for doing something with this data, e.g. writing it to a local file.

struct FileLogger;

impl FileReader for FileLogger {
fn opened(&mut self, size: u32) -> FileAction {
tracing::info!("File opened - size: {size}");
FileAction::Continue
}

fn block_received(&mut self, block_num: u32, data: &[u8]) -> MaybeAsync<FileAction> {
tracing::info!("Received block {block_num} with size: {}", data.len());
MaybeAsync::ready(FileAction::Continue)
}

fn aborted(&mut self, err: FileError) {
tracing::info!("File transfer aborted: {}", err);
}

fn completed(&mut self) {
tracing::info!("File transfer completed");
}
}

// asynchronously start the transfer
if let Err(err) = association
.read_file(
".", // this reads the root "directory" file but doesn't parse it
FileReadConfig::default(),
Box::new(FileLogger),
None,
)
.await
{
tracing::warn!("Unable to start file transfer: {err}");
}
tip

There is no need to manually close a file. The master will always attempt to close a file, even if the file operation is aborted by the user.

Read a Directory

A directory listing in DNP3 is just a special kind of file with a special encoding. We implement this feature by starting a background file transfer, accumulating the file data over possibly multiple blocks, and then parsing it into a collection of FileInfo instances.

danger

The default limit for the size of directory file is 2048 bytes. This is a reasonable and safe limit, however, if its too small in your application you can raise this limit using the provided parameter in DirReadConfig.

match association
.read_directory(".", DirReadConfig::default(), None)
.await
{
Err(err) => {
tracing::warn!("Unable to read directory: {err}");
}
Ok(items) => {
for info in items {
print_file_info(info);
}
}
}