From bee986c72cb671ccbbd20b79f296ac1a42cedefc Mon Sep 17 00:00:00 2001
From: Stanislav Bogatyrev <stanislav@nspcc.ru>
Date: Thu, 17 Dec 2020 17:55:53 +0300
Subject: [PATCH] [#109] Define data audit result structure

Inner ring nodes conduct data audit and submit audit results. Results are then
saved in audit smart contract for settlement routines to transfer payments.

We don't need to save the full detailed audit report, but only provide enough
information for other subsystems to reward or punish storage nodes.

Signed-off-by: Stanislav Bogatyrev <stanislav@nspcc.ru>
---
 audit/types.proto   | 59 ++++++++++++++++++++++++++++++++++++
 proto-docs/audit.md | 74 +++++++++++++++++++++++++++++++++++++++++++++
 2 files changed, 133 insertions(+)
 create mode 100644 audit/types.proto
 create mode 100644 proto-docs/audit.md

diff --git a/audit/types.proto b/audit/types.proto
new file mode 100644
index 0000000..551e8c2
--- /dev/null
+++ b/audit/types.proto
@@ -0,0 +1,59 @@
+syntax = "proto3";
+
+package neo.fs.v2.audit;
+
+option go_package = "github.com/nspcc-dev/neofs-api-go/v2/audit/grpc;audit";
+option csharp_namespace = "NeoFS.API.v2.Audit";
+
+import "refs/types.proto";
+
+// DataAuditResult keeps record of conducted Data Audits. The detailed report is
+// generated separately.
+message DataAuditResult {
+  // Data Audit Result format version. Effectively the version of API library
+  // used to report DataAuditResult structure.
+  neo.fs.v2.refs.Version version = 1 [json_name = "version"];
+
+  // Epoch number when the Data Audit was conducted
+  fixed64 audit_epoch = 2 [json_name = "auditEpoch"];
+
+  // Container under audit
+  neo.fs.v2.refs.ContainerID container_id = 3 [json_name = "containerID"];
+
+  // Public key of the auditing InnerRing node in a binary format
+  bytes public_key = 4 [json_name = "publicKey"];
+
+  // Shows if Data Audit process was complete in time or if it was cancelled
+  bool complete = 5 [json_name = "complete"];
+
+  // Number of request done at PoR stage
+  uint32 requests = 6 [json_name = "requests"];
+
+  // Number of retries done at PoR stage
+  uint32 retries = 7 [json_name = "retries"];
+
+  // List of Storage Groups that passed audit PoR stage
+  repeated neo.fs.v2.refs.ObjectID pass_sg = 8 [json_name = "passSG"];
+
+  // List of Storage Groups that failed audit PoR stage
+  repeated neo.fs.v2.refs.ObjectID fail_sg = 9 [json_name = "failSG"];
+
+  // Number of sampled objects under audit placed in an optimal way according to
+  // the containers placement policy when checking PoP
+  uint32 hit = 10 [json_name = "hit"];
+
+  // Number of sampled objects under audit placed in suboptimal way according to
+  // the containers placement policy, but still at a satisfactory level when
+  // checking PoP
+  uint32 miss = 11 [json_name = "miss"];
+
+  // Number of sampled objects under audit stored in a way not confirming
+  // placement policy or not found at all when checking PoP
+  uint32 fail = 12 [json_name = "fail"];
+
+  // List of storage node public keys that passed at least one PDP
+  repeated bytes pass_nodes = 13 [json_name = "passNodes"];
+
+  // List of storage node public keys that failed at least one PDP
+  repeated bytes fail_nodes = 14 [json_name = "failNodes"];
+}
diff --git a/proto-docs/audit.md b/proto-docs/audit.md
new file mode 100644
index 0000000..073c2cb
--- /dev/null
+++ b/proto-docs/audit.md
@@ -0,0 +1,74 @@
+# Protocol Documentation
+<a name="top"></a>
+
+## Table of Contents
+
+- [audit/types.proto](#audit/types.proto)
+
+  - Messages
+    - [DataAuditResult](#neo.fs.v2.audit.DataAuditResult)
+    
+
+- [Scalar Value Types](#scalar-value-types)
+
+
+
+<a name="audit/types.proto"></a>
+<p align="right"><a href="#top">Top</a></p>
+
+## audit/types.proto
+
+
+ <!-- end services -->
+
+
+<a name="neo.fs.v2.audit.DataAuditResult"></a>
+
+### Message DataAuditResult
+DataAuditResult keeps record of conducted Data Audits. The detailed report is
+generated separately.
+
+
+| Field | Type | Label | Description |
+| ----- | ---- | ----- | ----------- |
+| version | [neo.fs.v2.refs.Version](#neo.fs.v2.refs.Version) |  | Data Audit Result format version. Effectively the version of API library used to report DataAuditResult structure. |
+| audit_epoch | [fixed64](#fixed64) |  | Epoch number when the Data Audit was conducted |
+| container_id | [neo.fs.v2.refs.ContainerID](#neo.fs.v2.refs.ContainerID) |  | Container under audit |
+| public_key | [bytes](#bytes) |  | Public key of the auditing InnerRing node in a binary format |
+| complete | [bool](#bool) |  | Shows if Data Audit process was complete in time or if it was cancelled |
+| requests | [uint32](#uint32) |  | Number of request done at PoR stage |
+| retries | [uint32](#uint32) |  | Number of retries done at PoR stage |
+| pass_sg | [neo.fs.v2.refs.ObjectID](#neo.fs.v2.refs.ObjectID) | repeated | List of Storage Groups that passed audit PoR stage |
+| fail_sg | [neo.fs.v2.refs.ObjectID](#neo.fs.v2.refs.ObjectID) | repeated | List of Storage Groups that failed audit PoR stage |
+| hit | [uint32](#uint32) |  | Number of sampled objects under audit placed in an optimal way according to the containers placement policy when checking PoP |
+| miss | [uint32](#uint32) |  | Number of sampled objects under audit placed in suboptimal way according to the containers placement policy, but still at a satisfactory level when checking PoP |
+| fail | [uint32](#uint32) |  | Number of sampled objects under audit stored in a way not confirming placement policy or not found at all when checking PoP |
+| pass_nodes | [bytes](#bytes) | repeated | List of storage node public keys that passed at least one PDP |
+| fail_nodes | [bytes](#bytes) | repeated | List of storage node public keys that failed at least one PDP |
+
+ <!-- end messages -->
+
+ <!-- end enums -->
+
+
+
+## Scalar Value Types
+
+| .proto Type | Notes | C++ Type | Java Type | Python Type |
+| ----------- | ----- | -------- | --------- | ----------- |
+| <a name="double" /> double |  | double | double | float |
+| <a name="float" /> float |  | float | float | float |
+| <a name="int32" /> int32 | Uses variable-length encoding. Inefficient for encoding negative numbers – if your field is likely to have negative values, use sint32 instead. | int32 | int | int |
+| <a name="int64" /> int64 | Uses variable-length encoding. Inefficient for encoding negative numbers – if your field is likely to have negative values, use sint64 instead. | int64 | long | int/long |
+| <a name="uint32" /> uint32 | Uses variable-length encoding. | uint32 | int | int/long |
+| <a name="uint64" /> uint64 | Uses variable-length encoding. | uint64 | long | int/long |
+| <a name="sint32" /> sint32 | Uses variable-length encoding. Signed int value. These more efficiently encode negative numbers than regular int32s. | int32 | int | int |
+| <a name="sint64" /> sint64 | Uses variable-length encoding. Signed int value. These more efficiently encode negative numbers than regular int64s. | int64 | long | int/long |
+| <a name="fixed32" /> fixed32 | Always four bytes. More efficient than uint32 if values are often greater than 2^28. | uint32 | int | int |
+| <a name="fixed64" /> fixed64 | Always eight bytes. More efficient than uint64 if values are often greater than 2^56. | uint64 | long | int/long |
+| <a name="sfixed32" /> sfixed32 | Always four bytes. | int32 | int | int |
+| <a name="sfixed64" /> sfixed64 | Always eight bytes. | int64 | long | int/long |
+| <a name="bool" /> bool |  | bool | boolean | boolean |
+| <a name="string" /> string | A string must always contain UTF-8 encoded or 7-bit ASCII text. | string | String | str/unicode |
+| <a name="bytes" /> bytes | May contain any arbitrary sequence of bytes. | string | ByteString | str |
+