forked from TrueCloudLab/frostfs-api
Update container package docs
Signed-off-by: Stanislav Bogatyrev <stanislav@nspcc.ru>
This commit is contained in:
parent
d2f984eb57
commit
f51ee1ba9d
2 changed files with 102 additions and 86 deletions
|
@ -10,46 +10,51 @@ import "container/types.proto";
|
||||||
import "refs/types.proto";
|
import "refs/types.proto";
|
||||||
import "session/types.proto";
|
import "session/types.proto";
|
||||||
|
|
||||||
// ContainerService provides API to access container smart-contract in morph chain
|
// `ContainerService` provides API to interact with `Container` smart contract
|
||||||
// via NeoFS node.
|
// in NeoFS sidechain via other NeoFS nodes. All of those actions can be done
|
||||||
|
// equivalently by directly issuing transactions and RPC calls to sidechain
|
||||||
|
// nodes.
|
||||||
service ContainerService {
|
service ContainerService {
|
||||||
// Put invokes 'Put' method in container smart-contract and returns
|
// `Put` invokes `Container` smart contract's `Put` method and returns
|
||||||
// response immediately. After new block in morph chain, request is verified
|
// response immediately. After a new block is issued in sidechain, request is
|
||||||
// by inner ring nodes. After one more block in morph chain, container
|
// verified by Inner Ring nodes. After one more block in sidechain, container
|
||||||
// added into smart-contract storage.
|
// is added into smart contract storage.
|
||||||
rpc Put(PutRequest) returns (PutResponse);
|
rpc Put(PutRequest) returns (PutResponse);
|
||||||
|
|
||||||
// Delete invokes 'Delete' method in container smart-contract and returns
|
// `Delete` invokes `Container` smart contract's `Delete` method and returns
|
||||||
// response immediately. After new block in morph chain, request is verified
|
// response immediately. After a new block is issued in sidechain, request is
|
||||||
// by inner ring nodes. After one more block in morph chain, container
|
// verified by Inner Ring nodes. After one more block in sidechain, container
|
||||||
// removed from smart-contract storage.
|
// is added into smart contract storage.
|
||||||
rpc Delete(DeleteRequest) returns (DeleteResponse);
|
rpc Delete(DeleteRequest) returns (DeleteResponse);
|
||||||
|
|
||||||
// Get returns container from container smart-contract storage.
|
// Returns container structure from `Container` smart contract storage.
|
||||||
rpc Get(GetRequest) returns (GetResponse);
|
rpc Get(GetRequest) returns (GetResponse);
|
||||||
|
|
||||||
// List returns all owner's containers from container smart-contract
|
// Returns all owner's containers from 'Container` smart contract' storage.
|
||||||
// storage.
|
|
||||||
rpc List(ListRequest) returns (ListResponse);
|
rpc List(ListRequest) returns (ListResponse);
|
||||||
|
|
||||||
// SetExtendedACL invokes 'SetEACL' method in container smart-contract and
|
// Invokes 'SetEACL' method of 'Container` smart contract and returns response
|
||||||
// returns response immediately. After new block in morph chain,
|
// immediately. After one more block in sidechain, Extended ACL changes are
|
||||||
// Extended ACL added into smart-contract storage.
|
// added into smart contract storage.
|
||||||
rpc SetExtendedACL(SetExtendedACLRequest) returns (SetExtendedACLResponse);
|
rpc SetExtendedACL(SetExtendedACLRequest) returns (SetExtendedACLResponse);
|
||||||
|
|
||||||
// GetExtendedACL returns Extended ACL table and signature from container
|
// Returns Extended ACL table and signature from `Container` smart contract
|
||||||
// smart-contract storage.
|
// storage.
|
||||||
rpc GetExtendedACL(GetExtendedACLRequest) returns (GetExtendedACLResponse);
|
rpc GetExtendedACL(GetExtendedACLRequest) returns (GetExtendedACLResponse);
|
||||||
}
|
}
|
||||||
|
|
||||||
// New NeoFS Container creation request
|
// New NeoFS Container creation request
|
||||||
message PutRequest {
|
message PutRequest {
|
||||||
// Request body
|
// Container creation request has container structure's signature as a
|
||||||
|
// separate field. It's not stored in sidechain, just verified on container
|
||||||
|
// creation by `Container` smart contract. `ContainerID` is a SHA256 hash of
|
||||||
|
// the stable-marshalled container strucutre, hence there is no need for
|
||||||
|
// additional signature checks.
|
||||||
message Body {
|
message Body {
|
||||||
// Container to create in NeoFS.
|
// Container structure to register in NeoFS
|
||||||
container.Container container = 1;
|
container.Container container = 1;
|
||||||
|
|
||||||
//Signature of stable-marshalled container according to RFC-6979.
|
// Signature of a stable-marshalled container according to RFC-6979
|
||||||
neo.fs.v2.refs.Signature signature =2;
|
neo.fs.v2.refs.Signature signature =2;
|
||||||
}
|
}
|
||||||
// Body of container put request message.
|
// Body of container put request message.
|
||||||
|
@ -60,16 +65,19 @@ message PutRequest {
|
||||||
neo.fs.v2.session.RequestMetaHeader meta_header = 2;
|
neo.fs.v2.session.RequestMetaHeader meta_header = 2;
|
||||||
|
|
||||||
// Carries request verification information. This header is used to
|
// Carries request verification information. This header is used to
|
||||||
// authenticate the nodes of the message route and check the correctness
|
// authenticate the nodes of the message route and check the correctness of
|
||||||
// of transmission.
|
// transmission.
|
||||||
neo.fs.v2.session.RequestVerificationHeader verify_header = 3;
|
neo.fs.v2.session.RequestVerificationHeader verify_header = 3;
|
||||||
}
|
}
|
||||||
|
|
||||||
// New NeoFS Container creation response
|
// New NeoFS Container creation response
|
||||||
message PutResponse {
|
message PutResponse {
|
||||||
// Response body
|
// Container put response body contains information about the newly registered
|
||||||
|
// container as seen by `Container` smart contract. `ContainerID` can be
|
||||||
|
// calculated beforehand from the container structure and compared to the one
|
||||||
|
// returned here to make sure everything was done as expected.
|
||||||
message Body {
|
message Body {
|
||||||
// container_id carries identifier of the new container.
|
// Unique identifier of the newly created container
|
||||||
neo.fs.v2.refs.ContainerID container_id = 1;
|
neo.fs.v2.refs.ContainerID container_id = 1;
|
||||||
}
|
}
|
||||||
// Body of container put response message.
|
// Body of container put response message.
|
||||||
|
@ -80,20 +88,21 @@ message PutResponse {
|
||||||
neo.fs.v2.session.ResponseMetaHeader meta_header = 2;
|
neo.fs.v2.session.ResponseMetaHeader meta_header = 2;
|
||||||
|
|
||||||
// Carries response verification information. This header is used to
|
// Carries response verification information. This header is used to
|
||||||
// authenticate the nodes of the message route and check the correctness
|
// authenticate the nodes of the message route and check the correctness of
|
||||||
// of transmission.
|
// transmission.
|
||||||
neo.fs.v2.session.ResponseVerificationHeader verify_header = 3;
|
neo.fs.v2.session.ResponseVerificationHeader verify_header = 3;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Container removal request
|
// Container removal request
|
||||||
message DeleteRequest {
|
message DeleteRequest {
|
||||||
// Request body
|
// Container removal request body has a signed `ContainerID` as a proof of
|
||||||
|
// container owner's intent. The signature will be verified by `Container`
|
||||||
|
// smart contract, so signing algorithm must be supported by NeoVM.
|
||||||
message Body {
|
message Body {
|
||||||
// container_id carries identifier of the container to delete
|
// Identifier of the container to delete from NeoFS
|
||||||
// from NeoFS.
|
|
||||||
neo.fs.v2.refs.ContainerID container_id = 1;
|
neo.fs.v2.refs.ContainerID container_id = 1;
|
||||||
|
|
||||||
// Signature of container id according to RFC-6979.
|
// `ContainerID` signed with the container owner's key according to RFC-6979
|
||||||
neo.fs.v2.refs.Signature signature = 2;
|
neo.fs.v2.refs.Signature signature = 2;
|
||||||
}
|
}
|
||||||
// Body of container delete request message.
|
// Body of container delete request message.
|
||||||
|
@ -104,15 +113,16 @@ message DeleteRequest {
|
||||||
neo.fs.v2.session.RequestMetaHeader meta_header = 2;
|
neo.fs.v2.session.RequestMetaHeader meta_header = 2;
|
||||||
|
|
||||||
// Carries request verification information. This header is used to
|
// Carries request verification information. This header is used to
|
||||||
// authenticate the nodes of the message route and check the correctness
|
// authenticate the nodes of the message route and check the correctness of
|
||||||
// of transmission.
|
// transmission.
|
||||||
neo.fs.v2.session.RequestVerificationHeader verify_header = 3;
|
neo.fs.v2.session.RequestVerificationHeader verify_header = 3;
|
||||||
}
|
}
|
||||||
|
|
||||||
// DeleteResponse is empty because delete operation is asynchronous and done
|
// `DeleteResponse` has an empty body because delete operation is asynchronous
|
||||||
// via consensus in inner ring nodes
|
// and done via consensus in Inner Ring nodes.
|
||||||
message DeleteResponse {
|
message DeleteResponse {
|
||||||
// Response body
|
// `DeleteResponse` has an empty body because delete operation is asynchronous
|
||||||
|
// and done via consensus in Inner Ring nodes.
|
||||||
message Body {}
|
message Body {}
|
||||||
// Body of container delete response message.
|
// Body of container delete response message.
|
||||||
Body body = 1;
|
Body body = 1;
|
||||||
|
@ -122,16 +132,16 @@ message DeleteResponse {
|
||||||
neo.fs.v2.session.ResponseMetaHeader meta_header = 2;
|
neo.fs.v2.session.ResponseMetaHeader meta_header = 2;
|
||||||
|
|
||||||
// Carries response verification information. This header is used to
|
// Carries response verification information. This header is used to
|
||||||
// authenticate the nodes of the message route and check the correctness
|
// authenticate the nodes of the message route and check the correctness of
|
||||||
// of transmission.
|
// transmission.
|
||||||
neo.fs.v2.session.ResponseVerificationHeader verify_header = 3;
|
neo.fs.v2.session.ResponseVerificationHeader verify_header = 3;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Get container structure
|
// Get container structure
|
||||||
message GetRequest {
|
message GetRequest {
|
||||||
// Request body
|
// Get container structure request body.
|
||||||
message Body {
|
message Body {
|
||||||
// container_id carries identifier of the container to get.
|
// Identifier of the container to get
|
||||||
neo.fs.v2.refs.ContainerID container_id = 1;
|
neo.fs.v2.refs.ContainerID container_id = 1;
|
||||||
}
|
}
|
||||||
// Body of container get request message.
|
// Body of container get request message.
|
||||||
|
@ -142,16 +152,17 @@ message GetRequest {
|
||||||
neo.fs.v2.session.RequestMetaHeader meta_header = 2;
|
neo.fs.v2.session.RequestMetaHeader meta_header = 2;
|
||||||
|
|
||||||
// Carries request verification information. This header is used to
|
// Carries request verification information. This header is used to
|
||||||
// authenticate the nodes of the message route and check the correctness
|
// authenticate the nodes of the message route and check the correctness of
|
||||||
// of transmission.
|
// transmission.
|
||||||
neo.fs.v2.session.RequestVerificationHeader verify_header = 3;
|
neo.fs.v2.session.RequestVerificationHeader verify_header = 3;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Get container structure
|
// Get container structure
|
||||||
message GetResponse {
|
message GetResponse {
|
||||||
// Response body
|
// Get container response body does not have container structure signature. It
|
||||||
|
// was already verified on container creation.
|
||||||
message Body {
|
message Body {
|
||||||
// Container that has been requested.
|
// Requested container structure
|
||||||
Container container = 1;
|
Container container = 1;
|
||||||
}
|
}
|
||||||
// Body of container get response message.
|
// Body of container get response message.
|
||||||
|
@ -162,19 +173,19 @@ message GetResponse {
|
||||||
neo.fs.v2.session.ResponseMetaHeader meta_header = 2;
|
neo.fs.v2.session.ResponseMetaHeader meta_header = 2;
|
||||||
|
|
||||||
// Carries response verification information. This header is used to
|
// Carries response verification information. This header is used to
|
||||||
// authenticate the nodes of the message route and check the correctness
|
// authenticate the nodes of the message route and check the correctness of
|
||||||
// of transmission.
|
// transmission.
|
||||||
neo.fs.v2.session.ResponseVerificationHeader verify_header = 3;
|
neo.fs.v2.session.ResponseVerificationHeader verify_header = 3;
|
||||||
}
|
}
|
||||||
|
|
||||||
// List containers
|
// List containers
|
||||||
message ListRequest {
|
message ListRequest {
|
||||||
// Request body
|
// List containers request body.
|
||||||
message Body {
|
message Body {
|
||||||
// owner_id carries identifier of the container owner.
|
// Identifier of the container owner
|
||||||
neo.fs.v2.refs.OwnerID owner_id = 1;
|
neo.fs.v2.refs.OwnerID owner_id = 1;
|
||||||
}
|
}
|
||||||
// Body of list containers request message.
|
// Body of list containers request message
|
||||||
Body body = 1;
|
Body body = 1;
|
||||||
|
|
||||||
// Carries request meta information. Header data is used only to regulate
|
// Carries request meta information. Header data is used only to regulate
|
||||||
|
@ -182,16 +193,16 @@ message ListRequest {
|
||||||
neo.fs.v2.session.RequestMetaHeader meta_header = 2;
|
neo.fs.v2.session.RequestMetaHeader meta_header = 2;
|
||||||
|
|
||||||
// Carries request verification information. This header is used to
|
// Carries request verification information. This header is used to
|
||||||
// authenticate the nodes of the message route and check the correctness
|
// authenticate the nodes of the message route and check the correctness of
|
||||||
// of transmission.
|
// transmission.
|
||||||
neo.fs.v2.session.RequestVerificationHeader verify_header = 3;
|
neo.fs.v2.session.RequestVerificationHeader verify_header = 3;
|
||||||
}
|
}
|
||||||
|
|
||||||
// List containers
|
// List containers
|
||||||
message ListResponse {
|
message ListResponse {
|
||||||
// Response body
|
// List containers response body.
|
||||||
message Body {
|
message Body {
|
||||||
// ContainerIDs carries list of identifiers of the containers that belong to the owner.
|
// List of `ContainerID`s belonging to the requested `OwnerID`
|
||||||
repeated refs.ContainerID container_ids = 1;
|
repeated refs.ContainerID container_ids = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -203,19 +214,20 @@ message ListResponse {
|
||||||
neo.fs.v2.session.ResponseMetaHeader meta_header = 2;
|
neo.fs.v2.session.ResponseMetaHeader meta_header = 2;
|
||||||
|
|
||||||
// Carries response verification information. This header is used to
|
// Carries response verification information. This header is used to
|
||||||
// authenticate the nodes of the message route and check the correctness
|
// authenticate the nodes of the message route and check the correctness of
|
||||||
// of transmission.
|
// transmission.
|
||||||
neo.fs.v2.session.ResponseVerificationHeader verify_header = 3;
|
neo.fs.v2.session.ResponseVerificationHeader verify_header = 3;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Set Extended ACL
|
// Set Extended ACL
|
||||||
message SetExtendedACLRequest {
|
message SetExtendedACLRequest {
|
||||||
// Request body
|
// Set Extended ACL request body does not have separate `ContainerID`
|
||||||
|
// reference. It will be taken from `EACLTable.container_id` field.
|
||||||
message Body {
|
message Body {
|
||||||
// Extended ACL to set for the container.
|
// Extended ACL table to set for container
|
||||||
neo.fs.v2.acl.EACLTable eacl = 1;
|
neo.fs.v2.acl.EACLTable eacl = 1;
|
||||||
|
|
||||||
// Signature of stable-marshalled Extended ACL according to RFC-6979.
|
// Signature of stable-marshalled Extended ACL table according to RFC-6979
|
||||||
neo.fs.v2.refs.Signature signature = 2;
|
neo.fs.v2.refs.Signature signature = 2;
|
||||||
}
|
}
|
||||||
// Body of set extended acl request message.
|
// Body of set extended acl request message.
|
||||||
|
@ -226,14 +238,16 @@ message SetExtendedACLRequest {
|
||||||
neo.fs.v2.session.RequestMetaHeader meta_header = 2;
|
neo.fs.v2.session.RequestMetaHeader meta_header = 2;
|
||||||
|
|
||||||
// Carries request verification information. This header is used to
|
// Carries request verification information. This header is used to
|
||||||
// authenticate the nodes of the message route and check the correctness
|
// authenticate the nodes of the message route and check the correctness of
|
||||||
// of transmission.
|
// transmission.
|
||||||
neo.fs.v2.session.RequestVerificationHeader verify_header = 3;
|
neo.fs.v2.session.RequestVerificationHeader verify_header = 3;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Set Extended ACL
|
// Set Extended ACL
|
||||||
message SetExtendedACLResponse {
|
message SetExtendedACLResponse {
|
||||||
// Response body
|
// `SetExtendedACLResponse` has an empty body because the operation is
|
||||||
|
// asynchronous and update should be reflected in `Container` smart contract's
|
||||||
|
// storage after next block is issued in sidechain.
|
||||||
message Body { }
|
message Body { }
|
||||||
|
|
||||||
// Body of set extended acl response message.
|
// Body of set extended acl response message.
|
||||||
|
@ -244,16 +258,16 @@ message SetExtendedACLResponse {
|
||||||
neo.fs.v2.session.ResponseMetaHeader meta_header = 2;
|
neo.fs.v2.session.ResponseMetaHeader meta_header = 2;
|
||||||
|
|
||||||
// Carries response verification information. This header is used to
|
// Carries response verification information. This header is used to
|
||||||
// authenticate the nodes of the message route and check the correctness
|
// authenticate the nodes of the message route and check the correctness of
|
||||||
// of transmission.
|
// transmission.
|
||||||
neo.fs.v2.session.ResponseVerificationHeader verify_header = 3;
|
neo.fs.v2.session.ResponseVerificationHeader verify_header = 3;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Get Extended ACL
|
// Get Extended ACL
|
||||||
message GetExtendedACLRequest {
|
message GetExtendedACLRequest {
|
||||||
// Request body
|
// Get Extended ACL request body
|
||||||
message Body {
|
message Body {
|
||||||
// container_id carries identifier of the container that has Extended ACL.
|
// Identifier of the container having Extended ACL
|
||||||
neo.fs.v2.refs.ContainerID container_id = 1;
|
neo.fs.v2.refs.ContainerID container_id = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -265,19 +279,21 @@ message GetExtendedACLRequest {
|
||||||
neo.fs.v2.session.RequestMetaHeader meta_header = 2;
|
neo.fs.v2.session.RequestMetaHeader meta_header = 2;
|
||||||
|
|
||||||
// Carries request verification information. This header is used to
|
// Carries request verification information. This header is used to
|
||||||
// authenticate the nodes of the message route and check the correctness
|
// authenticate the nodes of the message route and check the correctness of
|
||||||
// of transmission.
|
// transmission.
|
||||||
neo.fs.v2.session.RequestVerificationHeader verify_header = 3;
|
neo.fs.v2.session.RequestVerificationHeader verify_header = 3;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Get Extended ACL
|
// Get Extended ACL
|
||||||
message GetExtendedACLResponse {
|
message GetExtendedACLResponse {
|
||||||
// Response body
|
// Get Extended ACL Response body can be empty if the requested container did
|
||||||
|
// not have Extended ACL Table attached or Extended ACL was not allowed at
|
||||||
|
// container creation.
|
||||||
message Body {
|
message Body {
|
||||||
// Extended ACL that has been requested if it was set up.
|
// Extended ACL requested, if available
|
||||||
neo.fs.v2.acl.EACLTable eacl = 1;
|
neo.fs.v2.acl.EACLTable eacl = 1;
|
||||||
|
|
||||||
// Signature of stable-marshalled Extended ACL according to RFC-6979.
|
// Signature of stable-marshalled Extended ACL according to RFC-6979
|
||||||
neo.fs.v2.refs.Signature signature = 2;
|
neo.fs.v2.refs.Signature signature = 2;
|
||||||
}
|
}
|
||||||
// Body of get extended acl response message.
|
// Body of get extended acl response message.
|
||||||
|
@ -288,7 +304,7 @@ message GetExtendedACLResponse {
|
||||||
neo.fs.v2.session.ResponseMetaHeader meta_header = 2;
|
neo.fs.v2.session.ResponseMetaHeader meta_header = 2;
|
||||||
|
|
||||||
// Carries response verification information. This header is used to
|
// Carries response verification information. This header is used to
|
||||||
// authenticate the nodes of the message route and check the correctness
|
// authenticate the nodes of the message route and check the correctness of
|
||||||
// of transmission.
|
// transmission.
|
||||||
neo.fs.v2.session.ResponseVerificationHeader verify_header = 3;
|
neo.fs.v2.session.ResponseVerificationHeader verify_header = 3;
|
||||||
}
|
}
|
||||||
|
|
|
@ -8,23 +8,23 @@ option csharp_namespace = "NeoFS.API.v2.Container";
|
||||||
import "netmap/types.proto";
|
import "netmap/types.proto";
|
||||||
import "refs/types.proto";
|
import "refs/types.proto";
|
||||||
|
|
||||||
// Container is a structure that defines object placement behaviour. Objects
|
// Container is a structure that defines object placement behaviour. Objects can
|
||||||
// can be stored only within containers. They define placement rule, attributes
|
// be stored only within containers. They define placement rule, attributes and
|
||||||
// and access control information. ID of the container is a 32 byte long
|
// access control information. ID of the container is a 32 byte long SHA256 hash
|
||||||
// SHA256 hash of stable-marshalled container message.
|
// of stable-marshalled container message.
|
||||||
message Container {
|
message Container {
|
||||||
// Container format version.
|
// Container format version. Effectively the version of API library used to
|
||||||
// Effectively the version of API library used to create container
|
// create container.
|
||||||
neo.fs.v2.refs.Version version = 1;
|
neo.fs.v2.refs.Version version = 1;
|
||||||
|
|
||||||
// OwnerID carries identifier of the container owner.
|
// Identifier of the container owner
|
||||||
neo.fs.v2.refs.OwnerID owner_id = 2;
|
neo.fs.v2.refs.OwnerID owner_id = 2;
|
||||||
|
|
||||||
// Nonce is a 16 byte UUID, used to avoid collisions of container id.
|
// Nonce is a 16 byte UUID, used to avoid collisions of container id
|
||||||
bytes nonce = 3;
|
bytes nonce = 3;
|
||||||
|
|
||||||
// BasicACL contains access control rules for owner, system, others groups and
|
// `BasicACL` contains access control rules for owner, system, others groups
|
||||||
// permission bits for bearer token and Extended ACL.
|
// and permission bits for `BearerToken` and `Extended ACL`
|
||||||
uint32 basic_acl = 4;
|
uint32 basic_acl = 4;
|
||||||
|
|
||||||
// `Attribute` is a user-defined Key-Value metadata pair attached to the
|
// `Attribute` is a user-defined Key-Value metadata pair attached to the
|
||||||
|
@ -37,15 +37,15 @@ message Container {
|
||||||
// String ID of container's storage subnet. Container can be attached to
|
// String ID of container's storage subnet. Container can be attached to
|
||||||
// only one subnet.
|
// only one subnet.
|
||||||
message Attribute {
|
message Attribute {
|
||||||
// Key of immutable container attribute.
|
// Attribute name key
|
||||||
string key = 1;
|
string key = 1;
|
||||||
|
|
||||||
// Value of immutable container attribute.
|
// Attribute value
|
||||||
string value = 2;
|
string value = 2;
|
||||||
}
|
}
|
||||||
// Attributes define any immutable characteristics of container.
|
// Attributes represent immutable container's meta data
|
||||||
repeated Attribute attributes = 5;
|
repeated Attribute attributes = 5;
|
||||||
|
|
||||||
// Placement policy for the object inside the container.
|
// Placement policy for the object inside the container
|
||||||
neo.fs.v2.netmap.PlacementPolicy placement_policy = 6;
|
neo.fs.v2.netmap.PlacementPolicy placement_policy = 6;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue