From d404a1026e3fcac070a73319ab33964a371ad74f Mon Sep 17 00:00:00 2001 From: Pavel Pogodaev
DU8%B#qTBejIq&WgBNwOFkg^oRX^ zbv2RVRgeT(C_l>oPGwpB{wg0|iLxrDVI{uwd&e_4;n7&^k%4{n=}`PZd=Cb>dO7%` zf8Fn21O<87@4_6{|3F%N7Zl}rkU%OSUJ7rogWb@};$+w+{W(i5@k&yX$ak56pZae- zQkR*bTp=(tI2wtQDmYE#N&G)43Rz8ytQ6I`6eT^#BXKF>OhxbMG!ju>q(S9_PLYzD zr-dxVEUwNKU64u<#WER-iy(nX#CdQb>sP0778I}$w)gwdGz*1yDLS?4NahWN#{$J! zH6*jfB-qQB0*a@PJtevLvERGy_ju)BUMi{m_wWiiK4Fi5s3&0TJ=+(T+8M?$oTer^ z@&qVF8-hK6&W$XT$W&&PQqCe5fq1frf^g;0=Jxl-WhMUopVO;K4)x!3tW+G}6vzj) zV|<`b|CE&nvBxrkTXwwvFs-~43oiXPXVg)>uY7sSRu&&K)WB|2~<>Y^+b_J);v{Yb)BAFe+GX zN4p$aUE7kD$nfyWBDA*jWKjtG$qPO><(McGwR-+Rf$} wJ%+!>dm|Atsc-XILJEUysY!B}M2LPsb<`@3?m zF9sc1Z2(15H1$fvc6d(AneqE+%5sH4DaEq`foRc57HfOspbGXy{+T=%6xu9@cuF42 z;y%G!idh@A9fD3x>ePZNDK#IKj75Wo>u1nTCTE_Tfv8oitN&) zmFz%#H-qZ1J2N6p;SYXvHbH8F8!itV1hSSN8LoLnnIz{JR&Q(%vMINdn%hN_8#|}n z7rMM?=D~lR9KWBBow*BZ?SRo~hPI@%uCgBMQ3uJ4tffl<|5vUEzCf*)CPBSH4Z_sl z&!Bm&l$rl(8X+CxuQ(aMLLx1_;l7xu2v}(dEtK6lpH^b8DMM%G!aDzBFvIPFlGGby z( 9XKPt-z=jK-pUTGYE1GC{!cYqb&q z7T}TcyIz=QU^7868(&iW19@P_!EdI>`E7833d^w90;}b;y)}MI%a8>NV1R~D*APlw zl{T6dI*DCzqrA$drU11`6(NU22~BWZtLo8zOJ(L4N;C(8R { zF|CTz_CCm*doE|?9p$Hd@>X8RgzOz>{!xkQs^H5S7Ul=<;Ue&RZS=UNV9R0&Zi8}~ z$}#6loT7(kC!ga9e|G!|j}K)Mr`n6??s&o+^ix=fxqOyaTAH7;GKi!~;;7O*b{{f6 z$bJp7>UEyv8MEmSwjoAMk)}Ga!~f8*k%x0;)|Un`r-SijVO()=fG6zsy&m6)&16I` zd(onV-izw|ZLJz-bPUIfK^@T((x29$I-!;q&uaaVFH;2083PMkTX0d)(Q!D<2KOGO z`0lD{%{%^EHzB8JSkQs=&^!e{;eO^$gYSZ8zHVw7uhbrzIJHEy#V@jmqbt2ha|?bI zhe3rdRWHz6TsTBu(FxtG396~)t`kR7?fz$u%0?%vPE#(V`ewVH;jSAVa!+{r>h|b4 zwMVtntwr;KXj}&bmCmzHU-c0VeiEN$a!i>T#a)t=f*LIDA4`72>6=?m>NCYrL<^|X zL1}!3E;BCiuaX+EAOp{OJ8tc8>7x)dBv>0HNe;i4W7D;M)Lm|nT~YOhyh-gG@FrOR ztD2q1D0as&>SPqmL_8A&InaGdeKFre06owTRx-I_5s&ern8l^sZ?w=_Y+My;Q%er1 zXv<1wMkd>yZqP|P+FIC2!>6pam3pYN;k%Qy0R5o`CCBiFevJh1mB<}_gW8^sT?pU+ zov2X23p#o-tI`jSOucNmrQ TR7}Me#900PT_#z zJkoGqk$|DtHVD9%snh*j`t)SI_MdKZH-owDvHN_*ZqF5%pmuX}TXR46&59jg+!{PF z6!u!QZhilei48yr@)jxqDPivnksu}wID)-aVG$je3hu-nd6R`C*rnTsBAK^ 8K{GD}U?K+tkzx)|4B~JJ*%Vtf4LUMVG$#eWRXl z>CN3@cbY>dg&%#VF2%ft!j>AHfIn3wl(*gFYVT?l&1-W4)I}5?aIUT}Bv$G^fo=)w zS^(3r$hH3&kgmgPmkM9+uz|90gqn^HLfB>pXYTUdOKpuKxvtSVRNJ+d9?sosv*Wpx ztpm2D& Z6OGyOT4upC|i{=_H_3krIp759>C%r3O;#?I9&-&@;%uB(jw_0 zClJDf%&{v3HMJniN;&I@U5umdSlSB*&siz28%&EOz9r<>DOv9n^mXL00r!aaHt @AkCy>=Z*-8|sRisahD`tfWF*m*g@?>XfBxQ;fx*gKSAoCb-R)Hzi{ zJ~zXs9ac4R+#KPsEg0&926EgnAT-yk?s|IzeU{IJ>kyLYvls)X5ADxu<{&! zH~*{Zrmfek1RH0YV{i{qy2f#~j@2pKxSbo?EA?;%B6ceSJB`EJ;IkmYy~f3yvu%Qo zDdW&RvRi;m@o8WiCdn(9OlNw&r5ezfcAvJnwuJfM9>dQH Cz1MZT_mgNtr4_0Pf*|M3mstx3-0x<+X z+kLmzFo})NyXx7xrLlT>?A&hBxhZGtXiRi2YBV4wKdFZ`*6i4T(>r0VHhI}T4KBOW z9%J=@j^jbL8mE7(HKC)3omS6mvB8Lr+N0gN83`^NOy_HQpc}S7HQ7d;&m5Ue6br|6 zlo~YIx7+VL4(hnG9`qd%o`N}g9MtLl%8R3|u0b8`^$&Ye#~kurj_MfU9|{WaIIN>a zU{^pBw=%3_e9PXB4+)jn1dsrFzk4)-ej&l7C>5Rc2mpO`NLQvGrLqVk)4a3+N??P! zzO|;>5+ZRw=5>^Q>VmhWr25s4Rdr7Wc9g=})qP8G?r&jsr#iSke%8Xhx<_`L09iU2 zLcfsPgjd(lj>AjOf9PNHbx-Z6 =AJ5Ryitev?r>=im$p)Z-dxc6 z#TA{tnO=!xG@`Y(E;pqc^TXj2TWI~g_(8nfU8dcC>mS9@D{;i@&eT@vF-fINUHZIS zj`-4%({`OJErLHuQc3keF7=LClH{w(SWjuvQ>OI#qo;oJp`Y~l5b7_DSr%F>O(Dyr zFg6~JWre`qzV%U9g_ZNt*Za-laz7a)9Ld#f?g;-SDYL5bLLd5k1& *^^L@m5>VA(Lze*ZIpnS5SXcozt z|15#4((|*rDBfv*YqE2RkdLW9Dzy;{C)P(|%dF^}FltU{G3*#jid>f=S_VPIr<3 0R25pn z(&~k_Oy8~dl^c&n)rTc=N@RLHdOE{KKY4j`oEi0P3}9l)Qk?6%S-3zX{=U*(_uF8R z%yuD_rcX%^QCw Evp9aXz12PbR0c*&v$SM5^gla^m4KS1YBR zTY&m|+$gix%~2DxS|QsPGB1$pj?(tPvy *O#aB$;G>O7v}>w)3Bd1B~9rHK*hB@ zH^b6Yud%4E7Hqc8i_Bi-SsfW$=siK2Db_4nk&}5iQ z?P+{WrM^oy^hC@DNe`g&q7q7pX}LyPGP%QU{Cl*6X$Q@9!!@YJq4|Qzq;GI(Ua$)- zW_S!dC%uOaj$tnr7&D}Hhb?9w)eQaQQenZs)3X$XPKtuibPBzKCPAQ)XXI{8U1X$J zHz>F3a|97qw#3>Ph;XZ9Ap?@A2TAX{08UO&ngtDXZin z4HJ)e yyVou9|DKVl=)mH|3jBfD!v;H3|uJ3GRlNxkzTrVhjMKZuSbnU0xJ?I$Bj-AilDh zr~ sr^p{J$%+a~+K1)n=Jd(2@0G4QNr5%#9 z>wO^(n0hDLqno|yi_RVMX6cP~DxA3WYu+_(#DD$^2B|{Zls(vB@abU~2CYt4&$Qyv zMWs8Yg9tNgKSEc!^3cZ+=XELVJ$t3?bi0PC-`U-EdX5t8ecqa)PHdV8mf-^S4_-f~ zsi5`jI(6J>L5NzU9dM${1F_SRIXd(*G^)H@7 )c}4xB!2^!WZm`v literal 0 HcmV?d00001 diff --git a/api/accounting/grpc/types_frostfs.pb.go b/api/accounting/grpc/types_frostfs.pb.go new file mode 100644 index 0000000000000000000000000000000000000000..9d0e9ead422fda3f3b8c0da7fbeedd64cee3ffca GIT binary patch literal 4327 zcmcInTTk3Z6n?gU#l%t@FI{W`Q67+}lmM*~B?J{n9-69|wa4J#%W6E{U5fbc{mz;3 z>@I9*)U=43$KN^go%`7|6^CuE#HDJKmPX}bawDd?HEp(yv+c`vyU=ZCiY^gH@5Rac zb8&q1`aB84sm!kBr4lmB+P*Pma~XzZJ#DoSaTt8*+D1gG$=V!mQhJQSps07a!a%I;>#?5dw6SzD(c%FML7l<$<$W!9x01&J?}S}4r~26!#YO81N+*QFLh*tssv zKY=X<=+5Y@ZDt~J|3u*^49(3{i6fPjwX6hQ^qCR2Vemm#y%O|QHs-~1AwJX1`_YWf zqt9V*sv*;rh-(kT-tGXH4(9;fh2czUN`ODb;FIK(T>Vb!?n+jmF2sR|w;X{{+?3Vm zA^f7Ly}(8|kxN#UGIjd(b@-~%N8s>Lnu;?cCzU!Y|58FL)9a=a=1Pf<4%$L2;3iur zlfDo|t7~Zz$XQWxYOhr%=B2r^a jU_Gl-r IKcdF~q(R@ ;1tkkHnB8Na3}S5>F1q4u`m|lb6$}YH|+fdN2`m z7V%-*Aa()w@9oA9!h7d}>I-G?&f{VAa5((Yz cO zN11Gz)}ZQTE*hwCWs7?Mp^L^Ma;+;u9#Dw|9f~}=q6_)5lfLF4WwKM^*}g!LaG?Pi zl9#K@!LnN#j)`4{jUQCsP}Z_eJI+%M02!|~CYbd8PF z&}gxJmW#;M;hd7y?H$QSUO#RTv&GO7kFoHu^pA5Sj*G)QdDRza!1z;E`75;UE*_1= z+!onJ=Q<)@UE3S6wSMo$UI|?Co1C9vv1%^ksOhVUv(7SMzZa!O8;2R#c2pQR31SsL zy?qMSRCZK>7RW`o1{xD6xD%>EP4jDuL={UM(L4NL*TOYZt%`C%n-%yVUePDJLw$<& zv62m!JGvH3%Gd6T_&z%FZY-b*I%%I ZDV_}lr6TiV=*KpRC z3{YHi#vPntQ{K;yuO_`meDf&c)TK7*=Maw|in8X+=j}XsOFR1Q5pet5RJj8DZ9Zv# z)Ns420d12V%gCmt_D~iBX3H-6_BNaC-FZ8Gt7fVqGG&vz?y#9^UU87Id~kx4NA0?} zmP-lFR6sB^yS`RA?M0laI6J#8r~JH9CpbP*Rav{Lpb`6oZ*ZrG1mAM4ZXBD7rjv!T za}OJC%;iA0K bPqYNy!y6y|i?P)&`+i{};z zbMCJaZ1l@K=RJa!8H;r3!1}361i^G>y@yd2&n$?ldphH{%OO *e*-i& BKuiDt literal 0 HcmV?d00001 diff --git a/api/accounting/grpc/types_frostfs_fuzz.go b/api/accounting/grpc/types_frostfs_fuzz.go new file mode 100644 index 0000000..5eb5e97 --- /dev/null +++ b/api/accounting/grpc/types_frostfs_fuzz.go @@ -0,0 +1,26 @@ +//go:build gofuzz +// +build gofuzz + +// Code generated by protoc-gen-go-frostfs. DO NOT EDIT. + +package accounting + +func DoFuzzProtoDecimal(data []byte) int { + msg := new(Decimal) + if err := msg.UnmarshalProtobuf(data); err != nil { + return 0 + } + _ = msg.MarshalProtobuf(nil) + return 1 +} +func DoFuzzJSONDecimal(data []byte) int { + msg := new(Decimal) + if err := msg.UnmarshalJSON(data); err != nil { + return 0 + } + _, err := msg.MarshalJSON() + if err != nil { + panic(err) + } + return 1 +} diff --git a/api/accounting/grpc/types_frostfs_test.go b/api/accounting/grpc/types_frostfs_test.go new file mode 100644 index 0000000..404b75e --- /dev/null +++ b/api/accounting/grpc/types_frostfs_test.go @@ -0,0 +1,21 @@ +//go:build gofuzz +// +build gofuzz + +// Code generated by protoc-gen-go-frostfs. DO NOT EDIT. + +package accounting + +import ( + testing "testing" +) + +func FuzzProtoDecimal(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzProtoDecimal(data) + }) +} +func FuzzJSONDecimal(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzJSONDecimal(data) + }) +} diff --git a/api/accounting/json.go b/api/accounting/json.go new file mode 100644 index 0000000..c5ff977 --- /dev/null +++ b/api/accounting/json.go @@ -0,0 +1,14 @@ +package accounting + +import ( + accounting "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/accounting/grpc" + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/rpc/message" +) + +func (d *Decimal) MarshalJSON() ([]byte, error) { + return message.MarshalJSON(d) +} + +func (d *Decimal) UnmarshalJSON(data []byte) error { + return message.UnmarshalJSON(d, data, new(accounting.Decimal)) +} diff --git a/api/accounting/marshal.go b/api/accounting/marshal.go new file mode 100644 index 0000000..84771cc --- /dev/null +++ b/api/accounting/marshal.go @@ -0,0 +1,104 @@ +package accounting + +import ( + accounting "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/accounting/grpc" + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/rpc/message" + protoutil "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/util/proto" +) + +const ( + decimalValueField = 1 + decimalPrecisionField = 2 + + balanceReqBodyOwnerField = 1 + + balanceRespBodyDecimalField = 1 +) + +func (d *Decimal) StableMarshal(buf []byte) []byte { + if d == nil { + return []byte{} + } + + if buf == nil { + buf = make([]byte, d.StableSize()) + } + + var offset int + + offset += protoutil.Int64Marshal(decimalValueField, buf[offset:], d.val) + protoutil.UInt32Marshal(decimalPrecisionField, buf[offset:], d.prec) + + return buf +} + +func (d *Decimal) StableSize() (size int) { + if d == nil { + return 0 + } + + size += protoutil.Int64Size(decimalValueField, d.val) + size += protoutil.UInt32Size(decimalPrecisionField, d.prec) + + return size +} + +func (d *Decimal) Unmarshal(data []byte) error { + return message.Unmarshal(d, data, new(accounting.Decimal)) +} + +func (b *BalanceRequestBody) StableMarshal(buf []byte) []byte { + if b == nil { + return []byte{} + } + + if buf == nil { + buf = make([]byte, b.StableSize()) + } + + protoutil.NestedStructureMarshal(balanceReqBodyOwnerField, buf, b.ownerID) + + return buf +} + +func (b *BalanceRequestBody) StableSize() (size int) { + if b == nil { + return 0 + } + + size = protoutil.NestedStructureSize(balanceReqBodyOwnerField, b.ownerID) + + return size +} + +func (b *BalanceRequestBody) Unmarshal(data []byte) error { + return message.Unmarshal(b, data, new(accounting.BalanceRequest_Body)) +} + +func (br *BalanceResponseBody) StableMarshal(buf []byte) []byte { + if br == nil { + return []byte{} + } + + if buf == nil { + buf = make([]byte, br.StableSize()) + } + + protoutil.NestedStructureMarshal(balanceRespBodyDecimalField, buf, br.bal) + + return buf +} + +func (br *BalanceResponseBody) StableSize() (size int) { + if br == nil { + return 0 + } + + size = protoutil.NestedStructureSize(balanceRespBodyDecimalField, br.bal) + + return size +} + +func (br *BalanceResponseBody) Unmarshal(data []byte) error { + return message.Unmarshal(br, data, new(accounting.BalanceResponse_Body)) +} diff --git a/api/accounting/message_test.go b/api/accounting/message_test.go new file mode 100644 index 0000000..214c171 --- /dev/null +++ b/api/accounting/message_test.go @@ -0,0 +1,19 @@ +package accounting_test + +import ( + "testing" + + accountingtest "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/accounting/test" + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/rpc/message" + messagetest "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/rpc/message/test" +) + +func TestMessage(t *testing.T) { + messagetest.TestRPCMessage(t, + func(empty bool) message.Message { return accountingtest.GenerateDecimal(empty) }, + func(empty bool) message.Message { return accountingtest.GenerateBalanceRequestBody(empty) }, + func(empty bool) message.Message { return accountingtest.GenerateBalanceRequest(empty) }, + func(empty bool) message.Message { return accountingtest.GenerateBalanceResponseBody(empty) }, + func(empty bool) message.Message { return accountingtest.GenerateBalanceResponse(empty) }, + ) +} diff --git a/api/accounting/test/generate.go b/api/accounting/test/generate.go new file mode 100644 index 0000000..e9f45df --- /dev/null +++ b/api/accounting/test/generate.go @@ -0,0 +1,64 @@ +package accountingtest + +import ( + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/accounting" + accountingtest "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/refs/test" + sessiontest "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/session/test" +) + +func GenerateBalanceRequest(empty bool) *accounting.BalanceRequest { + m := new(accounting.BalanceRequest) + + if !empty { + m.SetBody(GenerateBalanceRequestBody(false)) + } + + m.SetMetaHeader(sessiontest.GenerateRequestMetaHeader(empty)) + m.SetVerificationHeader(sessiontest.GenerateRequestVerificationHeader(empty)) + + return m +} + +func GenerateBalanceRequestBody(empty bool) *accounting.BalanceRequestBody { + m := new(accounting.BalanceRequestBody) + + if !empty { + m.SetOwnerID(accountingtest.GenerateOwnerID(false)) + } + + return m +} + +func GenerateBalanceResponse(empty bool) *accounting.BalanceResponse { + m := new(accounting.BalanceResponse) + + if !empty { + m.SetBody(GenerateBalanceResponseBody(false)) + } + + m.SetMetaHeader(sessiontest.GenerateResponseMetaHeader(empty)) + m.SetVerificationHeader(sessiontest.GenerateResponseVerificationHeader(empty)) + + return m +} + +func GenerateBalanceResponseBody(empty bool) *accounting.BalanceResponseBody { + m := new(accounting.BalanceResponseBody) + + if !empty { + m.SetBalance(GenerateDecimal(false)) + } + + return m +} + +func GenerateDecimal(empty bool) *accounting.Decimal { + m := new(accounting.Decimal) + + if !empty { + m.SetValue(1) + m.SetPrecision(2) + } + + return m +} diff --git a/api/acl/bench_test.go b/api/acl/bench_test.go new file mode 100644 index 0000000..200270c --- /dev/null +++ b/api/acl/bench_test.go @@ -0,0 +1,51 @@ +package acl_test + +import ( + "testing" + + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/acl" + aclGrpc "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/acl/grpc" + acltest "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/acl/test" +) + +func BenchmarkTable_ToGRPCMessage(b *testing.B) { + const size = 4 + + tb := new(acl.Table) + rs := make([]acl.Record, size) + for i := range rs { + fs := make([]acl.HeaderFilter, size) + for j := range fs { + fs[j] = *acltest.GenerateFilter(false) + } + ts := make([]acl.Target, size) + for j := range ts { + ts[j] = *acltest.GenerateTarget(false) + } + + rs[i].SetFilters(fs) + rs[i].SetTargets(ts) + } + tb.SetRecords(rs) + + raw := tb.ToGRPCMessage() + + b.Run("to grpc message", func(b *testing.B) { + b.ReportAllocs() + for range b.N { + raw := tb.ToGRPCMessage() + if len(tb.GetRecords()) != len(raw.(*aclGrpc.EACLTable).Records) { + b.FailNow() + } + } + }) + b.Run("from grpc message", func(b *testing.B) { + b.ReportAllocs() + for range b.N { + tb := new(acl.Table) + if tb.FromGRPCMessage(raw) != nil { + b.FailNow() + } + } + }) +} diff --git a/api/acl/convert.go b/api/acl/convert.go new file mode 100644 index 0000000..3f389e3 --- /dev/null +++ b/api/acl/convert.go @@ -0,0 +1,592 @@ +package acl + +import ( + acl "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/acl/grpc" + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/ape" + apeGRPC "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/ape/grpc" + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/refs" + refsGRPC "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/refs/grpc" + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/rpc/grpc" + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/rpc/message" +) + +// RoleToGRPCField converts unified role enum into grpc enum. +func RoleToGRPCField(t Role) acl.Role { + switch t { + case RoleUser: + return acl.Role_USER + case RoleSystem: + return acl.Role_SYSTEM + case RoleOthers: + return acl.Role_OTHERS + default: + return acl.Role_ROLE_UNSPECIFIED + } +} + +// RoleFromGRPCField converts grpc enum into unified role enum. +func RoleFromGRPCField(t acl.Role) Role { + switch t { + case acl.Role_USER: + return RoleUser + case acl.Role_SYSTEM: + return RoleSystem + case acl.Role_OTHERS: + return RoleOthers + default: + return RoleUnknown + } +} + +// OperationToGRPCField converts unified operation enum into grpc enum. +func OperationToGRPCField(t Operation) acl.Operation { + switch t { + case OperationPut: + return acl.Operation_PUT + case OperationDelete: + return acl.Operation_DELETE + case OperationGet: + return acl.Operation_GET + case OperationHead: + return acl.Operation_HEAD + case OperationSearch: + return acl.Operation_SEARCH + case OperationRange: + return acl.Operation_GETRANGE + case OperationRangeHash: + return acl.Operation_GETRANGEHASH + default: + return acl.Operation_OPERATION_UNSPECIFIED + } +} + +// OperationFromGRPCField converts grpc enum into unified operation enum. +func OperationFromGRPCField(t acl.Operation) Operation { + switch t { + case acl.Operation_PUT: + return OperationPut + case acl.Operation_DELETE: + return OperationDelete + case acl.Operation_GET: + return OperationGet + case acl.Operation_HEAD: + return OperationHead + case acl.Operation_SEARCH: + return OperationSearch + case acl.Operation_GETRANGE: + return OperationRange + case acl.Operation_GETRANGEHASH: + return OperationRangeHash + default: + return OperationUnknown + } +} + +// ActionToGRPCField converts unified action enum into grpc enum. +func ActionToGRPCField(t Action) acl.Action { + switch t { + case ActionDeny: + return acl.Action_DENY + case ActionAllow: + return acl.Action_ALLOW + default: + return acl.Action_ACTION_UNSPECIFIED + } +} + +// ActionFromGRPCField converts grpc enum into unified action enum. +func ActionFromGRPCField(t acl.Action) Action { + switch t { + case acl.Action_DENY: + return ActionDeny + case acl.Action_ALLOW: + return ActionAllow + default: + return ActionUnknown + } +} + +// HeaderTypeToGRPCField converts unified header type enum into grpc enum. +func HeaderTypeToGRPCField(t HeaderType) acl.HeaderType { + switch t { + case HeaderTypeRequest: + return acl.HeaderType_REQUEST + case HeaderTypeObject: + return acl.HeaderType_OBJECT + case HeaderTypeService: + return acl.HeaderType_SERVICE + default: + return acl.HeaderType_HEADER_UNSPECIFIED + } +} + +// HeaderTypeFromGRPCField converts grpc enum into unified header type enum. +func HeaderTypeFromGRPCField(t acl.HeaderType) HeaderType { + switch t { + case acl.HeaderType_REQUEST: + return HeaderTypeRequest + case acl.HeaderType_OBJECT: + return HeaderTypeObject + case acl.HeaderType_SERVICE: + return HeaderTypeService + default: + return HeaderTypeUnknown + } +} + +// MatchTypeToGRPCField converts unified match type enum into grpc enum. +func MatchTypeToGRPCField(t MatchType) acl.MatchType { + switch t { + case MatchTypeStringEqual: + return acl.MatchType_STRING_EQUAL + case MatchTypeStringNotEqual: + return acl.MatchType_STRING_NOT_EQUAL + default: + return acl.MatchType_MATCH_TYPE_UNSPECIFIED + } +} + +// MatchTypeFromGRPCField converts grpc enum into unified match type enum. +func MatchTypeFromGRPCField(t acl.MatchType) MatchType { + switch t { + case acl.MatchType_STRING_EQUAL: + return MatchTypeStringEqual + case acl.MatchType_STRING_NOT_EQUAL: + return MatchTypeStringNotEqual + default: + return MatchTypeUnknown + } +} + +func (f *HeaderFilter) ToGRPCMessage() grpc.Message { + var m *acl.EACLRecord_Filter + + if f != nil { + m = new(acl.EACLRecord_Filter) + + m.SetKey(f.key) + m.SetValue(f.value) + m.SetHeaderType(HeaderTypeToGRPCField(f.hdrType)) + m.SetMatchType(MatchTypeToGRPCField(f.matchType)) + } + + return m +} + +func (f *HeaderFilter) FromGRPCMessage(m grpc.Message) error { + v, ok := m.(*acl.EACLRecord_Filter) + if !ok { + return message.NewUnexpectedMessageType(m, v) + } + + f.key = v.GetKey() + f.value = v.GetValue() + f.hdrType = HeaderTypeFromGRPCField(v.GetHeaderType()) + f.matchType = MatchTypeFromGRPCField(v.GetMatchType()) + + return nil +} + +func HeaderFiltersToGRPC(fs []HeaderFilter) (res []acl.EACLRecord_Filter) { + if fs != nil { + res = make([]acl.EACLRecord_Filter, 0, len(fs)) + + for i := range fs { + res = append(res, *fs[i].ToGRPCMessage().(*acl.EACLRecord_Filter)) + } + } + + return +} + +func HeaderFiltersFromGRPC(fs []acl.EACLRecord_Filter) (res []HeaderFilter, err error) { + if fs != nil { + res = make([]HeaderFilter, len(fs)) + + for i := range fs { + err = res[i].FromGRPCMessage(&fs[i]) + if err != nil { + return + } + } + } + + return +} + +func (t *Target) ToGRPCMessage() grpc.Message { + var m *acl.EACLRecord_Target + + if t != nil { + m = new(acl.EACLRecord_Target) + + m.SetRole(RoleToGRPCField(t.role)) + m.SetKeys(t.keys) + } + + return m +} + +func (t *Target) FromGRPCMessage(m grpc.Message) error { + v, ok := m.(*acl.EACLRecord_Target) + if !ok { + return message.NewUnexpectedMessageType(m, v) + } + + t.role = RoleFromGRPCField(v.GetRole()) + t.keys = v.GetKeys() + + return nil +} + +func TargetsToGRPC(ts []Target) (res []acl.EACLRecord_Target) { + if ts != nil { + res = make([]acl.EACLRecord_Target, 0, len(ts)) + + for i := range ts { + res = append(res, *ts[i].ToGRPCMessage().(*acl.EACLRecord_Target)) + } + } + + return +} + +func TargetsFromGRPC(fs []acl.EACLRecord_Target) (res []Target, err error) { + if fs != nil { + res = make([]Target, len(fs)) + + for i := range fs { + err = res[i].FromGRPCMessage(&fs[i]) + if err != nil { + return + } + } + } + + return +} + +func (r *Record) ToGRPCMessage() grpc.Message { + var m *acl.EACLRecord + + if r != nil { + m = new(acl.EACLRecord) + + m.SetOperation(OperationToGRPCField(r.op)) + m.SetAction(ActionToGRPCField(r.action)) + m.SetFilters(HeaderFiltersToGRPC(r.filters)) + m.SetTargets(TargetsToGRPC(r.targets)) + } + + return m +} + +func (r *Record) FromGRPCMessage(m grpc.Message) error { + v, ok := m.(*acl.EACLRecord) + if !ok { + return message.NewUnexpectedMessageType(m, v) + } + + var err error + + r.filters, err = HeaderFiltersFromGRPC(v.GetFilters()) + if err != nil { + return err + } + + r.targets, err = TargetsFromGRPC(v.GetTargets()) + if err != nil { + return err + } + + r.op = OperationFromGRPCField(v.GetOperation()) + r.action = ActionFromGRPCField(v.GetAction()) + + return nil +} + +func RecordsToGRPC(ts []Record) (res []acl.EACLRecord) { + if ts != nil { + res = make([]acl.EACLRecord, 0, len(ts)) + + for i := range ts { + res = append(res, *ts[i].ToGRPCMessage().(*acl.EACLRecord)) + } + } + + return +} + +func RecordsFromGRPC(fs []acl.EACLRecord) (res []Record, err error) { + if fs != nil { + res = make([]Record, len(fs)) + + for i := range fs { + err = res[i].FromGRPCMessage(&fs[i]) + if err != nil { + return + } + } + } + + return +} + +func (t *Table) ToGRPCMessage() grpc.Message { + var m *acl.EACLTable + + if t != nil { + m = new(acl.EACLTable) + + m.SetVersion(t.version.ToGRPCMessage().(*refsGRPC.Version)) + m.SetContainerId(t.cid.ToGRPCMessage().(*refsGRPC.ContainerID)) + m.SetRecords(RecordsToGRPC(t.records)) + } + + return m +} + +func (t *Table) FromGRPCMessage(m grpc.Message) error { + v, ok := m.(*acl.EACLTable) + if !ok { + return message.NewUnexpectedMessageType(m, v) + } + + var err error + + cid := v.GetContainerId() + if cid == nil { + t.cid = nil + } else { + if t.cid == nil { + t.cid = new(refs.ContainerID) + } + + err = t.cid.FromGRPCMessage(cid) + if err != nil { + return err + } + } + + version := v.GetVersion() + if version == nil { + t.version = nil + } else { + if t.version == nil { + t.version = new(refs.Version) + } + + err = t.version.FromGRPCMessage(version) + if err != nil { + return err + } + } + + t.records, err = RecordsFromGRPC(v.GetRecords()) + + return err +} + +func (l *TokenLifetime) ToGRPCMessage() grpc.Message { + var m *acl.BearerToken_Body_TokenLifetime + + if l != nil { + m = new(acl.BearerToken_Body_TokenLifetime) + + m.SetExp(l.exp) + m.SetIat(l.iat) + m.SetNbf(l.nbf) + } + + return m +} + +func (l *TokenLifetime) FromGRPCMessage(m grpc.Message) error { + v, ok := m.(*acl.BearerToken_Body_TokenLifetime) + if !ok { + return message.NewUnexpectedMessageType(m, v) + } + + l.exp = v.GetExp() + l.iat = v.GetIat() + l.nbf = v.GetNbf() + + return nil +} + +func (c *APEOverride) ToGRPCMessage() grpc.Message { + var m *acl.BearerToken_Body_APEOverride + + if c != nil { + m = new(acl.BearerToken_Body_APEOverride) + + m.SetTarget(c.target.ToGRPCMessage().(*apeGRPC.ChainTarget)) + + if len(c.chains) > 0 { + apeChains := make([]apeGRPC.Chain, len(c.chains)) + for i := range c.chains { + apeChains[i] = *c.chains[i].ToGRPCMessage().(*apeGRPC.Chain) + } + m.SetChains(apeChains) + } + } + + return m +} + +func (c *APEOverride) FromGRPCMessage(m grpc.Message) error { + v, ok := m.(*acl.BearerToken_Body_APEOverride) + if !ok { + return message.NewUnexpectedMessageType(m, v) + } + + if targetGRPC := v.GetTarget(); targetGRPC != nil { + if c.target == nil { + c.target = new(ape.ChainTarget) + } + if err := c.target.FromGRPCMessage(v.GetTarget()); err != nil { + return err + } + } + + if apeChains := v.GetChains(); len(apeChains) > 0 { + c.chains = make([]*ape.Chain, len(apeChains)) + for i := range apeChains { + c.chains[i] = new(ape.Chain) + if err := c.chains[i].FromGRPCMessage(&apeChains[i]); err != nil { + return err + } + } + } + + return nil +} + +func (bt *BearerTokenBody) ToGRPCMessage() grpc.Message { + var m *acl.BearerToken_Body + + if bt != nil { + m = new(acl.BearerToken_Body) + + m.SetOwnerId(bt.ownerID.ToGRPCMessage().(*refsGRPC.OwnerID)) + m.SetLifetime(bt.lifetime.ToGRPCMessage().(*acl.BearerToken_Body_TokenLifetime)) + m.SetEaclTable(bt.eacl.ToGRPCMessage().(*acl.EACLTable)) + m.SetAllowImpersonate(bt.impersonate) + m.SetApeOverride(bt.apeOverride.ToGRPCMessage().(*acl.BearerToken_Body_APEOverride)) + } + + return m +} + +func (bt *BearerTokenBody) FromGRPCMessage(m grpc.Message) error { + v, ok := m.(*acl.BearerToken_Body) + if !ok { + return message.NewUnexpectedMessageType(m, v) + } + + var err error + + ownerID := v.GetOwnerId() + if ownerID == nil { + bt.ownerID = nil + } else { + if bt.ownerID == nil { + bt.ownerID = new(refs.OwnerID) + } + + err = bt.ownerID.FromGRPCMessage(ownerID) + if err != nil { + return err + } + } + + lifetime := v.GetLifetime() + if lifetime == nil { + bt.lifetime = nil + } else { + if bt.lifetime == nil { + bt.lifetime = new(TokenLifetime) + } + + err = bt.lifetime.FromGRPCMessage(lifetime) + if err != nil { + return err + } + } + + eacl := v.GetEaclTable() + if eacl == nil { + bt.eacl = nil + } else { + if bt.eacl == nil { + bt.eacl = new(Table) + } + + if err = bt.eacl.FromGRPCMessage(eacl); err != nil { + return err + } + } + + if apeOverrideGRPC := v.GetApeOverride(); apeOverrideGRPC != nil { + if bt.apeOverride == nil { + bt.apeOverride = new(APEOverride) + } + err = bt.apeOverride.FromGRPCMessage(apeOverrideGRPC) + if err != nil { + return err + } + } + + bt.impersonate = v.GetAllowImpersonate() + + return err +} + +func (bt *BearerToken) ToGRPCMessage() grpc.Message { + var m *acl.BearerToken + + if bt != nil { + m = new(acl.BearerToken) + + m.SetBody(bt.body.ToGRPCMessage().(*acl.BearerToken_Body)) + m.SetSignature(bt.sig.ToGRPCMessage().(*refsGRPC.Signature)) + } + + return m +} + +func (bt *BearerToken) FromGRPCMessage(m grpc.Message) error { + v, ok := m.(*acl.BearerToken) + if !ok { + return message.NewUnexpectedMessageType(m, v) + } + + var err error + + body := v.GetBody() + if body == nil { + bt.body = nil + } else { + if bt.body == nil { + bt.body = new(BearerTokenBody) + } + + err = bt.body.FromGRPCMessage(body) + if err != nil { + return err + } + } + + sig := v.GetSignature() + if sig == nil { + bt.sig = nil + } else { + if bt.sig == nil { + bt.sig = new(refs.Signature) + } + + err = bt.sig.FromGRPCMessage(sig) + } + + return err +} diff --git a/api/acl/filters.go b/api/acl/filters.go new file mode 100644 index 0000000..c1d8afe --- /dev/null +++ b/api/acl/filters.go @@ -0,0 +1,33 @@ +package acl + +// ObjectFilterPrefix is a prefix of key to object header value or property. +const ObjectFilterPrefix = "$Object:" + +const ( + // FilterObjectVersion is a filter key to "version" field of the object header. + FilterObjectVersion = ObjectFilterPrefix + "version" + + // FilterObjectID is a filter key to "object_id" field of the object. + FilterObjectID = ObjectFilterPrefix + "objectID" + + // FilterObjectContainerID is a filter key to "container_id" field of the object header. + FilterObjectContainerID = ObjectFilterPrefix + "containerID" + + // FilterObjectOwnerID is a filter key to "owner_id" field of the object header. + FilterObjectOwnerID = ObjectFilterPrefix + "ownerID" + + // FilterObjectCreationEpoch is a filter key to "creation_epoch" field of the object header. + FilterObjectCreationEpoch = ObjectFilterPrefix + "creationEpoch" + + // FilterObjectPayloadLength is a filter key to "payload_length" field of the object header. + FilterObjectPayloadLength = ObjectFilterPrefix + "payloadLength" + + // FilterObjectPayloadHash is a filter key to "payload_hash" field of the object header. + FilterObjectPayloadHash = ObjectFilterPrefix + "payloadHash" + + // FilterObjectType is a filter key to "object_type" field of the object header. + FilterObjectType = ObjectFilterPrefix + "objectType" + + // FilterObjectHomomorphicHash is a filter key to "homomorphic_hash" field of the object header. + FilterObjectHomomorphicHash = ObjectFilterPrefix + "homomorphicHash" +) diff --git a/api/acl/grpc/types_frostfs.pb.go b/api/acl/grpc/types_frostfs.pb.go new file mode 100644 index 0000000000000000000000000000000000000000..76e460dc0c7c890feb390adc3320f3c21fbfff80 GIT binary patch literal 48052 zcmeHQ`)?b!vi{lrE4BuDD(AX#UQK}< {D_+8cv3-?~(TX zyW#hp(QrKJjO)q%%jEFoNz&LqIH_-Lo;O<`n(xx2+3Ia>cKheU(IlyDZu~hO4w9{O z&>FV8gLk|1xV5>_=}(d^{DWWLjm}&5lC5{$N!``l9dw4fC!=ZltT&vtUo_8lHRt |LUY>7?7+rLy3HN9oda&=8{%H&9 lo4=-$QMWbTH7b<&vzK0^BWA#n_nY0`Xv$PHfxALN zv>!*^NfoisfO %-$$jb{fx9W?fZ^ht7ebA$6=A2*JY@Q15 OT!f{pRFgFsY$`j~=4jPIB+=_U316{tgZMcJk9`*jFCL%ENYYM$5H`6CS@e zW6#5#3rrf-RGV#kmL^axAy=JdZ=9lSk;Y$}lh*qal1m=@KqM{a^w+(UXU|Vh{&*Ej zrxBo4h^RfD9-kZ?9R7UT_}|xiFM^3O0? |aD)&5{JAm4AiXq+^Bz4Dt?huR;8 y2Tq#f~<4qz93t+E9y=`W g)ODG>JGe`MY{u0V(uT`jd zB>8mjSx)C@0p2$q)r%J|e-C9PNsJ1+rBdxT4*!UWJXN?yDtUz=m*Wlfp}4GWLa<{nVclI^-24cq8ALn>a8`$!@~AjIA}SGO){4d zx94fIosJ5PHlM}&QR67u>U^&3b^iJE2-aHTINFkYu55Mw`Sj(}UmDLM;>735PUoLb zaVq%j;90b}_*_tj=MSGzwOf!Djr(W9tDQ0N6?XZNHVoxQ1v+T4RHzHub3*Ofi-cE% zTAPk?Sg1FQjVLd7GV$zH%Dg9ClfLy0ETM3Hp*b4Y6D+X?8hg)P9Hp({sD1iVw@2r4 zoVKQ|37wRDg`rgbmyYv~x8B?0+4^sA2uTvNI+H|6`|fN+l>bgYImN4|LwZ!o59ueO z`HfGsOqQo%;Q|#-UV7F(O&oYpe?^$cug%f;eY1zYBv=)-JGnk?*9P4lj&|9CuLpfA zT2YV&p&mBG+|oab6bu_$s36WPySq47H_v+MarghyM2^PeWb!^u#*{GZBp0yR1$aA~ zc9IUCmg^`dMQK*2qja2n>`vZGIf5rT-L%(Ecu>d5`{pe5S!dlrbM%R;uiKHSU^D~1 zaOSTuJ2*XT>uDRO=_d&WJUrjyoC_O~DgWn*@LX>UrhV#L?cR *RFguiqiFJCzEkQT{ z>J4hTQ`>4a2ZP}Rs<7Ek2B<=H8I<6+ z)sVuDYvj__N5rJ1f=By3W)U_bSj4(0U{06CmyaQ;kNCnfIHI9)BLzGI zZ1753CnMZI#0sZ)W*~U@qgx5&8mqZXZ$gU)bH5!30?8`m?jl?tT5TviVRLsY8as{} zn+1*W5)Z3Ihh`PEfpsqu+g?0M8MhKk@@?g6GvAVOZF!3^4cW;$Es6)3W!g{X)>g%U zV0fwSnK>gcgS=xb#up3HVw|a%FUOnIpW)6)FHLq}@%(c9@-VyKWQ^ejHyJfEs6sLu z#jfF9K{u9 L!H4gW&NE}SXSfS-~}|KlU{h^~N-^{3Mg3?Ka0?X{o6WE$7t zC;BM6M#VfSOIGo4I!W%t92D9ppsOD>KdMsO8ccgV)63v{PDf+2+ti62Qz5P!Ar z{`wC5LnL@N5kh*}h=_KGXybE|!tKPn)rL@F2W3x`>}=u*J%^vVd*R$~Pm- 8LuokV0&sZ0t zq0DNXtD}tK5~eU~8nP?Q4rUXWmg0yN6~{o-Ur;2(U;ChY%vK01ENj;^%RffwimI%U z=q8-m$}NI?EiiTONMMDuE6QnnGCY5g&eEQ& z)pxE!t!Q|PnHym&?1 otl|k*;gIV-AYo7SY!^ znv_??n^2X4i;#opKp?u67(K%w=mYb_{U+9-sOby^Wsa6HF0=Y8>Is;vA?x1Vo#erN z8g~4Eo9uCmQU~?DcAF1dbmCK5722q+N(Oo0JH25OVH-LE^UFv>SU33Tz(5KpIW63k zklmp~qL$e;O`o4!)7o4OO?XhXw;efmMN?}{DrQ?cvqoIInG>7muX0|4BGuDs#Q9mp z3eL}`tSFu?k`q!58hFHM&cMzH2rscqR3;iAkT;S!(^;y4kcK3$`vho;)hh@>?aHVQ zN3;ca#YuDYE}aBeJcpq1KZF-6Ts}gKl49y(>`ICZdMtf>S(sd!zXVL4EtokO7}lMG z$qNhGhR0C87S7=7QNf!DlPimBgvm)T@`(3QdY)1|E+2vz_y&&2t744BsKy6j>hTHB zVSd?D-X$?@*ZZ#l({_DX52nqtSbzn~VJYy6xao$rA%p8r+T=&g0V1hocG=i5FkH^S zHuZ1*e2dv%KbGvq$`-MctJ{txOUL+pPg_7WmQ2${I}#lp)_@{Y^%3q!3D^(mzZj4e z%=q8wr*Q#03YAa}?7_=}+EjhS4kdt{CVOVYQ8M1)vk~NwEI0g7Wi%fZb H+7TkI-xAkIjS|;{S-AybTkiE?RykH|wf>D+3Cgt;W<`1-e`ckHPI8&U zsVrkfFa~Rs1+XciyP@Dk#9Ae8A=JF+MP+OhBT+;L^B@B;d>|7jhb_wS+~%v-<)EG* z)guR@>HvRb;Z$f7HtsR!3V8%(+r|dhX{HuE?Ia!PMGmSUea6h_nc+!7^qk@uVeKV^ zM2w6`c#-C5NvW|8UK;KQa1IL_PyBBlHBe|{=w4DG@I%6WiN^nyE79c0aw3-~$1y10 zybS;s4Sv)HIvff~@X3ZKvK+l6J0NDlaLr}mmgR94Ut&>!B~!w7i22Bhmbj6S3We~x zP)PC}b6{Y1H6e#4cX>d?A{2voFcCVU%~>W+)X4R~99pR3VLApb1GG|NG6|b`J!rj8 zTOZ(YL8#b2E)bYR(YR!xplYnMT XTkTf_22m^s$L1Oqa0??Fx0Ch2 zKVI;>5o;h$4g>PkA|GV@s?%OYa&9e!(pw~Ap#X;xnTZ>9QI{G%T6;jNp{Tx4WJ4LO zsPCcFmvQns5h{l%Lq&)2DG9#aPK?%7ean%mB_Jx3Cffhkgi~1?{ w+X?`=5EIv2G(_0PNI7zCKRA{fMYs z=ShZs5Ra{dXR2yg`!H%d(J(r-6bz##MZ+j** q6Yl_g`TvN)k(o`_50-0r6l1rJ^l8ZC#n#wRO$;!0SDATJ9 zD>c)SF|X^5CpBIS^!kb*>Y+P)S75ghPHhXH_8@F79;~*UTYy+M8AVv8-=bJ@otETV z(BqDrLMyr)VcT A`BO1_yR$*ruFl Tx;f{kFLz|HnN+7!F@bmGEb4#C)kE)R(N6#x z1bSDxdRZ@(vmmM*RXp-rx;vK+j_jW$gA)*Z(cnlu>l_?uf#R&L!Hai7zL>y?_Zu7+ zbcG!K+@T9s>tTxe*k(dRJ~o6yPnuoa9(&MMdY`iId~CE!3DG{4kxNU+FBg0AIKh<} z@y#a7`{lSp-end?z8V|`-;jPA-Bv)(x^e0Baa9phW=KGQy|)_Z1x5e9NuVufev zo_eZqO1z+#GE55vH%~+v74_Yp=lKeS4MC-%mgwA;84SI@B;!fToXwnj>dQS}IwPid zT+O%E@x>itaGwm3>d f_!7^C;umo*ILldNj!{d|pb3^3w+G}=LiQiAx@Y;t++lhw9X_LTVrAeW~D(k6I z67^K+)6Mc!sXl_EN*4$G7O??|-I|njsO;78WvisZW1(%pQ{~#2#cimo!Y|KU@V{p> zkUC@4r|p+ISUg>OVU~k3cojf~#;5Ho>prE8 p2Vr2ZU(5RnGcn35FmPL%b0GZnWA!bw?e;5q4OY*!@jDU=v4YnK z#Gm7LaZ7Aj--4FAalQpzQ)UTWgKHtnf-UU7JqwrcET{^}19j2HPj=P@;oF6_aw*CUnUC)1kG9l_p1iz)}oPB z9m}9{UwCIR?pcO@$=_1^nl(-+6)=PuiX^o&mp+gXM^D;lE`1;^Yr6HJYB)_}+-iKW zyq8OWj*tf0uQPz7N(0*w5Q57`ilh1nEh6EIZ^RX#4??z1Hc)-U2FY~Hh5%`h-F32o z>LWI2sTdnrKloD|IbA;89Hxws1$3f@Lm4u~Gwxf8!l$d+0Z>Kl%(NXvPGnLeXs4 m- z!AdXAmqA&!_!0^NUhJA-QfoL$;4C>)_Z_M7T1-P8Y^+)NU}Yv=mFtA;5=29qe@*eK zgR{<6^%RT!I(yn}POhq_&=RFRjUc*J#@SXaS+lvW7$94iFH M`SH-d#8^~R;j~=pNO8SYs(m{L#8bdn; zH^qus#oR!-;pd!R2@xjc6IBA~o#Si2_>p5$4i|&k`uch*)=&lkgxdQ0dMaj=^^_z5 z;u1+b(mZsz>@6ZWeY5wf@e-KwsEbe1dxTHoBgJESum0>kA_+A~!x1O^HU~j_oJseo z@@M9SHi{~2QR(G^3d^{>G}IUL)Yq{j&^Nyh+dwK|zI+`w8RQGOT@l1LH)c^p44N|+ zeW*f4CSME%7S!c~4*~t)&21@rHL-`Kf_r;!^C<7us^LG?G{L6nZojCA|8wz)W#SXa zLq%;Gow$~0!twz%UGjASfPpGm0{# XjRKa{ zMk%r_kQEn$#}W-_XiW%W#LSJu4~w|71OHcq9!~hwPysfXCe>*Q2LehL%W2KXUyEh2 zg3Ac0niEVW5>HA3r?|!6;g@1JDwM2TN!pfo-QZcYj*Kf4pueHB2uN7$Rd!jkd88%3 z8 pKW!sqnAo6yqtjIhM!9K#Z2f<$1EDxD1`nGOR!#cyg zR7qTWv{!)sG@7j*M{T;-D)~dzQHslnke<|)BV6d^M|nflC?!BqlJ_mu|Ke_|<#PPe zs)e38qKYUm7me1_kTqgsuh$!XJm}*V9zeVJyi0P1Z+^S_ivCwNnwfH3*LiAK85hm- z6KEh%xVn5fRCD>5HxFBtzhax!*1p_wa95_BDby;6OUvD0SZh@`;ugS98ldEs0#sWi zVkIx&>VjFQ00$|#<&i7#6e7+Mbo@Q^> bhtnn7zyo8u)HE|!0S~w`5;e*GxoR55n_^Tg! zCM&1YN>Y5!`Fu2-3~~8&+G|s^e0qW4FhV~GYC#R#$-k1jJbrkP4_v}w*s33bkP9&p zOec(5)=#O+J-nQ>)f^0llLQ~nY~#g(i%HU S$5=-;H)Ao7% FNn*@rvfX&rH{XttXqu$(^H@ks(??#RECM$-Z+VxE@cDRM{ zWC^-;wgLgzrqxP;YtwK`wMBuq))pXZYs-b;x=f61DOxQzFN;~?P 7nAEnEj2H7%U zkN898I1G2DgH}?z$g|8YkZcX;8?~w($mTBA5x>j}Cn{LurPCCw_po1?tE#gTA-xJ% zD4MK*1=mwwLOZ$oFl#$=F^qd1cS7d}tH1~FwOj^}I)zx_X$TK?1;c@T2>Nc#E|H;t za+RE_w!F5B0D6l2RWj;S-?eet%;1X^9CLU)9er`Q&6K 0_b#@Fng3OEUTA5dq#^dcSAU(mm-?z{ezaIh@lxi}c(170J%+r#( EIZ5+`3Y~yJkAIKs@6&iwqMvp&q@d_4xcfrO;_l}Xj=w%H;55D0R_S0UsU;7So z!>!;eePrh%>-AvV?4*nz95(xDjdyaY6?x)d>J10n49?A5Rlsb-hA*+y=N(OR<~>5* z>;ymF8|95rN%jVMFcZ`Tj;nTx89Qyp5v6*tpk4A7IV&}4I?Rw3%5Ho0fLyL^r`rk| z&+eP?DKuiIoOBTeo$AbwPFAPoIVIc8*nw#H^C!H2S;bb+fY(C#y7M+~U`AP^1n%U< zARt_11Om%1 JMBOS3i4k~C$&BZU_bY}6KkYjS2Rg&X?oR% zszKXN5@gC37i)O-V}S>9+ZG*FSN(az3+P0w@5miPDVa`WKOz|UbZj=`YAm^;t}60w!$6{tn)a4wj1r z@6rNU;G|3&?}m%qEi?=&u4>dBAcK~e@E5QUZvhblaTf2t70uh)fjT!H$PP0^O7SJ6 VA;}|Kf~R@7jkQ6HozW#={}0mjby5HT literal 0 HcmV?d00001 diff --git a/api/acl/grpc/types_frostfs_fuzz.go b/api/acl/grpc/types_frostfs_fuzz.go new file mode 100644 index 0000000..5d5b763 --- /dev/null +++ b/api/acl/grpc/types_frostfs_fuzz.go @@ -0,0 +1,64 @@ +//go:build gofuzz +// +build gofuzz + +// Code generated by protoc-gen-go-frostfs. DO NOT EDIT. + +package acl + +func DoFuzzProtoEACLRecord(data []byte) int { + msg := new(EACLRecord) + if err := msg.UnmarshalProtobuf(data); err != nil { + return 0 + } + _ = msg.MarshalProtobuf(nil) + return 1 +} +func DoFuzzJSONEACLRecord(data []byte) int { + msg := new(EACLRecord) + if err := msg.UnmarshalJSON(data); err != nil { + return 0 + } + _, err := msg.MarshalJSON() + if err != nil { + panic(err) + } + return 1 +} +func DoFuzzProtoEACLTable(data []byte) int { + msg := new(EACLTable) + if err := msg.UnmarshalProtobuf(data); err != nil { + return 0 + } + _ = msg.MarshalProtobuf(nil) + return 1 +} +func DoFuzzJSONEACLTable(data []byte) int { + msg := new(EACLTable) + if err := msg.UnmarshalJSON(data); err != nil { + return 0 + } + _, err := msg.MarshalJSON() + if err != nil { + panic(err) + } + return 1 +} +func DoFuzzProtoBearerToken(data []byte) int { + msg := new(BearerToken) + if err := msg.UnmarshalProtobuf(data); err != nil { + return 0 + } + _ = msg.MarshalProtobuf(nil) + return 1 +} +func DoFuzzJSONBearerToken(data []byte) int { + msg := new(BearerToken) + if err := msg.UnmarshalJSON(data); err != nil { + return 0 + } + _, err := msg.MarshalJSON() + if err != nil { + panic(err) + } + return 1 +} diff --git a/api/acl/grpc/types_frostfs_test.go b/api/acl/grpc/types_frostfs_test.go new file mode 100644 index 0000000..c6d1c43 --- /dev/null +++ b/api/acl/grpc/types_frostfs_test.go @@ -0,0 +1,41 @@ +//go:build gofuzz +// +build gofuzz + +// Code generated by protoc-gen-go-frostfs. DO NOT EDIT. + +package acl + +import ( + testing "testing" +) + +func FuzzProtoEACLRecord(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzProtoEACLRecord(data) + }) +} +func FuzzJSONEACLRecord(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzJSONEACLRecord(data) + }) +} +func FuzzProtoEACLTable(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzProtoEACLTable(data) + }) +} +func FuzzJSONEACLTable(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzJSONEACLTable(data) + }) +} +func FuzzProtoBearerToken(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzProtoBearerToken(data) + }) +} +func FuzzJSONBearerToken(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzJSONBearerToken(data) + }) +} diff --git a/api/acl/json.go b/api/acl/json.go new file mode 100644 index 0000000..9192956 --- /dev/null +++ b/api/acl/json.go @@ -0,0 +1,70 @@ +package acl + +import ( + acl "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/acl/grpc" + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/rpc/message" +) + +func (f *HeaderFilter) MarshalJSON() ([]byte, error) { + return message.MarshalJSON(f) +} + +func (f *HeaderFilter) UnmarshalJSON(data []byte) error { + return message.UnmarshalJSON(f, data, new(acl.EACLRecord_Filter)) +} + +func (t *Target) MarshalJSON() ([]byte, error) { + return message.MarshalJSON(t) +} + +func (t *Target) UnmarshalJSON(data []byte) error { + return message.UnmarshalJSON(t, data, new(acl.EACLRecord_Target)) +} + +func (a *APEOverride) MarshalJSON() ([]byte, error) { + return message.MarshalJSON(a) +} + +func (a *APEOverride) UnmarshalJSON(data []byte) error { + return message.UnmarshalJSON(a, data, new(acl.BearerToken_Body_APEOverride)) +} + +func (r *Record) MarshalJSON() ([]byte, error) { + return message.MarshalJSON(r) +} + +func (r *Record) UnmarshalJSON(data []byte) error { + return message.UnmarshalJSON(r, data, new(acl.EACLRecord)) +} + +func (t *Table) MarshalJSON() ([]byte, error) { + return message.MarshalJSON(t) +} + +func (t *Table) UnmarshalJSON(data []byte) error { + return message.UnmarshalJSON(t, data, new(acl.EACLTable)) +} + +func (l *TokenLifetime) MarshalJSON() ([]byte, error) { + return message.MarshalJSON(l) +} + +func (l *TokenLifetime) UnmarshalJSON(data []byte) error { + return message.UnmarshalJSON(l, data, new(acl.BearerToken_Body_TokenLifetime)) +} + +func (bt *BearerTokenBody) MarshalJSON() ([]byte, error) { + return message.MarshalJSON(bt) +} + +func (bt *BearerTokenBody) UnmarshalJSON(data []byte) error { + return message.UnmarshalJSON(bt, data, new(acl.BearerToken_Body)) +} + +func (bt *BearerToken) MarshalJSON() ([]byte, error) { + return message.MarshalJSON(bt) +} + +func (bt *BearerToken) UnmarshalJSON(data []byte) error { + return message.UnmarshalJSON(bt, data, new(acl.BearerToken)) +} diff --git a/api/acl/marshal.go b/api/acl/marshal.go new file mode 100644 index 0000000..29bfa06 --- /dev/null +++ b/api/acl/marshal.go @@ -0,0 +1,350 @@ +package acl + +import ( + acl "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/acl/grpc" + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/rpc/message" + protoutil "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/util/proto" +) + +const ( + filterHeaderTypeField = 1 + filterMatchTypeField = 2 + filterNameField = 3 + filterValueField = 4 + + targetTypeField = 1 + targetKeysField = 2 + + recordOperationField = 1 + recordActionField = 2 + recordFiltersField = 3 + recordTargetsField = 4 + + tableVersionField = 1 + tableContainerIDField = 2 + tableRecordsField = 3 + + lifetimeExpirationField = 1 + lifetimeNotValidBeforeField = 2 + lifetimeIssuedAtField = 3 + + tokenAPEChainsTargetField = 1 + tokenAPEChainsChainsField = 2 + + bearerTokenBodyACLField = 1 + bearerTokenBodyOwnerField = 2 + bearerTokenBodyLifetimeField = 3 + bearerTokenBodyImpersonate = 4 + bearerTokenTokenAPEChainsField = 5 + + bearerTokenBodyField = 1 + bearerTokenSignatureField = 2 +) + +// StableMarshal marshals unified acl table structure in a protobuf +// compatible way without field order shuffle. +func (t *Table) StableMarshal(buf []byte) []byte { + if t == nil { + return []byte{} + } + + if buf == nil { + buf = make([]byte, t.StableSize()) + } + + var offset int + + offset += protoutil.NestedStructureMarshal(tableVersionField, buf[offset:], t.version) + offset += protoutil.NestedStructureMarshal(tableContainerIDField, buf[offset:], t.cid) + + for i := range t.records { + offset += protoutil.NestedStructureMarshal(tableRecordsField, buf[offset:], &t.records[i]) + } + + return buf +} + +// StableSize of acl table structure marshalled by StableMarshal function. +func (t *Table) StableSize() (size int) { + if t == nil { + return 0 + } + + size += protoutil.NestedStructureSize(tableVersionField, t.version) + size += protoutil.NestedStructureSize(tableContainerIDField, t.cid) + + for i := range t.records { + size += protoutil.NestedStructureSize(tableRecordsField, &t.records[i]) + } + + return size +} + +func (t *Table) Unmarshal(data []byte) error { + return message.Unmarshal(t, data, new(acl.EACLTable)) +} + +// StableMarshal marshals unified acl record structure in a protobuf +// compatible way without field order shuffle. +func (r *Record) StableMarshal(buf []byte) []byte { + if r == nil { + return []byte{} + } + + if buf == nil { + buf = make([]byte, r.StableSize()) + } + + var offset int + + offset += protoutil.EnumMarshal(recordOperationField, buf[offset:], int32(r.op)) + offset += protoutil.EnumMarshal(recordActionField, buf[offset:], int32(r.action)) + + for i := range r.filters { + offset += protoutil.NestedStructureMarshal(recordFiltersField, buf[offset:], &r.filters[i]) + } + + for i := range r.targets { + offset += protoutil.NestedStructureMarshal(recordTargetsField, buf[offset:], &r.targets[i]) + } + + return buf +} + +// StableSize of acl record structure marshalled by StableMarshal function. +func (r *Record) StableSize() (size int) { + if r == nil { + return 0 + } + + size += protoutil.EnumSize(recordOperationField, int32(r.op)) + size += protoutil.EnumSize(recordActionField, int32(r.action)) + + for i := range r.filters { + size += protoutil.NestedStructureSize(recordFiltersField, &r.filters[i]) + } + + for i := range r.targets { + size += protoutil.NestedStructureSize(recordTargetsField, &r.targets[i]) + } + + return size +} + +func (r *Record) Unmarshal(data []byte) error { + return message.Unmarshal(r, data, new(acl.EACLRecord)) +} + +// StableMarshal marshals unified header filter structure in a protobuf +// compatible way without field order shuffle. +func (f *HeaderFilter) StableMarshal(buf []byte) []byte { + if f == nil { + return []byte{} + } + + if buf == nil { + buf = make([]byte, f.StableSize()) + } + + var offset int + + offset += protoutil.EnumMarshal(filterHeaderTypeField, buf[offset:], int32(f.hdrType)) + offset += protoutil.EnumMarshal(filterMatchTypeField, buf[offset:], int32(f.matchType)) + offset += protoutil.StringMarshal(filterNameField, buf[offset:], f.key) + protoutil.StringMarshal(filterValueField, buf[offset:], f.value) + + return buf +} + +// StableSize of header filter structure marshalled by StableMarshal function. +func (f *HeaderFilter) StableSize() (size int) { + if f == nil { + return 0 + } + + size += protoutil.EnumSize(filterHeaderTypeField, int32(f.hdrType)) + size += protoutil.EnumSize(filterMatchTypeField, int32(f.matchType)) + size += protoutil.StringSize(filterNameField, f.key) + size += protoutil.StringSize(filterValueField, f.value) + + return size +} + +func (f *HeaderFilter) Unmarshal(data []byte) error { + return message.Unmarshal(f, data, new(acl.EACLRecord_Filter)) +} + +// StableMarshal marshals unified role info structure in a protobuf +// compatible way without field order shuffle. +func (t *Target) StableMarshal(buf []byte) []byte { + if t == nil { + return []byte{} + } + + if buf == nil { + buf = make([]byte, t.StableSize()) + } + + var offset int + + offset += protoutil.EnumMarshal(targetTypeField, buf[offset:], int32(t.role)) + protoutil.RepeatedBytesMarshal(targetKeysField, buf[offset:], t.keys) + + return buf +} + +// StableSize of role info structure marshalled by StableMarshal function. +func (t *Target) StableSize() (size int) { + if t == nil { + return 0 + } + + size += protoutil.EnumSize(targetTypeField, int32(t.role)) + size += protoutil.RepeatedBytesSize(targetKeysField, t.keys) + + return size +} + +func (t *Target) Unmarshal(data []byte) error { + return message.Unmarshal(t, data, new(acl.EACLRecord_Target)) +} + +func (l *TokenLifetime) StableMarshal(buf []byte) []byte { + if l == nil { + return []byte{} + } + + if buf == nil { + buf = make([]byte, l.StableSize()) + } + + var offset int + + offset += protoutil.UInt64Marshal(lifetimeExpirationField, buf[offset:], l.exp) + offset += protoutil.UInt64Marshal(lifetimeNotValidBeforeField, buf[offset:], l.nbf) + protoutil.UInt64Marshal(lifetimeIssuedAtField, buf[offset:], l.iat) + + return buf +} + +func (l *TokenLifetime) StableSize() (size int) { + if l == nil { + return 0 + } + + size += protoutil.UInt64Size(lifetimeExpirationField, l.exp) + size += protoutil.UInt64Size(lifetimeNotValidBeforeField, l.nbf) + size += protoutil.UInt64Size(lifetimeIssuedAtField, l.iat) + + return size +} + +func (l *TokenLifetime) Unmarshal(data []byte) error { + return message.Unmarshal(l, data, new(acl.BearerToken_Body_TokenLifetime)) +} + +func (c *APEOverride) StableMarshal(buf []byte) []byte { + if c == nil { + return []byte{} + } + + if buf == nil { + buf = make([]byte, c.StableSize()) + } + + var offset int + + offset += protoutil.NestedStructureMarshal(tokenAPEChainsTargetField, buf[offset:], c.target) + for i := range c.chains { + offset += protoutil.NestedStructureMarshal(tokenAPEChainsChainsField, buf[offset:], c.chains[i]) + } + + return buf +} + +func (c *APEOverride) StableSize() (size int) { + if c == nil { + return 0 + } + + size += protoutil.NestedStructureSize(tokenAPEChainsTargetField, c.target) + for i := range c.chains { + size += protoutil.NestedStructureSize(tokenAPEChainsChainsField, c.chains[i]) + } + + return size +} + +func (c *APEOverride) Unmarshal(data []byte) error { + return message.Unmarshal(c, data, new(acl.BearerToken_Body_APEOverride)) +} + +func (bt *BearerTokenBody) StableMarshal(buf []byte) []byte { + if bt == nil { + return []byte{} + } + + if buf == nil { + buf = make([]byte, bt.StableSize()) + } + + var offset int + + offset += protoutil.NestedStructureMarshal(bearerTokenBodyACLField, buf[offset:], bt.eacl) + offset += protoutil.NestedStructureMarshal(bearerTokenBodyOwnerField, buf[offset:], bt.ownerID) + offset += protoutil.NestedStructureMarshal(bearerTokenBodyLifetimeField, buf[offset:], bt.lifetime) + offset += protoutil.BoolMarshal(bearerTokenBodyImpersonate, buf[offset:], bt.impersonate) + protoutil.NestedStructureMarshal(bearerTokenTokenAPEChainsField, buf[offset:], bt.apeOverride) + + return buf +} + +func (bt *BearerTokenBody) StableSize() (size int) { + if bt == nil { + return 0 + } + + size += protoutil.NestedStructureSize(bearerTokenBodyACLField, bt.eacl) + size += protoutil.NestedStructureSize(bearerTokenBodyOwnerField, bt.ownerID) + size += protoutil.NestedStructureSize(bearerTokenBodyLifetimeField, bt.lifetime) + size += protoutil.BoolSize(bearerTokenBodyImpersonate, bt.impersonate) + size += protoutil.NestedStructureSize(bearerTokenTokenAPEChainsField, bt.apeOverride) + + return size +} + +func (bt *BearerTokenBody) Unmarshal(data []byte) error { + return message.Unmarshal(bt, data, new(acl.BearerToken_Body)) +} + +func (bt *BearerToken) StableMarshal(buf []byte) []byte { + if bt == nil { + return []byte{} + } + + if buf == nil { + buf = make([]byte, bt.StableSize()) + } + + var offset int + + offset += protoutil.NestedStructureMarshal(bearerTokenBodyField, buf[offset:], bt.body) + protoutil.NestedStructureMarshal(bearerTokenSignatureField, buf[offset:], bt.sig) + + return buf +} + +func (bt *BearerToken) StableSize() (size int) { + if bt == nil { + return 0 + } + + size += protoutil.NestedStructureSize(bearerTokenBodyField, bt.body) + size += protoutil.NestedStructureSize(bearerTokenSignatureField, bt.sig) + + return size +} + +func (bt *BearerToken) Unmarshal(data []byte) error { + return message.Unmarshal(bt, data, new(acl.BearerToken)) +} diff --git a/api/acl/message_test.go b/api/acl/message_test.go new file mode 100644 index 0000000..0131137 --- /dev/null +++ b/api/acl/message_test.go @@ -0,0 +1,21 @@ +package acl_test + +import ( + "testing" + + acltest "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/acl/test" + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/rpc/message" + messagetest "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/rpc/message/test" +) + +func TestMessageConvert(t *testing.T) { + messagetest.TestRPCMessage(t, + func(empty bool) message.Message { return acltest.GenerateFilter(empty) }, + func(empty bool) message.Message { return acltest.GenerateTarget(empty) }, + func(empty bool) message.Message { return acltest.GenerateRecord(empty) }, + func(empty bool) message.Message { return acltest.GenerateTable(empty) }, + func(empty bool) message.Message { return acltest.GenerateTokenLifetime(empty) }, + func(empty bool) message.Message { return acltest.GenerateBearerTokenBody(empty) }, + func(empty bool) message.Message { return acltest.GenerateBearerToken(empty) }, + ) +} diff --git a/api/acl/string.go b/api/acl/string.go new file mode 100644 index 0000000..e5a4462 --- /dev/null +++ b/api/acl/string.go @@ -0,0 +1,110 @@ +package acl + +import ( + acl "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/acl/grpc" +) + +// String returns string representation of Action. +func (x Action) String() string { + return ActionToGRPCField(x).String() +} + +// FromString parses Action from a string representation. +// It is a reverse action to String(). +// +// Returns true if s was parsed successfully. +func (x *Action) FromString(s string) bool { + var g acl.Action + + ok := g.FromString(s) + + if ok { + *x = ActionFromGRPCField(g) + } + + return ok +} + +// String returns string representation of Role. +func (x Role) String() string { + return RoleToGRPCField(x).String() +} + +// FromString parses Role from a string representation. +// It is a reverse action to String(). +// +// Returns true if s was parsed successfully. +func (x *Role) FromString(s string) bool { + var g acl.Role + + ok := g.FromString(s) + + if ok { + *x = RoleFromGRPCField(g) + } + + return ok +} + +// String returns string representation of Operation. +func (x Operation) String() string { + return OperationToGRPCField(x).String() +} + +// FromString parses Operation from a string representation. +// It is a reverse action to String(). +// +// Returns true if s was parsed successfully. +func (x *Operation) FromString(s string) bool { + var g acl.Operation + + ok := g.FromString(s) + + if ok { + *x = OperationFromGRPCField(g) + } + + return ok +} + +// String returns string representation of MatchType. +func (x MatchType) String() string { + return MatchTypeToGRPCField(x).String() +} + +// FromString parses MatchType from a string representation. +// It is a reverse action to String(). +// +// Returns true if s was parsed successfully. +func (x *MatchType) FromString(s string) bool { + var g acl.MatchType + + ok := g.FromString(s) + + if ok { + *x = MatchTypeFromGRPCField(g) + } + + return ok +} + +// String returns string representation of HeaderType. +func (x HeaderType) String() string { + return HeaderTypeToGRPCField(x).String() +} + +// FromString parses HeaderType from a string representation. +// It is a reverse action to String(). +// +// Returns true if s was parsed successfully. +func (x *HeaderType) FromString(s string) bool { + var g acl.HeaderType + + ok := g.FromString(s) + + if ok { + *x = HeaderTypeFromGRPCField(g) + } + + return ok +} diff --git a/api/acl/test/generate.go b/api/acl/test/generate.go new file mode 100644 index 0000000..8b265ad --- /dev/null +++ b/api/acl/test/generate.go @@ -0,0 +1,144 @@ +package acltest + +import ( + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/acl" + apetest "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/ape/test" + accountingtest "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/refs/test" +) + +func GenerateBearerToken(empty bool) *acl.BearerToken { + m := new(acl.BearerToken) + + if !empty { + m.SetBody(GenerateBearerTokenBody(false)) + } + + m.SetSignature(accountingtest.GenerateSignature(empty)) + + return m +} + +func GenerateBearerTokenBody(empty bool) *acl.BearerTokenBody { + m := new(acl.BearerTokenBody) + + if !empty { + m.SetOwnerID(accountingtest.GenerateOwnerID(false)) + m.SetLifetime(GenerateTokenLifetime(false)) + m.SetAPEOverride(GenerateAPEOverride(empty)) + } + + return m +} + +func GenerateAPEOverride(empty bool) *acl.APEOverride { + var m *acl.APEOverride + + if !empty { + m = new(acl.APEOverride) + m.SetTarget(apetest.GenerateChainTarget(empty)) + m.SetChains(apetest.GenerateRawChains(false, 3)) + } + + return m +} + +func GenerateTable(empty bool) *acl.Table { + m := new(acl.Table) + + if !empty { + m.SetRecords(GenerateRecords(false)) + m.SetContainerID(accountingtest.GenerateContainerID(false)) + } + + m.SetVersion(accountingtest.GenerateVersion(empty)) + + return m +} + +func GenerateRecords(empty bool) []acl.Record { + var rs []acl.Record + + if !empty { + rs = append(rs, + *GenerateRecord(false), + *GenerateRecord(false), + ) + } + + return rs +} + +func GenerateRecord(empty bool) *acl.Record { + m := new(acl.Record) + + if !empty { + m.SetAction(acl.ActionAllow) + m.SetOperation(acl.OperationGet) + m.SetFilters(GenerateFilters(false)) + m.SetTargets(GenerateTargets(false)) + } + + return m +} + +func GenerateFilters(empty bool) []acl.HeaderFilter { + var fs []acl.HeaderFilter + + if !empty { + fs = append(fs, + *GenerateFilter(false), + *GenerateFilter(false), + ) + } + + return fs +} + +func GenerateFilter(empty bool) *acl.HeaderFilter { + m := new(acl.HeaderFilter) + + if !empty { + m.SetKey("key") + m.SetValue("val") + m.SetHeaderType(acl.HeaderTypeRequest) + m.SetMatchType(acl.MatchTypeStringEqual) + } + + return m +} + +func GenerateTargets(empty bool) []acl.Target { + var ts []acl.Target + + if !empty { + ts = append(ts, + *GenerateTarget(false), + *GenerateTarget(false), + ) + } + + return ts +} + +func GenerateTarget(empty bool) *acl.Target { + m := new(acl.Target) + + if !empty { + m.SetRole(acl.RoleSystem) + m.SetKeys([][]byte{{1}, {2}}) + } + + return m +} + +func GenerateTokenLifetime(empty bool) *acl.TokenLifetime { + m := new(acl.TokenLifetime) + + if !empty { + m.SetExp(1) + m.SetIat(2) + m.SetExp(3) + } + + return m +} diff --git a/api/acl/types.go b/api/acl/types.go new file mode 100644 index 0000000..e0bae3a --- /dev/null +++ b/api/acl/types.go @@ -0,0 +1,426 @@ +package acl + +import ( + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/ape" + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/refs" +) + +// HeaderFilter is a unified structure of FilterInfo +// message from proto definition. +type HeaderFilter struct { + hdrType HeaderType + + matchType MatchType + + key, value string +} + +// Target is a unified structure of Target +// message from proto definition. +type Target struct { + role Role + + keys [][]byte +} + +// Record is a unified structure of EACLRecord +// message from proto definition. +type Record struct { + op Operation + + action Action + + filters []HeaderFilter + + targets []Target +} + +// Table is a unified structure of EACLTable +// message from proto definition. +type Table struct { + version *refs.Version + + cid *refs.ContainerID + + records []Record +} + +type TokenLifetime struct { + exp, nbf, iat uint64 +} + +type APEOverride struct { + target *ape.ChainTarget + + chains []*ape.Chain +} + +type BearerTokenBody struct { + eacl *Table + + ownerID *refs.OwnerID + + lifetime *TokenLifetime + + apeOverride *APEOverride + + impersonate bool +} + +type BearerToken struct { + body *BearerTokenBody + + sig *refs.Signature +} + +// Target is a unified enum of MatchType enum from proto definition. +type MatchType uint32 + +// HeaderType is a unified enum of HeaderType enum from proto definition. +type HeaderType uint32 + +// Action is a unified enum of Action enum from proto definition. +type Action uint32 + +// Operation is a unified enum of Operation enum from proto definition. +type Operation uint32 + +// Role is a unified enum of Role enum from proto definition. +type Role uint32 + +const ( + MatchTypeUnknown MatchType = iota + MatchTypeStringEqual + MatchTypeStringNotEqual +) + +const ( + HeaderTypeUnknown HeaderType = iota + HeaderTypeRequest + HeaderTypeObject + HeaderTypeService +) + +const ( + ActionUnknown Action = iota + ActionAllow + ActionDeny +) + +const ( + OperationUnknown Operation = iota + OperationGet + OperationHead + OperationPut + OperationDelete + OperationSearch + OperationRange + OperationRangeHash +) + +const ( + RoleUnknown Role = iota + RoleUser + RoleSystem + RoleOthers +) + +func (f *HeaderFilter) GetHeaderType() HeaderType { + if f != nil { + return f.hdrType + } + + return HeaderTypeUnknown +} + +func (f *HeaderFilter) SetHeaderType(v HeaderType) { + f.hdrType = v +} + +func (f *HeaderFilter) GetMatchType() MatchType { + if f != nil { + return f.matchType + } + + return MatchTypeUnknown +} + +func (f *HeaderFilter) SetMatchType(v MatchType) { + f.matchType = v +} + +func (f *HeaderFilter) GetKey() string { + if f != nil { + return f.key + } + + return "" +} + +func (f *HeaderFilter) SetKey(v string) { + f.key = v +} + +func (f *HeaderFilter) GetValue() string { + if f != nil { + return f.value + } + + return "" +} + +func (f *HeaderFilter) SetValue(v string) { + f.value = v +} + +func (t *Target) GetRole() Role { + if t != nil { + return t.role + } + + return RoleUnknown +} + +func (t *Target) SetRole(v Role) { + t.role = v +} + +func (t *Target) GetKeys() [][]byte { + if t != nil { + return t.keys + } + + return nil +} + +func (t *Target) SetKeys(v [][]byte) { + t.keys = v +} + +func (r *Record) GetOperation() Operation { + if r != nil { + return r.op + } + + return OperationUnknown +} + +func (r *Record) SetOperation(v Operation) { + r.op = v +} + +func (r *Record) GetAction() Action { + if r != nil { + return r.action + } + + return ActionUnknown +} + +func (r *Record) SetAction(v Action) { + r.action = v +} + +func (r *Record) GetFilters() []HeaderFilter { + if r != nil { + return r.filters + } + + return nil +} + +func (r *Record) SetFilters(v []HeaderFilter) { + r.filters = v +} + +func (r *Record) GetTargets() []Target { + if r != nil { + return r.targets + } + + return nil +} + +func (r *Record) SetTargets(v []Target) { + r.targets = v +} + +func (t *Table) GetVersion() *refs.Version { + if t != nil { + return t.version + } + + return nil +} + +func (t *Table) SetVersion(v *refs.Version) { + t.version = v +} + +func (t *Table) GetContainerID() *refs.ContainerID { + if t != nil { + return t.cid + } + + return nil +} + +func (t *Table) SetContainerID(v *refs.ContainerID) { + t.cid = v +} + +func (t *Table) GetRecords() []Record { + if t != nil { + return t.records + } + + return nil +} + +func (t *Table) SetRecords(v []Record) { + t.records = v +} + +func (l *TokenLifetime) GetExp() uint64 { + if l != nil { + return l.exp + } + + return 0 +} + +func (l *TokenLifetime) SetExp(v uint64) { + l.exp = v +} + +func (l *TokenLifetime) GetNbf() uint64 { + if l != nil { + return l.nbf + } + + return 0 +} + +func (l *TokenLifetime) SetNbf(v uint64) { + l.nbf = v +} + +func (l *TokenLifetime) GetIat() uint64 { + if l != nil { + return l.iat + } + + return 0 +} + +func (l *TokenLifetime) SetIat(v uint64) { + l.iat = v +} + +func (bt *BearerTokenBody) GetEACL() *Table { + if bt != nil { + return bt.eacl + } + + return nil +} + +func (bt *BearerTokenBody) SetEACL(v *Table) { + bt.eacl = v +} + +func (t *APEOverride) GetTarget() *ape.ChainTarget { + if t == nil { + return nil + } + + return t.target +} + +func (t *APEOverride) GetChains() []*ape.Chain { + if t == nil { + return nil + } + + return t.chains +} + +func (t *APEOverride) SetTarget(v *ape.ChainTarget) { + t.target = v +} + +func (t *APEOverride) SetChains(v []*ape.Chain) { + t.chains = v +} + +func (bt *BearerTokenBody) GetAPEOverride() *APEOverride { + if bt != nil { + return bt.apeOverride + } + + return nil +} + +func (bt *BearerTokenBody) SetAPEOverride(v *APEOverride) { + bt.apeOverride = v +} + +func (bt *BearerTokenBody) GetOwnerID() *refs.OwnerID { + if bt != nil { + return bt.ownerID + } + + return nil +} + +func (bt *BearerTokenBody) SetOwnerID(v *refs.OwnerID) { + bt.ownerID = v +} + +func (bt *BearerTokenBody) GetLifetime() *TokenLifetime { + if bt != nil { + return bt.lifetime + } + + return nil +} + +func (bt *BearerTokenBody) SetLifetime(v *TokenLifetime) { + bt.lifetime = v +} + +func (bt *BearerTokenBody) GetImpersonate() bool { + if bt != nil { + return bt.impersonate + } + + return false +} + +func (bt *BearerTokenBody) SetImpersonate(v bool) { + bt.impersonate = v +} + +func (bt *BearerToken) GetBody() *BearerTokenBody { + if bt != nil { + return bt.body + } + + return nil +} + +func (bt *BearerToken) SetBody(v *BearerTokenBody) { + bt.body = v +} + +func (bt *BearerToken) GetSignature() *refs.Signature { + if bt != nil { + return bt.sig + } + + return nil +} + +func (bt *BearerToken) SetSignature(v *refs.Signature) { + bt.sig = v +} diff --git a/api/ape/convert.go b/api/ape/convert.go new file mode 100644 index 0000000..7bb0bde --- /dev/null +++ b/api/ape/convert.go @@ -0,0 +1,132 @@ +package ape + +import ( + "fmt" + + ape "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/ape/grpc" + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/rpc/grpc" + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/rpc/message" +) + +func TargetTypeToGRPCField(typ TargetType) ape.TargetType { + switch typ { + case TargetTypeNamespace: + return ape.TargetType_NAMESPACE + case TargetTypeContainer: + return ape.TargetType_CONTAINER + case TargetTypeUser: + return ape.TargetType_USER + case TargetTypeGroup: + return ape.TargetType_GROUP + default: + return ape.TargetType_UNDEFINED + } +} + +func TargetTypeFromGRPCField(typ ape.TargetType) TargetType { + switch typ { + case ape.TargetType_NAMESPACE: + return TargetTypeNamespace + case ape.TargetType_CONTAINER: + return TargetTypeContainer + case ape.TargetType_USER: + return TargetTypeUser + case ape.TargetType_GROUP: + return TargetTypeGroup + default: + return TargetTypeUndefined + } +} + +func TargetTypeToGRPC(typ TargetType) ape.TargetType { + return ape.TargetType(typ) +} + +func TargetTypeFromGRPC(typ ape.TargetType) TargetType { + return TargetType(typ) +} + +func (v2 *ChainTarget) ToGRPCMessage() grpc.Message { + var mgrpc *ape.ChainTarget + + if v2 != nil { + mgrpc = new(ape.ChainTarget) + + mgrpc.SetType(TargetTypeToGRPC(v2.GetTargetType())) + mgrpc.SetName(v2.GetName()) + } + + return mgrpc +} + +func (v2 *ChainTarget) FromGRPCMessage(m grpc.Message) error { + mgrpc, ok := m.(*ape.ChainTarget) + if !ok { + return message.NewUnexpectedMessageType(m, mgrpc) + } + + v2.SetTargetType(TargetTypeFromGRPC(mgrpc.GetType())) + v2.SetName(mgrpc.GetName()) + + return nil +} + +func (v2 *ChainRaw) ToGRPCMessage() grpc.Message { + var mgrpc *ape.Chain_Raw + + if v2 != nil { + mgrpc = new(ape.Chain_Raw) + + mgrpc.SetRaw(v2.GetRaw()) + } + + return mgrpc +} + +func (v2 *ChainRaw) FromGRPCMessage(m grpc.Message) error { + mgrpc, ok := m.(*ape.Chain_Raw) + if !ok { + return message.NewUnexpectedMessageType(m, mgrpc) + } + + v2.SetRaw(mgrpc.GetRaw()) + + return nil +} + +func (v2 *Chain) ToGRPCMessage() grpc.Message { + var mgrpc *ape.Chain + + if v2 != nil { + mgrpc = new(ape.Chain) + + switch chainKind := v2.GetKind().(type) { + default: + panic(fmt.Sprintf("unsupported chain kind: %T", chainKind)) + case *ChainRaw: + mgrpc.SetKind(chainKind.ToGRPCMessage().(*ape.Chain_Raw)) + } + } + + return mgrpc +} + +func (v2 *Chain) FromGRPCMessage(m grpc.Message) error { + mgrpc, ok := m.(*ape.Chain) + if !ok { + return message.NewUnexpectedMessageType(m, mgrpc) + } + + switch chainKind := mgrpc.GetKind().(type) { + default: + return fmt.Errorf("unsupported chain kind: %T", chainKind) + case *ape.Chain_Raw: + chainRaw := new(ChainRaw) + if err := chainRaw.FromGRPCMessage(chainKind); err != nil { + return err + } + v2.SetKind(chainRaw) + } + + return nil +} diff --git a/api/ape/grpc/types_frostfs.pb.go b/api/ape/grpc/types_frostfs.pb.go new file mode 100644 index 0000000000000000000000000000000000000000..c16e8673897c7071da1cb3c81841ec2663f0ef70 GIT binary patch literal 8595 zcmeHMYj4{|7X2*!iV1-xq*tN*LO*EX0*>tz?k2Jk+wF%maA|31v*ttKL!#I4f8TTO z42MHmit9Gnwm^+0F^6~VoVoAA4u|3(n@VvlQ<(>)oQkVEG0U?uo4m%_>+9@wm}f;9 z7QQ$<5#y6{addcm?)Uq%U~(H=OA*XuzaJ&DEH8!I@BLb2sc>XE$)-_yJ)}pc-wTsc zIQR>{&ax~P&UIA!jp8T`v*CGO$%8nnrtgBQp?>qanBD@xFqlQds*K_x?Ijh2?|18< z%}~-n{6l(%jTTZE6n8DMo9fD+WXbTyXi{c*6ug&Z9!-j&k;0B&V_D0b0W{)C5XE`L zQe?WJ5+T_|9+giCD;Y=^Wj@K$xp4F^PT%YI%e$Er=Rtlg%X9iQO3R(?ejkDbyRLn8 zc`-gbdV4%RI<)2Y#Ad%|361yPADx}<9~^aswmL!wC*$*dX!_6<+U^KloZ&tHwCcs4 z*y#v;|Ka4~^s&$`U2PuZ6nyRO%QQ$NR3yRdBj58AEP;T$@ArC}BjK0}Is>e1Va42r zSGKWYp!3QO?E+4I*u?|mp{Iw=!Sf)lnv>O;pBMn0Y#loin{={u>_}|!N_*@`Z1V~q zJ5mPP;T3hZTH3|R0~sN#(ur_u%g~-U<6FC)P*4O=M4^}uM0P7idzL60Oqt59y-)up z0fL;%vdU92=WFT*=%VQI-)4Cdl*ei5;)dVuV&6b)ZF>C&17)N2h$r6WS)%b28jmNg zs2r?E#n${-w0JgZc(v8)&b=07i5g<#41>6kK%*l6;3kMtwUbV-CMBKAnb7JJzfjtZ z9Lg2vmws v-&rV{w4RAAl*d2tiONLT^9?uK=^=cZ8%)XQ5g z(!^|9yM@sCwsI8}pF4%O*vg;Sf=C(;5jeqBEYG6 rPRoKYqHpE2XFZrOPIi$J7|x zusEfs?`7!%?o@^d!WHjNtI|y%Q2R$oR5sw<#D6b~0*&1SnM V}8HFw+IAmk7bI@ z?8z0Ib#b!2{h)np@~1J>=e=yiVf!cjY=j%#Zjn!nr(K!n0)Mic%|rDjoQT$lZz-t` zG6YjyV$jNRz9ckw;yY~}JPb$$JQz;=F$8%N9RnwlUFao@+2l_X Cok@_U zS&3p0OhpP6I-5}jzA2mmx@B8l9nuPe_X_0y xT>alYFr8wsb@3{2eD2I-3CG@(?JXhZ6Ba0K30Iz;qk{& zk^T-US^z47f^%Ux{HN;-T!Ii>omxn7x(~{s)xcCQt!;N|RqFP(^3AoX&$Q~frJCc< zm~g&!FJRGymWPhjkd&?4`#R2Ay}-nJ=Vo}oT~#*2c!bBQ#y^~$jF)qCsd+w`n7Ra# zDrz~Z_HriaWL!|~)swmZ6A$S3xJuoW$s ZOlN)q*kM=(_l6w#RFGK}hGrrHzE&pd4X?2N!$8nqg& zk|o`9f(M7S#UG0(4fD05?7W>BS*un#=Yi3NN;gsUM!jN_eB~Rb$d4y(Ot3U&GkYMm zI(C16-Br|^9jl{EB5=Sr9CDV_%6_Jy34t}Pv{yQSF_lJ-haTVhvFejRreVN0T zIt_9a@0d~6mAA+H(QGypBUUG>fG^Io**iIxv5FD@xWMq5^YnuViBc?#5%ox6?bWE= z!LbYxG|-|- 0;c{6qOm_k<%Ed zL=%Kn*69r|&Okrp^hf5eZA0PLTCvVTN*6V{x}NWrXhG`I`ow}cC!dI0uE|kF?Cc~3 z3L8O}<#B2f&?^iew~UEzOwdsRXnsL~z{#-4+&`t80HljoTbl#1vrPua56~=+I<%Vl z`_m~8Iy7x*SskdIvz`RVCJ*B*03X&CSieUdgj~T-h!$o^@|x+jAw80u@+egg=aEO1 z*^M%a98LbUGHRYU#V8|%CJ1#%VGT=dF!R*BlrIpl(}hQ53If3+AUQ#E>;FXQR73?Y zU(y2=tLG5rXHn%=c&X1DzH3qOC1F!)n6T>&hJ 2GD|)1=ip0^^ zQDUw%_~6@2+_g{M@%3klON WU&*mwlLULh FF)f=*)w zfUc%?T=A{Kuv#%#9V?!96mtzzWl3|pE(K;lxC8JgTGk+`8*A3|1x!I+UD$NFs%xrX zb^%+9lg^t9^?8enBpv!0kV*zHRsmf}(0>CGr4&|EXqpd#j08(xmI<4J`{#0C1H&*0 z(4hYt`_;p5UDJr!v`RDe@<`Uk1dU}hdvpF;e7Q^96k+Z#=0}v7G6xeM|KGV4q&W{3 znn55GZ#&o&$$o=xaNq8#dZSVRO&mj}#o=w*?6l>C8$xp?DR5tXaUN=Q zg+GIVTF0A+`-gifw)!mYi6B)(MI}!v7p`eR9B-z2)-{D6XO3p5s xDrUbbo#`|>%+-qWa-*?04Tv~cP0N3%K3g1!p7c#1rkYq=ExO&5zEC{${|9&BXJr5Y literal 0 HcmV?d00001 diff --git a/api/ape/grpc/types_frostfs_fuzz.go b/api/ape/grpc/types_frostfs_fuzz.go new file mode 100644 index 0000000..b7bf367 --- /dev/null +++ b/api/ape/grpc/types_frostfs_fuzz.go @@ -0,0 +1,45 @@ +//go:build gofuzz +// +build gofuzz + +// Code generated by protoc-gen-go-frostfs. DO NOT EDIT. + +package ape + +func DoFuzzProtoChainTarget(data []byte) int { + msg := new(ChainTarget) + if err := msg.UnmarshalProtobuf(data); err != nil { + return 0 + } + _ = msg.MarshalProtobuf(nil) + return 1 +} +func DoFuzzJSONChainTarget(data []byte) int { + msg := new(ChainTarget) + if err := msg.UnmarshalJSON(data); err != nil { + return 0 + } + _, err := msg.MarshalJSON() + if err != nil { + panic(err) + } + return 1 +} +func DoFuzzProtoChain(data []byte) int { + msg := new(Chain) + if err := msg.UnmarshalProtobuf(data); err != nil { + return 0 + } + _ = msg.MarshalProtobuf(nil) + return 1 +} +func DoFuzzJSONChain(data []byte) int { + msg := new(Chain) + if err := msg.UnmarshalJSON(data); err != nil { + return 0 + } + _, err := msg.MarshalJSON() + if err != nil { + panic(err) + } + return 1 +} diff --git a/api/ape/grpc/types_frostfs_test.go b/api/ape/grpc/types_frostfs_test.go new file mode 100644 index 0000000..93d7eea --- /dev/null +++ b/api/ape/grpc/types_frostfs_test.go @@ -0,0 +1,31 @@ +//go:build gofuzz +// +build gofuzz + +// Code generated by protoc-gen-go-frostfs. DO NOT EDIT. + +package ape + +import ( + testing "testing" +) + +func FuzzProtoChainTarget(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzProtoChainTarget(data) + }) +} +func FuzzJSONChainTarget(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzJSONChainTarget(data) + }) +} +func FuzzProtoChain(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzProtoChain(data) + }) +} +func FuzzJSONChain(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzJSONChain(data) + }) +} diff --git a/api/ape/json.go b/api/ape/json.go new file mode 100644 index 0000000..ffa3a8b --- /dev/null +++ b/api/ape/json.go @@ -0,0 +1,14 @@ +package ape + +import ( + ape "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/ape/grpc" + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/rpc/message" +) + +func (t *ChainTarget) MarshalJSON() ([]byte, error) { + return message.MarshalJSON(t) +} + +func (t *ChainTarget) UnmarshalJSON(data []byte) error { + return message.UnmarshalJSON(t, data, new(ape.ChainTarget)) +} diff --git a/api/ape/marshal.go b/api/ape/marshal.go new file mode 100644 index 0000000..e8f377b --- /dev/null +++ b/api/ape/marshal.go @@ -0,0 +1,92 @@ +package ape + +import ( + "fmt" + + ape "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/ape/grpc" + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/rpc/message" + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/util/proto" +) + +const ( + chainTargetTargetTypeField = 1 + chainTargetNameField = 2 + + chainRawField = 1 +) + +func (t *ChainTarget) StableSize() (size int) { + if t == nil { + return 0 + } + + size += proto.EnumSize(chainTargetTargetTypeField, int32(t.targeType)) + size += proto.StringSize(chainTargetNameField, t.name) + + return size +} + +func (t *ChainTarget) StableMarshal(buf []byte) []byte { + if t == nil { + return []byte{} + } + + if buf == nil { + buf = make([]byte, t.StableSize()) + } + + var offset int + + offset += proto.EnumMarshal(chainTargetTargetTypeField, buf[offset:], int32(t.targeType)) + proto.StringMarshal(chainTargetNameField, buf[offset:], t.name) + + return buf +} + +func (t *ChainTarget) Unmarshal(data []byte) error { + return message.Unmarshal(t, data, new(ape.ChainTarget)) +} + +func (c *Chain) StableSize() (size int) { + if c == nil { + return 0 + } + + switch v := c.GetKind().(type) { + case *ChainRaw: + if v != nil { + size += proto.BytesSize(chainRawField, v.GetRaw()) + } + default: + panic(fmt.Sprintf("unsupported chain kind: %T", v)) + } + + return size +} + +func (c *Chain) StableMarshal(buf []byte) []byte { + if c == nil { + return []byte{} + } + + if buf == nil { + buf = make([]byte, c.StableSize()) + } + + var offset int + + switch v := c.GetKind().(type) { + case *ChainRaw: + if v != nil { + proto.BytesMarshal(chainRawField, buf[offset:], v.GetRaw()) + } + default: + panic(fmt.Sprintf("unsupported chain kind: %T", v)) + } + + return buf +} + +func (c *Chain) Unmarshal(data []byte) error { + return message.Unmarshal(c, data, new(ape.Chain)) +} diff --git a/api/ape/message_test.go b/api/ape/message_test.go new file mode 100644 index 0000000..23b929b --- /dev/null +++ b/api/ape/message_test.go @@ -0,0 +1,15 @@ +package ape_test + +import ( + "testing" + + apetest "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/ape/test" + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/rpc/message" + messagetest "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/rpc/message/test" +) + +func TestMessageConvert(t *testing.T) { + messagetest.TestRPCMessage(t, + func(empty bool) message.Message { return apetest.GenerateChainTarget(empty) }, + ) +} diff --git a/api/ape/string.go b/api/ape/string.go new file mode 100644 index 0000000..1d26c28 --- /dev/null +++ b/api/ape/string.go @@ -0,0 +1,18 @@ +package ape + +import ( + apegrpc "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/ape/grpc" +) + +func (tt TargetType) String() string { + return TargetTypeToGRPCField(tt).String() +} + +func (tt *TargetType) FromString(s string) bool { + i, ok := apegrpc.TargetType_value[s] + if ok { + *tt = TargetType(i) + } + + return ok +} diff --git a/api/ape/test/generate.go b/api/ape/test/generate.go new file mode 100644 index 0000000..4fd3dc2 --- /dev/null +++ b/api/ape/test/generate.go @@ -0,0 +1,71 @@ +package test + +import ( + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/ape" +) + +func GenerateRawChains(empty bool, n int) []*ape.Chain { + if empty { + return []*ape.Chain{} + } + + res := make([]*ape.Chain, n) + for i := range res { + res[i] = GenerateRawChain(empty) + } + return res +} + +func GenerateRawChain(empty bool) *ape.Chain { + chRaw := new(ape.ChainRaw) + + if empty { + chRaw.SetRaw([]byte("{}")) + } else { + chRaw.SetRaw([]byte(`{ + "ID": "", + "Rules": [ + { + "Status": "Allow", + "Actions": { + "Inverted": false, + "Names": [ + "GetObject" + ] + }, + "Resources": { + "Inverted": false, + "Names": [ + "native:object/*" + ] + }, + "Any": false, + "Condition": [ + { + "Op": "StringEquals", + "Object": "Resource", + "Key": "Department", + "Value": "HR" + } + ] + } + ], + "MatchType": "DenyPriority" + }`)) + } + + ch := new(ape.Chain) + ch.SetKind(chRaw) + return ch +} + +func GenerateChainTarget(empty bool) *ape.ChainTarget { + m := new(ape.ChainTarget) + + if !empty { + m.SetTargetType(ape.TargetTypeContainer) + m.SetName("BzQw5HH3feoxFDD5tCT87Y1726qzgLfxEE7wgtoRzB3R") + } + + return m +} diff --git a/api/ape/types.go b/api/ape/types.go new file mode 100644 index 0000000..467a441 --- /dev/null +++ b/api/ape/types.go @@ -0,0 +1,79 @@ +package ape + +type TargetType uint32 + +const ( + TargetTypeUndefined TargetType = iota + TargetTypeNamespace + TargetTypeContainer + TargetTypeUser + TargetTypeGroup +) + +type ChainTarget struct { + targeType TargetType + + name string +} + +func (ct *ChainTarget) SetTargetType(targeType TargetType) { + ct.targeType = targeType +} + +func (ct *ChainTarget) SetName(name string) { + ct.name = name +} + +func (ct *ChainTarget) GetTargetType() TargetType { + if ct != nil { + return ct.targeType + } + + return 0 +} + +func (ct *ChainTarget) GetName() string { + if ct != nil { + return ct.name + } + + return "" +} + +type chainKind interface { + isChainKind() +} + +type Chain struct { + kind chainKind +} + +func (c *Chain) SetKind(kind chainKind) { + c.kind = kind +} + +func (c *Chain) GetKind() chainKind { + if c == nil { + return nil + } + + return c.kind +} + +type ChainRaw struct { + Raw []byte +} + +func (*ChainRaw) isChainKind() {} + +func (c *ChainRaw) SetRaw(raw []byte) { + c.Raw = raw +} + +func (c *ChainRaw) GetRaw() []byte { + if c == nil { + return nil + } + + return c.Raw +} diff --git a/api/apemanager/convert.go b/api/apemanager/convert.go new file mode 100644 index 0000000..5591791 --- /dev/null +++ b/api/apemanager/convert.go @@ -0,0 +1,358 @@ +package apemanager + +import ( + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/ape" + apeGRPC "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/ape/grpc" + apemanager "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/apemanager/grpc" + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/rpc/grpc" + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/rpc/message" +) + +func (reqBody *AddChainRequestBody) ToGRPCMessage() grpc.Message { + var reqBodygrpc *apemanager.AddChainRequest_Body + + if reqBody != nil { + reqBodygrpc = new(apemanager.AddChainRequest_Body) + + reqBodygrpc.SetTarget(reqBody.GetTarget().ToGRPCMessage().(*apeGRPC.ChainTarget)) + reqBodygrpc.SetChain(reqBody.GetChain().ToGRPCMessage().(*apeGRPC.Chain)) + } + + return reqBodygrpc +} + +func (reqBody *AddChainRequestBody) FromGRPCMessage(m grpc.Message) error { + reqBodygrpc, ok := m.(*apemanager.AddChainRequest_Body) + if !ok { + return message.NewUnexpectedMessageType(m, reqBodygrpc) + } + + if targetgrpc := reqBodygrpc.GetTarget(); targetgrpc != nil { + reqBody.target = new(ape.ChainTarget) + if err := reqBody.target.FromGRPCMessage(targetgrpc); err != nil { + return err + } + } + + if chaingrpc := reqBodygrpc.GetChain(); chaingrpc != nil { + reqBody.chain = new(ape.Chain) + if err := reqBody.GetChain().FromGRPCMessage(chaingrpc); err != nil { + return err + } + } + + return nil +} + +func (req *AddChainRequest) ToGRPCMessage() grpc.Message { + var reqgrpc *apemanager.AddChainRequest + + if req != nil { + reqgrpc = new(apemanager.AddChainRequest) + + reqgrpc.SetBody(req.GetBody().ToGRPCMessage().(*apemanager.AddChainRequest_Body)) + req.RequestHeaders.ToMessage(reqgrpc) + } + + return reqgrpc +} + +func (req *AddChainRequest) FromGRPCMessage(m grpc.Message) error { + reqgrpc, ok := m.(*apemanager.AddChainRequest) + if !ok { + return message.NewUnexpectedMessageType(m, reqgrpc) + } + + if reqBodygrpc := reqgrpc.GetBody(); reqBodygrpc != nil { + req.body = new(AddChainRequestBody) + if err := req.body.FromGRPCMessage(reqBodygrpc); err != nil { + return err + } + } + + return req.RequestHeaders.FromMessage(reqgrpc) +} + +func (respBody *AddChainResponseBody) ToGRPCMessage() grpc.Message { + var respBodygrpc *apemanager.AddChainResponse_Body + + if respBody != nil { + respBodygrpc = new(apemanager.AddChainResponse_Body) + + respBodygrpc.SetChainId(respBody.GetChainID()) + } + + return respBodygrpc +} + +func (respBody *AddChainResponseBody) FromGRPCMessage(m grpc.Message) error { + respBodygrpc, ok := m.(*apemanager.AddChainResponse_Body) + if !ok { + return message.NewUnexpectedMessageType(m, respBodygrpc) + } + + respBody.SetChainID(respBodygrpc.GetChainId()) + + return nil +} + +func (resp *AddChainResponse) ToGRPCMessage() grpc.Message { + var respgrpc *apemanager.AddChainResponse + + if resp != nil { + respgrpc = new(apemanager.AddChainResponse) + + respgrpc.SetBody(resp.body.ToGRPCMessage().(*apemanager.AddChainResponse_Body)) + resp.ResponseHeaders.ToMessage(respgrpc) + } + + return respgrpc +} + +func (resp *AddChainResponse) FromGRPCMessage(m grpc.Message) error { + respgrpc, ok := m.(*apemanager.AddChainResponse) + if !ok { + return message.NewUnexpectedMessageType(m, respgrpc) + } + + if respBodygrpc := respgrpc.GetBody(); respBodygrpc != nil { + resp.body = new(AddChainResponseBody) + if err := resp.body.FromGRPCMessage(respBodygrpc); err != nil { + return err + } + } + + return resp.ResponseHeaders.FromMessage(respgrpc) +} + +func (reqBody *RemoveChainRequestBody) ToGRPCMessage() grpc.Message { + var reqBodygrpc *apemanager.RemoveChainRequest_Body + + if reqBody != nil { + reqBodygrpc = new(apemanager.RemoveChainRequest_Body) + + reqBodygrpc.SetTarget(reqBody.target.ToGRPCMessage().(*apeGRPC.ChainTarget)) + reqBodygrpc.SetChainId(reqBody.GetChainID()) + } + + return reqBodygrpc +} + +func (reqBody *RemoveChainRequestBody) FromGRPCMessage(m grpc.Message) error { + reqBodygrpc, ok := m.(*apemanager.RemoveChainRequest_Body) + if !ok { + return message.NewUnexpectedMessageType(m, reqBodygrpc) + } + + if targetgrpc := reqBodygrpc.GetTarget(); targetgrpc != nil { + reqBody.target = new(ape.ChainTarget) + if err := reqBody.target.FromGRPCMessage(targetgrpc); err != nil { + return err + } + } + + reqBody.SetChainID(reqBodygrpc.GetChainId()) + + return nil +} + +func (req *RemoveChainRequest) ToGRPCMessage() grpc.Message { + var reqgrpc *apemanager.RemoveChainRequest + + if req != nil { + reqgrpc = new(apemanager.RemoveChainRequest) + + reqgrpc.SetBody(req.body.ToGRPCMessage().(*apemanager.RemoveChainRequest_Body)) + req.RequestHeaders.ToMessage(reqgrpc) + } + + return reqgrpc +} + +func (req *RemoveChainRequest) FromGRPCMessage(m grpc.Message) error { + reqgrpc, ok := m.(*apemanager.RemoveChainRequest) + if !ok { + return message.NewUnexpectedMessageType(m, reqgrpc) + } + + if reqBodygrpc := reqgrpc.GetBody(); reqBodygrpc != nil { + req.body = new(RemoveChainRequestBody) + if err := req.body.FromGRPCMessage(reqBodygrpc); err != nil { + return err + } + } + + return req.RequestHeaders.FromMessage(reqgrpc) +} + +func (respBody *RemoveChainResponseBody) ToGRPCMessage() grpc.Message { + var respBodygrpc *apemanager.RemoveChainResponse_Body + + if respBody != nil { + respBodygrpc = new(apemanager.RemoveChainResponse_Body) + } + + return respBodygrpc +} + +func (respBody *RemoveChainResponseBody) FromGRPCMessage(m grpc.Message) error { + respBodygrpc, ok := m.(*apemanager.RemoveChainResponse_Body) + if !ok { + return message.NewUnexpectedMessageType(m, respBodygrpc) + } + + return nil +} + +func (resp *RemoveChainResponse) ToGRPCMessage() grpc.Message { + var respgrpc *apemanager.RemoveChainResponse + + if resp != nil { + respgrpc = new(apemanager.RemoveChainResponse) + + respgrpc.SetBody(resp.body.ToGRPCMessage().(*apemanager.RemoveChainResponse_Body)) + resp.ResponseHeaders.ToMessage(respgrpc) + } + + return respgrpc +} + +func (resp *RemoveChainResponse) FromGRPCMessage(m grpc.Message) error { + respgrpc, ok := m.(*apemanager.RemoveChainResponse) + if !ok { + return message.NewUnexpectedMessageType(m, respgrpc) + } + + if respBodygrpc := respgrpc.GetBody(); respBodygrpc != nil { + resp.body = new(RemoveChainResponseBody) + if err := resp.body.FromGRPCMessage(respBodygrpc); err != nil { + return err + } + } + + return resp.ResponseHeaders.FromMessage(respgrpc) +} + +func (reqBody *ListChainsRequestBody) ToGRPCMessage() grpc.Message { + var reqBodygrpc *apemanager.ListChainsRequest_Body + + if reqBody != nil { + reqBodygrpc = new(apemanager.ListChainsRequest_Body) + + reqBodygrpc.SetTarget(reqBody.target.ToGRPCMessage().(*apeGRPC.ChainTarget)) + } + + return reqBodygrpc +} + +func (reqBody *ListChainsRequestBody) FromGRPCMessage(m grpc.Message) error { + reqBodygrpc, ok := m.(*apemanager.ListChainsRequest_Body) + if !ok { + return message.NewUnexpectedMessageType(m, reqBodygrpc) + } + + if targetgrpc := reqBodygrpc.GetTarget(); targetgrpc != nil { + reqBody.target = new(ape.ChainTarget) + if err := reqBody.target.FromGRPCMessage(targetgrpc); err != nil { + return err + } + } + + return nil +} + +func (req *ListChainsRequest) ToGRPCMessage() grpc.Message { + var reqgrpc *apemanager.ListChainsRequest + + if req != nil { + reqgrpc = new(apemanager.ListChainsRequest) + + reqgrpc.SetBody(req.body.ToGRPCMessage().(*apemanager.ListChainsRequest_Body)) + req.RequestHeaders.ToMessage(reqgrpc) + } + + return reqgrpc +} + +func (req *ListChainsRequest) FromGRPCMessage(m grpc.Message) error { + reqgrpc, ok := m.(*apemanager.ListChainsRequest) + if !ok { + return message.NewUnexpectedMessageType(m, reqgrpc) + } + + if reqBodygrpc := reqgrpc.GetBody(); reqBodygrpc != nil { + req.body = new(ListChainsRequestBody) + if err := req.body.FromGRPCMessage(reqBodygrpc); err != nil { + return err + } + } + + return req.RequestHeaders.FromMessage(reqgrpc) +} + +func (respBody *ListChainsResponseBody) ToGRPCMessage() grpc.Message { + var respBodygrpc *apemanager.ListChainsResponse_Body + + if respBody != nil { + respBodygrpc = new(apemanager.ListChainsResponse_Body) + + chainsgrpc := make([]apeGRPC.Chain, 0, len(respBody.GetChains())) + for _, chain := range respBody.GetChains() { + chainsgrpc = append(chainsgrpc, *chain.ToGRPCMessage().(*apeGRPC.Chain)) + } + + respBodygrpc.SetChains(chainsgrpc) + } + + return respBodygrpc +} + +func (respBody *ListChainsResponseBody) FromGRPCMessage(m grpc.Message) error { + respBodygrpc, ok := m.(*apemanager.ListChainsResponse_Body) + if !ok { + return message.NewUnexpectedMessageType(m, respBodygrpc) + } + + chains := make([]*ape.Chain, 0, len(respBodygrpc.GetChains())) + + for _, chaingrpc := range respBodygrpc.GetChains() { + chain := new(ape.Chain) + if err := chain.FromGRPCMessage(&chaingrpc); err != nil { + return err + } + chains = append(chains, chain) + } + + respBody.SetChains(chains) + + return nil +} + +func (resp *ListChainsResponse) ToGRPCMessage() grpc.Message { + var respgrpc *apemanager.ListChainsResponse + + if resp != nil { + respgrpc = new(apemanager.ListChainsResponse) + + respgrpc.SetBody(resp.body.ToGRPCMessage().(*apemanager.ListChainsResponse_Body)) + resp.ResponseHeaders.ToMessage(respgrpc) + } + + return respgrpc +} + +func (resp *ListChainsResponse) FromGRPCMessage(m grpc.Message) error { + respgrpc, ok := m.(*apemanager.ListChainsResponse) + if !ok { + return message.NewUnexpectedMessageType(m, respgrpc) + } + + if respBodygrpc := respgrpc.GetBody(); respBodygrpc != nil { + resp.body = new(ListChainsResponseBody) + if err := resp.body.FromGRPCMessage(respBodygrpc); err != nil { + return err + } + } + + return resp.ResponseHeaders.FromMessage(respgrpc) +} diff --git a/api/apemanager/grpc/service_frostfs.pb.go b/api/apemanager/grpc/service_frostfs.pb.go new file mode 100644 index 0000000000000000000000000000000000000000..b9e06704bc21332364533e49c4478cf2b1506e6e GIT binary patch literal 57153 zcmeHQTXWk+vVJCi#h6uzNv}-B$*J?8Ep64t_U_Rs$4Obv?n9E=g-B3BOoChhq@;EE zzi)TX9fQGzA|#Rvmu~{*(lgUD-TlosgVR&?dN~c*br^?9kcCrrb aoq2(g2|`gI%L5rTm&)vFX{KA#cG*kY}oJpo-SiH z2;<3e8pYSA_~W48n=LXnfWP3s*U4(a2G>zGmgPtBY Jt z+S??`suJ@wgZ`-B&+b S?2bh-pX^ub8JjJWMUaglAJ2-TvQEO3-A36B&lfC0HjBdflx>1J zj3B!SHlcoX6~#dU0}92D`?GaCVZ%FC+_Q#g!x0;jL{XfLPz@rK?EIWT=i`q(E;{?B z-}}< cTU!AbKF;^TIyXy4cVtP_2NG|HaT&O(IhhpSN zX|S7MjhlyyFwS^i`m_of6DG4@Qr#1|3Wn1(W54`%b)SVJ@h|qmf}KI5VZG6+^b`K{ z*DxD`+|zIds$B5?r*$@5K!UsR+eMVgRty*8A7B}TRa+_MaD>G`-RBHma{WeGbf>@% z*e4YOYge;iFVqU-R9Q0GG}fpHVkFr!tq<~9u0@OSpI56eo{EMIUtxnlW2h9qU|5CF zqlz-V BlqV!=HCJBQ-%Y^z4*S6V&=^p+IPR`d$ z(7n41% Qo0_^tqEml1n z4kkeyFEdzFf+>rk2(c=_a`II=IAMdL)pP*Ef#cCjVL_SPVC_`%`}KkZn*?ddUY#+R zbA0wf(%v-4f)lp ONL?lX9c%Y<4e3Itut6 Ef#063$PAVEJLs^^05T-;yM)4${1=c3YUZ0xWY?S;l{0y zW~(e3v};qV0o19s{HW< Fgt5Ug z(HG#hk0ydx=5BGe35Rq1ck(+wgF4(vGxZicUCRCSEhwfK$lLMv>lqkD_$QiAzXx-g z4o4^Kmak#e{V$|aWiQv@nc6}W+Al~q{t(=PPXd9@a1gKObDB_GEE=6zlz^RuHi6#~ z_In}+ia31!`8hNJmopgUXFMS-ZdRfbOw{|r!Z|FN^qQxHH8Oxl_zwO|mN8VbO2S!m zhh8?w10y;3Rrt}r4$i=jm$eIdd1@?OYOB-9APKyO3R;$Of O610#WyazRg$ti|B(m6=`APG1f z>3R`P$t 0&$0RoUA zsW@sVWczy%XRnv@WlYr+2y2O jeodq?zQH-t8GY$%v;RBk(=oAfTDh$d5@KntUF0WKB@v2TSdc1E% zPr)-iayXB0eafi;Brvt$3(mEou-}OI2>_5WZDFJ+U*f+axW59_06_qNFoORB50qkf zvhlCR0$H$L@ghX#h026Q{4GqP*}W*lE`nY~lOO}ENF)$NY>Z3~!5FrL7j(9q#WlLB z0F}sPQ)>hDp2+IYw!e77sBI_#brkfyCXk>tuU4M#5PUGT5ORry-oPL}fR ?GfY$)vL@E3)MbILMp-x@gC4wg489~G& zvq0KlK3{_Jhf@g%eGmXr5Z_~r0_!$HA8Uj;05!h7i6Eq#Mi34{T!@)(8KQsab0FkI zrV#K)z)rGoeorF)2rrV`C=E|6bV`T~rivG;vV&AuWERF!CI$R70I7`L@`F^MRI0mq z|6mAD)| H%7mU-xmEK>hX8oPCM6sL1+)yum3I)d8F8O vm^)b9Y{YgAnUPgB z0>ry9KB<7ZEVxXsVkQy<0hh9#6{QK_jAB)5eY49mSH6l8zk56=Zp4&zl$v`CC|m;o z4)q0HN?f*EgWs?c3u_|1 5Mw^P)fbap8AepN^?TD-dAR2R&SXviu#M*TfV*1Z)Oqg^&Es%@LL$xa$Ix7 zl*duLr-t4M%#J&Ag;ge-Pw4t}0h) pqbTaqAP}Is+mhP*6`Q7sF?8g z>S1EaMI$h550G6Ykb+(S9y}IM_X?(KC0a@rvjEwK M})!X#dt& m4o$w-&N>=(L@TyMKKVva;(2JFjXyW zS`mcavZ9UJR3OkmHv@5lT}GW&OT}8V2eD_7Sjrtfq6lw4k1k5M5a#sg}Ygi?NEuQL5ENT3bX~LW$il zf-jc1 87K!BYsnr1dDfO!;2od@AIR5AuH6|wm&q)zo<)UNM6d(ICdaIC8 z! >8ssLDCTO) njmv#_R(H1nD)8=oMLL$i@RwQpEwN3_AI>nDP=OkG)YL>oGT zivZp5c{`^*< k32cf3JorIt=l8pa9K>-E#PT+ zeTCPPt&m9 Qu6u9v%<0Zn= zt`PI^h-+SrQ)c;*L5pgE)Yy0AwJpeNe7d>ZHRl3ECk8hSSe=+#SsF!l IRs%5Df84x-58WiYwbOtl5yP1lm71yU5y=Cy|+?D7!nRG?%JR~k%k^Dylg>`(o zC+E`@X_zvZM+UGpz}7hX<_S4kURkjJ8ncQ6ovz*;d6wYS_d2zel+Itn3N z|K9;_&FG=Mo&zYx%W`vPfyJtGU-N(ppdAEO%*~LxmsYt&f@h@ft#V)UX^Hr3@zqss zjuLiVKDaAuh}P ^hDeeypk17aukTPE-MN+t*S1IgG6xw=SA~%FTUpz`Envk;~Q^ z&(0lvvf374?MYJyw-%K89BOSKu?}!;L$RhMFQD4x%`z8TJ7mW82A4K>>10kWbJ}`P z{Ss?!^_e8v`Rr*8Npr?l$=WW@&Y)Xrv>xsasrg+;OIM7tH1ynn5?7eN0qE_|uNJsc z=-bV|uK>6y>DIKkR C zqrILUd9RX1@WQ8QGJzxsT;X|e@AtB}LTJA>pWE<_y(|S@2Kk59)v8hg%a7ffrod}? zfa9jKHul+aS=Gd;;jW3S`|YW9Tm@tIz2hqEf~z2=>rsA5t6=FptgmSmHsCnu3R3SH z#yor*>KqPAM04C+5cjImP6knb$@m% {Y{Km{YyQ5JlZ1-_nW2ie(s*RM3ttF-&S@Dp3S!`x`V={OTmZm;VK@Riv* zp0ME^D+;0ftOee#-O?7MvGVa2j5@_0KbfJj6JD}NINhZ+)7;$W3G} UFyLhN zZ?U5n3a)4OJuWu*b+9>93WJ=0%JY4C_}9RqJLDHW+KnXCc-#p2<;UHw5L$I913v#6 zE9%K_bNLdHI2Bjk2q!>o!E id6MB}Vufcd+Q7uR*&82*sIb z3nJvt>Rs#83hTnxcr^zVQjsPT^`WLA9>{<{%Y@<- EMJ7id)wJisR#h_N7%Yj#oo+kyV5P2tr%- #gPdH;POiuV>zN{0js3lE> zRqY2VUfNNyRw>$$3|uHQ_Ogz0M0u6F!I#B6Om >f(eA+s4(is~DO}UI}{iuwW zM8 @l{Zh8_ zxNwaIkOuAzv<_%sB^I0^b|mnYVP{l2ZjsuUfz1Ya5wkkEsW(n277)l7gxTid(DwK- zv_sQfhXN2_5dyiEB*!!M)h$4d09w*0VF1<$G_!a`#Z#n|pTeg#R4KrJpsDu1^c9oI zDbczM#d*u8BnQmI)V1!6b}~HXO(+GGd&xw(K(dlcR8V#algJlimRgrI&e!6s^AtATCK-Bd^xcjW_Up0Z`F4T?zUd0~6b}5>za+&}C!; p zov^QE-e02MK-+bc(ayKSeEN2{SFd D@ bp%U8dl_!^{|&9u!_ zqr0P1Z?F#OrKkk(Fb0Eg3f@@ysFiMADY_yDrg?P`w2SICmN^!ox$5`bvyrS;?^9@A zB$#rij);#mxe1~;t+-FqVjKPP9A{YBLXI;lLR#cEsITzL%6~%nC54dd7MN@g06^)i z{J&v-&~fYGE^j~b+CgOjo-lf~^im0=G_xZ?Z9#(KbHF7*>2;U17#YNKI16Ddk@N=Q z=r`d!S`44_$-$?`GkAQFiWi`@Vlwpb@Qg0{GpI8DGE9&$nq G)@Am~c^m4O|X;KOw^0^IhUo|<47wKhZ z88r$7QOsa3Jxdn&i1$4et?*uk7MIVMO4yU;1OiQ}(>POc9_zLzPeUzHHcWjYgg3Qd z67VLQ*+iose$#uAi2DZycSJ3m#vYs&)(ymZ!Mt==JssLrPes#D;X_Z3bsLVi0gSR2 zzt6+FFd4tYzsbx*8l8l_KL95$(n~nsnjtm`dPxu7ujg}Ec!jDOIcTFI2WS$G3ePS- zMJu{tiCae6eSpUT*`-tCxE*5Qy#Sd~EuX@B+=+zbKgVe>3+V!h_wW_%A@TjtwSadV zw4N_xs-{5LN1|#OmK)vtp1kMD)e!A=mm=5dBbD2bwenWi5(yD>WEJNVVUL!h={dWd flWb$cyzaGFcD1gVERoSoTu0WN!Bx`k5VZdf>Ra>; literal 0 HcmV?d00001 diff --git a/api/apemanager/grpc/service_frostfs_fuzz.go b/api/apemanager/grpc/service_frostfs_fuzz.go new file mode 100644 index 0000000..08af63e --- /dev/null +++ b/api/apemanager/grpc/service_frostfs_fuzz.go @@ -0,0 +1,121 @@ +//go:build gofuzz +// +build gofuzz + +// Code generated by protoc-gen-go-frostfs. DO NOT EDIT. + +package apemanager + +func DoFuzzProtoAddChainRequest(data []byte) int { + msg := new(AddChainRequest) + if err := msg.UnmarshalProtobuf(data); err != nil { + return 0 + } + _ = msg.MarshalProtobuf(nil) + return 1 +} +func DoFuzzJSONAddChainRequest(data []byte) int { + msg := new(AddChainRequest) + if err := msg.UnmarshalJSON(data); err != nil { + return 0 + } + _, err := msg.MarshalJSON() + if err != nil { + panic(err) + } + return 1 +} +func DoFuzzProtoAddChainResponse(data []byte) int { + msg := new(AddChainResponse) + if err := msg.UnmarshalProtobuf(data); err != nil { + return 0 + } + _ = msg.MarshalProtobuf(nil) + return 1 +} +func DoFuzzJSONAddChainResponse(data []byte) int { + msg := new(AddChainResponse) + if err := msg.UnmarshalJSON(data); err != nil { + return 0 + } + _, err := msg.MarshalJSON() + if err != nil { + panic(err) + } + return 1 +} +func DoFuzzProtoRemoveChainRequest(data []byte) int { + msg := new(RemoveChainRequest) + if err := msg.UnmarshalProtobuf(data); err != nil { + return 0 + } + _ = msg.MarshalProtobuf(nil) + return 1 +} +func DoFuzzJSONRemoveChainRequest(data []byte) int { + msg := new(RemoveChainRequest) + if err := msg.UnmarshalJSON(data); err != nil { + return 0 + } + _, err := msg.MarshalJSON() + if err != nil { + panic(err) + } + return 1 +} +func DoFuzzProtoRemoveChainResponse(data []byte) int { + msg := new(RemoveChainResponse) + if err := msg.UnmarshalProtobuf(data); err != nil { + return 0 + } + _ = msg.MarshalProtobuf(nil) + return 1 +} +func DoFuzzJSONRemoveChainResponse(data []byte) int { + msg := new(RemoveChainResponse) + if err := msg.UnmarshalJSON(data); err != nil { + return 0 + } + _, err := msg.MarshalJSON() + if err != nil { + panic(err) + } + return 1 +} +func DoFuzzProtoListChainsRequest(data []byte) int { + msg := new(ListChainsRequest) + if err := msg.UnmarshalProtobuf(data); err != nil { + return 0 + } + _ = msg.MarshalProtobuf(nil) + return 1 +} +func DoFuzzJSONListChainsRequest(data []byte) int { + msg := new(ListChainsRequest) + if err := msg.UnmarshalJSON(data); err != nil { + return 0 + } + _, err := msg.MarshalJSON() + if err != nil { + panic(err) + } + return 1 +} +func DoFuzzProtoListChainsResponse(data []byte) int { + msg := new(ListChainsResponse) + if err := msg.UnmarshalProtobuf(data); err != nil { + return 0 + } + _ = msg.MarshalProtobuf(nil) + return 1 +} +func DoFuzzJSONListChainsResponse(data []byte) int { + msg := new(ListChainsResponse) + if err := msg.UnmarshalJSON(data); err != nil { + return 0 + } + _, err := msg.MarshalJSON() + if err != nil { + panic(err) + } + return 1 +} diff --git a/api/apemanager/grpc/service_frostfs_test.go b/api/apemanager/grpc/service_frostfs_test.go new file mode 100644 index 0000000..5c4653c --- /dev/null +++ b/api/apemanager/grpc/service_frostfs_test.go @@ -0,0 +1,71 @@ +//go:build gofuzz +// +build gofuzz + +// Code generated by protoc-gen-go-frostfs. DO NOT EDIT. + +package apemanager + +import ( + testing "testing" +) + +func FuzzProtoAddChainRequest(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzProtoAddChainRequest(data) + }) +} +func FuzzJSONAddChainRequest(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzJSONAddChainRequest(data) + }) +} +func FuzzProtoAddChainResponse(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzProtoAddChainResponse(data) + }) +} +func FuzzJSONAddChainResponse(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzJSONAddChainResponse(data) + }) +} +func FuzzProtoRemoveChainRequest(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzProtoRemoveChainRequest(data) + }) +} +func FuzzJSONRemoveChainRequest(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzJSONRemoveChainRequest(data) + }) +} +func FuzzProtoRemoveChainResponse(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzProtoRemoveChainResponse(data) + }) +} +func FuzzJSONRemoveChainResponse(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzJSONRemoveChainResponse(data) + }) +} +func FuzzProtoListChainsRequest(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzProtoListChainsRequest(data) + }) +} +func FuzzJSONListChainsRequest(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzJSONListChainsRequest(data) + }) +} +func FuzzProtoListChainsResponse(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzProtoListChainsResponse(data) + }) +} +func FuzzJSONListChainsResponse(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzJSONListChainsResponse(data) + }) +} diff --git a/api/apemanager/grpc/service_grpc.pb.go b/api/apemanager/grpc/service_grpc.pb.go new file mode 100644 index 0000000000000000000000000000000000000000..47814272fe891c0b1e467544d451c40630017f37 GIT binary patch literal 9993 zcmeHNYj5MY75yCl3f2Xtw&BRLSu6(dA}9jKvo*T0gFO1BExe3G+rr4AmZY6A2Knzj zmy|5pvQlSfXJ=X@m`QB$lJ~)Lc^_1-#|APIY%WrvxDgS1*sx4VBg0cXJDtnZxyr(T zT~63|a>s_3qr0Hd>#?;^I+m&K)0fi&Q`q{u;QQdMKg|4}^$)?>yWot*wOpxC^cl~@ zlBawwRF9bUv{36f6oHke(a3oC0fQb{qY*DNsSIm1PC}U)@zJnm{-N18p+#78F6BHC z!CWRhod;6Q?cx{?p}4La@7YmMX~WH`p3Dt58|?;Za<_;z!#~bKxy<53oSJwEi8WL) zFc&kDiBzwYU}nJ$eu22O{R}f`89b!<+{6!wV2`m`fD)TT`V>}(4Us*FIGq>FBF5)D zPPMT+bR&LU#R?kEZ>|TYKS{eGXTe)06-%(yDrmq|+OR1*w@U}NtBkC9?d&|#|8RT! zu8|YM$`{weE4M|rZgZv=Q8ZY9>-7C9Nv?!h$Y{)=Gk(rlvo}*xo0$&QXF<8WL1md< zu?#F;8JymTrCf`HY+k~{Tjn5}pJQz-6J1f}C2Y!N3RqM$8YHnu?FQ1mhVI+N^@z=| zITdPFI0CRi@OTd*O|D|%5Li=kl}0QyAK6L+YxFsh8o20*G$MFtqXb{lCm=z0S(XS6 zN+o7O5h54HWV+w$Wgq5nlOPi7-eITd{nhY?^GJq&b_xt+ng(O>cxxF14YSEaRs2F( z_@|ojP_R#p6KH`Qho7hw5FQeMtnQrYOoZ_)4w>Om0<95}7TrT0nGae^!Va-~_+V3w$Ki}UBr-RAW)nwc*XLLG) z$@uPKG#=heaehwUPwvN;B-r hx56#Hm2OQzF%h#uF{sQt(t4kog1f zXk~jvp|%e9fQjq@&nMMQ{Ek1 vvz_t5gxaDkEU=T3QGM0j3>hN5B^x 8kn-;RV~8p_ p%U;{T Ks& zuQkI9YglEQCtkJqTEz;KYF^N4Rf^b-H=0KY!rji(-DqHQcsj&iYl8pLwniC&58$C3 z^x)t>+(QGhbwsseqF)g&AA}2;XgJxP9F=qX5fQDjZ8&w&Ht)F|$Ec_?ic+F=%rrl% z|D3a#B6MD8*nHVF&VwN}duFgWdhx-IJYCjY+II4`tgx-GtpP;TY$Ln2MbVL(c>a4} zkG)ZElXsr(?(%1- Eu8B-Bc *8X@wRq9Uql*h< zz_KdmsqZ}8B(gegZ&y4wFvMI@i$^l zogGz2iEAIax1&56S7?q}an)s7t?N{56QFfaES=A7y{Yh+=0lf4hjztMu4$FHDA1>W z!`(z8REtE|8sXDZ*WO)JVsIovM$X;xNZYPCOl2i>4@G8BH&{jQQ?527mQbT|3_;tQ zU9^@-O>AGKf8g@VZK02G$bbdfKOR%~qdnkDbzW%`K<%JaF^`kE9TZk_oeD6u-Gn%u zNfPu;Rmquz8Ygag-`=c(K^Kq8i@1Jur)xVNmfgbb5OK7oKiJ_{;#YtB>TO({_b*W# ziwFLtfZG#ij68gJRxXT1fSAcHnxGb8&%)6=dcRRbdJ-0g_E!R>CrM%IFNI5w)qV%$ zt3lI~$*J$q={+v?=l29?8A!iTe3k+h>VGA8mV$g~be5dH8E}>&`PeL<7JKtspoui6 z5~%#3Uo;E-9D|Ypn~zmymUqWkcVzyenM`rHPUkqKVRlj!BQZ1f4F>zZi(Ppy6X{a= zMejFI9*0Z~$s*Rk3nqAFfL>ifF0hmlULL^bmRRF5pYl7TY1yXTSn~sKR_y3hyanS@ zCQ=6hHqUhEm6kqh9{Dy)6Ja<{cJA;>M4zs?4|iSLEfhm-kh?c$bf5k6-&S?^Js0AH zB7IJhOH!l@#OkDAX48NpGBBF+M8B%*dS}~FOMCRiY4yzNB|48 XWo1}Mdoxo&; c&D;ZPGa82pdBhFJzLl>Y9eA@0qC7SJ1D?O@ssI20 literal 0 HcmV?d00001 diff --git a/api/apemanager/marshal.go b/api/apemanager/marshal.go new file mode 100644 index 0000000..ac6fdde --- /dev/null +++ b/api/apemanager/marshal.go @@ -0,0 +1,205 @@ +package apemanager + +import ( + apemanager "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/apemanager/grpc" + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/rpc/message" + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/util/proto" +) + +const ( + addChainReqBodyTargetField = 1 + addChainReqBodyChainField = 2 + + addChainRespBodyChainIDField = 1 + + removeChainReqBodyTargetField = 1 + removeChainReqBodyChainField = 2 + + /* + Fields for RemoveResponseBody are missed since RemoveResponseBody is empty. + */ + + listChainsReqBodyTargetField = 1 + + listChainsRespBodyChainsField = 1 +) + +func (rb *AddChainRequestBody) StableSize() (size int) { + if rb == nil { + return 0 + } + + size += proto.NestedStructureSize(addChainReqBodyTargetField, rb.target) + size += proto.NestedStructureSize(addChainReqBodyChainField, rb.chain) + + return size +} + +func (rb *AddChainRequestBody) StableMarshal(buf []byte) []byte { + if rb == nil { + return []byte{} + } + + if buf == nil { + buf = make([]byte, rb.StableSize()) + } + + var offset int + + offset += proto.NestedStructureMarshal(addChainReqBodyTargetField, buf[offset:], rb.target) + proto.NestedStructureMarshal(addChainReqBodyChainField, buf[offset:], rb.chain) + + return buf +} + +func (rb *AddChainRequestBody) Unmarshal(data []byte) error { + return message.Unmarshal(rb, data, new(apemanager.AddChainRequest_Body)) +} + +func (rb *AddChainResponseBody) StableSize() (size int) { + if rb == nil { + return 0 + } + + size += proto.BytesSize(addChainRespBodyChainIDField, rb.chainID) + + return size +} + +func (rb *AddChainResponseBody) StableMarshal(buf []byte) []byte { + if rb == nil { + return []byte{} + } + + if buf == nil { + buf = make([]byte, rb.StableSize()) + } + + var offset int + + proto.BytesMarshal(addChainRespBodyChainIDField, buf[offset:], rb.chainID) + + return buf +} + +func (rb *AddChainResponseBody) Unmarshal(data []byte) error { + return message.Unmarshal(rb, data, new(apemanager.AddChainResponse_Body)) +} + +func (rb *RemoveChainRequestBody) StableSize() (size int) { + if rb == nil { + return 0 + } + + size += proto.NestedStructureSize(addChainReqBodyTargetField, rb.target) + size += proto.BytesSize(addChainReqBodyChainField, rb.chainID) + + return size +} + +func (rb *RemoveChainRequestBody) StableMarshal(buf []byte) []byte { + if rb == nil { + return []byte{} + } + + if buf == nil { + buf = make([]byte, rb.StableSize()) + } + + var offset int + + offset += proto.NestedStructureMarshal(removeChainReqBodyTargetField, buf[offset:], rb.target) + proto.BytesMarshal(removeChainReqBodyChainField, buf[offset:], rb.chainID) + + return buf +} + +func (rb *RemoveChainRequestBody) Unmarshal(data []byte) error { + return message.Unmarshal(rb, data, new(apemanager.RemoveChainRequest_Body)) +} + +func (rb *RemoveChainResponseBody) StableSize() (size int) { + if rb == nil { + return 0 + } + + return size +} + +func (rb *RemoveChainResponseBody) StableMarshal(buf []byte) []byte { + if rb == nil { + return []byte{} + } + + if buf == nil { + buf = make([]byte, rb.StableSize()) + } + + return buf +} + +func (rb *RemoveChainResponseBody) Unmarshal(data []byte) error { + return message.Unmarshal(rb, data, new(apemanager.RemoveChainResponse_Body)) +} + +func (rb *ListChainsRequestBody) StableSize() (size int) { + if rb == nil { + return 0 + } + + size += proto.NestedStructureSize(listChainsReqBodyTargetField, rb.target) + + return size +} + +func (rb *ListChainsRequestBody) StableMarshal(buf []byte) []byte { + if rb == nil { + return []byte{} + } + + if buf == nil { + buf = make([]byte, rb.StableSize()) + } + + var offset int + proto.NestedStructureMarshal(addChainReqBodyTargetField, buf[offset:], rb.target) + + return buf +} + +func (rb *ListChainsRequestBody) Unmarshal(data []byte) error { + return message.Unmarshal(rb, data, new(apemanager.ListChainsRequest_Body)) +} + +func (rb *ListChainsResponseBody) StableSize() (size int) { + if rb == nil { + return 0 + } + + for _, chain := range rb.GetChains() { + size += proto.NestedStructureSize(listChainsRespBodyChainsField, chain) + } + + return size +} + +func (rb *ListChainsResponseBody) StableMarshal(buf []byte) []byte { + if rb == nil { + return []byte{} + } + + if buf == nil { + buf = make([]byte, rb.StableSize()) + } + + var offset int + for _, chain := range rb.GetChains() { + offset += proto.NestedStructureMarshal(listChainsRespBodyChainsField, buf[offset:], chain) + } + + return buf +} + +func (rb *ListChainsResponseBody) Unmarshal(data []byte) error { + return message.Unmarshal(rb, data, new(apemanager.ListChainsResponse_Body)) +} diff --git a/api/apemanager/message_test.go b/api/apemanager/message_test.go new file mode 100644 index 0000000..d64688c --- /dev/null +++ b/api/apemanager/message_test.go @@ -0,0 +1,26 @@ +package apemanager_test + +import ( + "testing" + + apemanagertest "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/apemanager/test" + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/rpc/message" + messagetest "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/rpc/message/test" +) + +func TestMessageConvert(t *testing.T) { + messagetest.TestRPCMessage(t, + func(empty bool) message.Message { return apemanagertest.GenerateAddChainRequestBody(empty) }, + func(empty bool) message.Message { return apemanagertest.GenerateAddChainRequest(empty) }, + func(empty bool) message.Message { return apemanagertest.GenerateAddChainResponseBody(empty) }, + func(empty bool) message.Message { return apemanagertest.GenerateAddChainResponse(empty) }, + func(empty bool) message.Message { return apemanagertest.GenerateRemoveChainRequestBody(empty) }, + func(empty bool) message.Message { return apemanagertest.GenerateRemoveChainRequest(empty) }, + func(empty bool) message.Message { return apemanagertest.GenerateRemoveChainResponseBody(empty) }, + func(empty bool) message.Message { return apemanagertest.GenerateRemoveChainResponse(empty) }, + func(empty bool) message.Message { return apemanagertest.GenerateListChainsRequestBody(empty) }, + func(empty bool) message.Message { return apemanagertest.GenerateListChainsRequest(empty) }, + func(empty bool) message.Message { return apemanagertest.GenerateListChainsResponseBody(empty) }, + func(empty bool) message.Message { return apemanagertest.GenerateListChainsResponse(empty) }, + ) +} diff --git a/api/apemanager/status.go b/api/apemanager/status.go new file mode 100644 index 0000000..3b7f435 --- /dev/null +++ b/api/apemanager/status.go @@ -0,0 +1,76 @@ +package apemanager + +import ( + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/status" + statusgrpc "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/status/grpc" +) + +// LocalizeFailStatus checks if passed global status.Code is related to ape manager failure and: +// +// then localizes the code and returns true, +// else leaves the code unchanged and returns false. +// +// Arg must be non-nil. +func LocalizeFailStatus(c *status.Code) bool { + return status.LocalizeIfInSection(c, uint32(statusgrpc.Section_SECTION_APE_MANAGER)) +} + +// GlobalizeFail globalizes local code of ape manager failure. +// +// Arg must be non-nil. +func GlobalizeFail(c *status.Code) { + c.GlobalizeSection(uint32(statusgrpc.Section_SECTION_APE_MANAGER)) +} + +const ( + // StatusAPEManagerAccessDenied is a local status.Code value for + // ACCESS_DENIED ape manager failure. + StatusAPEManagerAccessDenied status.Code = iota +) + +const ( + // detailAccessDeniedDesc is a StatusAccessDenied detail ID for + // human-readable description. + detailAccessDeniedDesc = iota +) + +// WriteAccessDeniedDesc writes human-readable description of StatusAccessDenied +// into status.Status as a detail. The status must not be nil. +// +// Existing details are expected to be ID-unique, otherwise undefined behavior. +func WriteAccessDeniedDesc(st *status.Status, desc string) { + var found bool + + st.IterateDetails(func(d *status.Detail) bool { + if d.ID() == detailAccessDeniedDesc { + found = true + d.SetValue([]byte(desc)) + } + + return found + }) + + if !found { + var d status.Detail + + d.SetID(detailAccessDeniedDesc) + d.SetValue([]byte(desc)) + + st.AppendDetails(d) + } +} + +// ReadAccessDeniedDesc looks up for status detail with human-readable description +// of StatusAccessDenied. Returns empty string if detail is missing. +func ReadAccessDeniedDesc(st status.Status) (desc string) { + st.IterateDetails(func(d *status.Detail) bool { + if d.ID() == detailAccessDeniedDesc { + desc = string(d.Value()) + return true + } + + return false + }) + + return +} diff --git a/api/apemanager/status_test.go b/api/apemanager/status_test.go new file mode 100644 index 0000000..af6cd66 --- /dev/null +++ b/api/apemanager/status_test.go @@ -0,0 +1,30 @@ +package apemanager_test + +import ( + "testing" + + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/apemanager" + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/status" + statustest "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/status/test" + "github.com/stretchr/testify/require" +) + +func TestStatusCodes(t *testing.T) { + statustest.TestCodes(t, apemanager.LocalizeFailStatus, apemanager.GlobalizeFail, + apemanager.StatusAPEManagerAccessDenied, 5120, + ) +} + +func TestAccessDeniedDesc(t *testing.T) { + var st status.Status + + require.Empty(t, apemanager.ReadAccessDeniedDesc(st)) + + const desc = "some description" + + apemanager.WriteAccessDeniedDesc(&st, desc) + require.Equal(t, desc, apemanager.ReadAccessDeniedDesc(st)) + + apemanager.WriteAccessDeniedDesc(&st, desc+"1") + require.Equal(t, desc+"1", apemanager.ReadAccessDeniedDesc(st)) +} diff --git a/api/apemanager/test/generate.go b/api/apemanager/test/generate.go new file mode 100644 index 0000000..47427d7 --- /dev/null +++ b/api/apemanager/test/generate.go @@ -0,0 +1,143 @@ +package apemanagertest + +import ( + apetest "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/ape/test" + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/apemanager" + sessiontest "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/session/test" +) + +func generateChainID(empty bool) []byte { + if empty { + return []byte{} + } + + return []byte("616c6c6f774f626a476574436e72") +} + +func GenerateAddChainRequestBody(empty bool) *apemanager.AddChainRequestBody { + m := new(apemanager.AddChainRequestBody) + + if !empty { + m.SetTarget(apetest.GenerateChainTarget(empty)) + m.SetChain(apetest.GenerateRawChain(empty)) + } + + return m +} + +func GenerateAddChainRequest(empty bool) *apemanager.AddChainRequest { + m := new(apemanager.AddChainRequest) + + if !empty { + m.SetBody(GenerateAddChainRequestBody(empty)) + m.SetMetaHeader(sessiontest.GenerateRequestMetaHeader(empty)) + m.SetVerificationHeader(sessiontest.GenerateRequestVerificationHeader(empty)) + } + + return m +} + +func GenerateAddChainResponseBody(empty bool) *apemanager.AddChainResponseBody { + m := new(apemanager.AddChainResponseBody) + + if !empty { + m.SetChainID(generateChainID(empty)) + } + + return m +} + +func GenerateAddChainResponse(empty bool) *apemanager.AddChainResponse { + m := new(apemanager.AddChainResponse) + + if !empty { + m.SetBody(GenerateAddChainResponseBody(empty)) + m.SetMetaHeader(sessiontest.GenerateResponseMetaHeader(empty)) + m.SetVerificationHeader(sessiontest.GenerateResponseVerificationHeader(empty)) + } + + return m +} + +func GenerateRemoveChainRequestBody(empty bool) *apemanager.RemoveChainRequestBody { + m := new(apemanager.RemoveChainRequestBody) + + if !empty { + m.SetChainID(generateChainID(empty)) + m.SetTarget(apetest.GenerateChainTarget(empty)) + } + + return m +} + +func GenerateRemoveChainRequest(empty bool) *apemanager.RemoveChainRequest { + m := new(apemanager.RemoveChainRequest) + + if !empty { + m.SetBody(GenerateRemoveChainRequestBody(empty)) + m.SetMetaHeader(sessiontest.GenerateRequestMetaHeader(empty)) + m.SetVerificationHeader(sessiontest.GenerateRequestVerificationHeader(empty)) + } + + return m +} + +func GenerateRemoveChainResponseBody(_ bool) *apemanager.RemoveChainResponseBody { + return new(apemanager.RemoveChainResponseBody) +} + +func GenerateRemoveChainResponse(empty bool) *apemanager.RemoveChainResponse { + m := new(apemanager.RemoveChainResponse) + + if !empty { + m.SetBody(GenerateRemoveChainResponseBody(empty)) + m.SetMetaHeader(sessiontest.GenerateResponseMetaHeader(empty)) + m.SetVerificationHeader(sessiontest.GenerateResponseVerificationHeader(empty)) + } + + return m +} + +func GenerateListChainsRequestBody(empty bool) *apemanager.ListChainsRequestBody { + m := new(apemanager.ListChainsRequestBody) + + if !empty { + m.SetTarget(apetest.GenerateChainTarget(empty)) + } + + return m +} + +func GenerateListChainsRequest(empty bool) *apemanager.ListChainsRequest { + m := new(apemanager.ListChainsRequest) + + if !empty { + m.SetBody(GenerateListChainsRequestBody(empty)) + m.SetMetaHeader(sessiontest.GenerateRequestMetaHeader(empty)) + m.SetVerificationHeader(sessiontest.GenerateRequestVerificationHeader(empty)) + } + + return m +} + +func GenerateListChainsResponseBody(empty bool) *apemanager.ListChainsResponseBody { + m := new(apemanager.ListChainsResponseBody) + + if !empty { + m.SetChains(apetest.GenerateRawChains(empty, 10)) + } + + return m +} + +func GenerateListChainsResponse(empty bool) *apemanager.ListChainsResponse { + m := new(apemanager.ListChainsResponse) + + if !empty { + m.SetBody(GenerateListChainsResponseBody(empty)) + m.SetMetaHeader(sessiontest.GenerateResponseMetaHeader(empty)) + m.SetVerificationHeader(sessiontest.GenerateResponseVerificationHeader(empty)) + } + + return m +} diff --git a/api/apemanager/types.go b/api/apemanager/types.go new file mode 100644 index 0000000..077488b --- /dev/null +++ b/api/apemanager/types.go @@ -0,0 +1,226 @@ +package apemanager + +import ( + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/ape" + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/session" +) + +type AddChainRequest struct { + body *AddChainRequestBody + + session.RequestHeaders +} + +func (r *AddChainRequest) SetBody(body *AddChainRequestBody) { + r.body = body +} + +func (r *AddChainRequest) GetBody() *AddChainRequestBody { + if r == nil { + return nil + } + + return r.body +} + +type AddChainRequestBody struct { + target *ape.ChainTarget + + chain *ape.Chain +} + +func (rb *AddChainRequestBody) SetTarget(target *ape.ChainTarget) { + rb.target = target +} + +func (rb *AddChainRequestBody) GetTarget() *ape.ChainTarget { + if rb == nil { + return nil + } + + return rb.target +} + +func (rb *AddChainRequestBody) SetChain(chain *ape.Chain) { + rb.chain = chain +} + +func (rb *AddChainRequestBody) GetChain() *ape.Chain { + if rb == nil { + return nil + } + + return rb.chain +} + +type AddChainResponse struct { + body *AddChainResponseBody + + session.ResponseHeaders +} + +func (r *AddChainResponse) SetBody(body *AddChainResponseBody) { + r.body = body +} + +func (r *AddChainResponse) GetBody() *AddChainResponseBody { + if r == nil { + return nil + } + + return r.body +} + +type AddChainResponseBody struct { + chainID []byte +} + +func (rb *AddChainResponseBody) SetChainID(chainID []byte) { + rb.chainID = chainID +} + +func (rb *AddChainResponseBody) GetChainID() []byte { + if rb == nil { + return nil + } + + return rb.chainID +} + +type RemoveChainRequest struct { + body *RemoveChainRequestBody + + session.RequestHeaders +} + +func (r *RemoveChainRequest) SetBody(body *RemoveChainRequestBody) { + r.body = body +} + +func (r *RemoveChainRequest) GetBody() *RemoveChainRequestBody { + if r == nil { + return nil + } + + return r.body +} + +type RemoveChainRequestBody struct { + target *ape.ChainTarget + + chainID []byte +} + +func (rb *RemoveChainRequestBody) SetTarget(target *ape.ChainTarget) { + rb.target = target +} + +func (rb *RemoveChainRequestBody) GetTarget() *ape.ChainTarget { + if rb == nil { + return nil + } + + return rb.target +} + +func (rb *RemoveChainRequestBody) SetChainID(chainID []byte) { + rb.chainID = chainID +} + +func (rb *RemoveChainRequestBody) GetChainID() []byte { + if rb == nil { + return nil + } + + return rb.chainID +} + +type RemoveChainResponse struct { + body *RemoveChainResponseBody + + session.ResponseHeaders +} + +type RemoveChainResponseBody struct{} + +func (r *RemoveChainResponse) SetBody(body *RemoveChainResponseBody) { + r.body = body +} + +func (r *RemoveChainResponse) GetBody() *RemoveChainResponseBody { + if r == nil { + return nil + } + + return r.body +} + +type ListChainsRequest struct { + body *ListChainsRequestBody + + session.RequestHeaders +} + +func (r *ListChainsRequest) SetBody(body *ListChainsRequestBody) { + r.body = body +} + +func (r *ListChainsRequest) GetBody() *ListChainsRequestBody { + if r == nil { + return nil + } + + return r.body +} + +type ListChainsRequestBody struct { + target *ape.ChainTarget +} + +func (rb *ListChainsRequestBody) SetTarget(target *ape.ChainTarget) { + rb.target = target +} + +func (rb *ListChainsRequestBody) GetTarget() *ape.ChainTarget { + if rb == nil { + return nil + } + + return rb.target +} + +type ListChainsResponse struct { + body *ListChainsResponseBody + + session.ResponseHeaders +} + +func (r *ListChainsResponse) SetBody(body *ListChainsResponseBody) { + r.body = body +} + +func (r *ListChainsResponse) GetBody() *ListChainsResponseBody { + if r == nil { + return nil + } + + return r.body +} + +type ListChainsResponseBody struct { + chains []*ape.Chain + + session.RequestHeaders +} + +func (r *ListChainsResponseBody) SetChains(chains []*ape.Chain) { + r.chains = chains +} + +func (r *ListChainsResponseBody) GetChains() []*ape.Chain { + if r == nil { + return nil + } + + return r.chains +} diff --git a/api/container/attributes.go b/api/container/attributes.go new file mode 100644 index 0000000..288d048 --- /dev/null +++ b/api/container/attributes.go @@ -0,0 +1,90 @@ +package container + +// SysAttributePrefix is a prefix of key to system attribute. +const SysAttributePrefix = "__SYSTEM__" + +const ( + // SysAttributeName is a string of human-friendly container name registered as the domain in NNS contract. + SysAttributeName = SysAttributePrefix + "NAME" + + // SysAttributeZone is a string of zone for container name. + SysAttributeZone = SysAttributePrefix + "ZONE" + + // SysAttributeHomomorphicHashing is a container's homomorphic hashing state. + SysAttributeHomomorphicHashing = SysAttributePrefix + "DISABLE_HOMOMORPHIC_HASHING" +) + +// SysAttributePrefixNeoFS is a prefix of key to system attribute. +// Deprecated: use SysAttributePrefix. +const SysAttributePrefixNeoFS = "__NEOFS__" + +const ( + // SysAttributeNameNeoFS is a string of human-friendly container name registered as the domain in NNS contract. + // Deprecated: use SysAttributeName. + SysAttributeNameNeoFS = SysAttributePrefixNeoFS + "NAME" + + // SysAttributeZoneNeoFS is a string of zone for container name. + // Deprecated: use SysAttributeZone. + SysAttributeZoneNeoFS = SysAttributePrefixNeoFS + "ZONE" + + // SysAttributeHomomorphicHashingNeoFS is a container's homomorphic hashing state. + // Deprecated: use SysAttributeHomomorphicHashing. + SysAttributeHomomorphicHashingNeoFS = SysAttributePrefixNeoFS + "DISABLE_HOMOMORPHIC_HASHING" +) + +// SysAttributeZoneDefault is a default value for SysAttributeZone attribute. +const SysAttributeZoneDefault = "container" + +const disabledHomomorphicHashingValue = "true" + +// HomomorphicHashingState returns container's homomorphic +// hashing state: +// - true if hashing is enabled; +// - false if hashing is disabled. +// +// All container's attributes must be unique, otherwise behavior +// is undefined. +// +// See also SetHomomorphicHashingState. +func (c Container) HomomorphicHashingState() bool { + for i := range c.attr { + if c.attr[i].GetKey() == SysAttributeHomomorphicHashing || c.attr[i].GetKey() == SysAttributeHomomorphicHashingNeoFS { + return c.attr[i].GetValue() != disabledHomomorphicHashingValue + } + } + + return true +} + +// SetHomomorphicHashingState sets homomorphic hashing state for +// container. +// +// All container's attributes must be unique, otherwise behavior +// is undefined. +// +// See also HomomorphicHashingState. +func (c *Container) SetHomomorphicHashingState(enable bool) { + for i := range c.attr { + if c.attr[i].GetKey() == SysAttributeHomomorphicHashing || c.attr[i].GetKey() == SysAttributeHomomorphicHashingNeoFS { + if enable { + // approach without allocation/waste + // coping works since the attributes + // order is not important + c.attr[i] = c.attr[len(c.attr)-1] + c.attr = c.attr[:len(c.attr)-1] + } else { + c.attr[i].SetValue(disabledHomomorphicHashingValue) + } + + return + } + } + + if !enable { + attr := Attribute{} + attr.SetKey(SysAttributeHomomorphicHashing) + attr.SetValue(disabledHomomorphicHashingValue) + + c.attr = append(c.attr, attr) + } +} diff --git a/api/container/attributes_test.go b/api/container/attributes_test.go new file mode 100644 index 0000000..2b00d33 --- /dev/null +++ b/api/container/attributes_test.go @@ -0,0 +1,59 @@ +package container_test + +import ( + "testing" + + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/container" + containertest "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/container/test" + "github.com/stretchr/testify/require" +) + +func TestContainer_HomomorphicHashingDisabled(t *testing.T) { + cnr := containertest.GenerateContainer(false) + + t.Run("defaults", func(t *testing.T) { + require.True(t, cnr.HomomorphicHashingState()) + }) + + t.Run("disabled", func(t *testing.T) { + attr := container.Attribute{} + attr.SetKey(container.SysAttributeHomomorphicHashing) + attr.SetValue("NOT_true") + + cnr.SetAttributes(append(cnr.GetAttributes(), attr)) + require.True(t, cnr.HomomorphicHashingState()) + + attr.SetValue("true") + + cnr.SetAttributes([]container.Attribute{attr}) + require.False(t, cnr.HomomorphicHashingState()) + }) +} + +func TestContainer_SetHomomorphicHashingState(t *testing.T) { + cnr := containertest.GenerateContainer(false) + attrs := cnr.GetAttributes() + attrLen := len(attrs) + + cnr.SetHomomorphicHashingState(true) + + // enabling hashing should not add any new attributes + require.Equal(t, attrLen, len(cnr.GetAttributes())) + require.True(t, cnr.HomomorphicHashingState()) + + cnr.SetHomomorphicHashingState(false) + + // disabling hashing should add exactly one attribute + require.Equal(t, attrLen+1, len(cnr.GetAttributes())) + require.False(t, cnr.HomomorphicHashingState()) + + cnr.SetHomomorphicHashingState(true) + + // enabling hashing should remove 1 attribute if + // hashing was disabled before + require.Equal(t, attrLen, len(cnr.GetAttributes())) + require.True(t, cnr.HomomorphicHashingState()) + + // hashing operations should not change any other attributes + require.ElementsMatch(t, attrs, cnr.GetAttributes()) +} diff --git a/api/container/convert.go b/api/container/convert.go new file mode 100644 index 0000000..c91bdfd --- /dev/null +++ b/api/container/convert.go @@ -0,0 +1,764 @@ +package container + +import ( + container "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/container/grpc" + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/netmap" + netmapGRPC "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/netmap/grpc" + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/refs" + refsGRPC "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/refs/grpc" + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/rpc/grpc" + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/rpc/message" + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/session" + sessionGRPC "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/session/grpc" +) + +func (a *Attribute) ToGRPCMessage() grpc.Message { + var m *container.Container_Attribute + + if a != nil { + m = new(container.Container_Attribute) + + m.SetKey(a.key) + m.SetValue(a.val) + } + + return m +} + +func (a *Attribute) FromGRPCMessage(m grpc.Message) error { + v, ok := m.(*container.Container_Attribute) + if !ok { + return message.NewUnexpectedMessageType(m, v) + } + + a.key = v.GetKey() + a.val = v.GetValue() + + return nil +} + +func AttributesToGRPC(xs []Attribute) (res []container.Container_Attribute) { + if xs != nil { + res = make([]container.Container_Attribute, 0, len(xs)) + + for i := range xs { + res = append(res, *xs[i].ToGRPCMessage().(*container.Container_Attribute)) + } + } + + return +} + +func AttributesFromGRPC(xs []container.Container_Attribute) (res []Attribute, err error) { + if xs != nil { + res = make([]Attribute, len(xs)) + + for i := range xs { + err = res[i].FromGRPCMessage(&xs[i]) + if err != nil { + return + } + } + } + + return +} + +func (c *Container) ToGRPCMessage() grpc.Message { + var m *container.Container + + if c != nil { + m = new(container.Container) + + m.SetVersion(c.version.ToGRPCMessage().(*refsGRPC.Version)) + m.SetOwnerId(c.ownerID.ToGRPCMessage().(*refsGRPC.OwnerID)) + m.SetPlacementPolicy(c.policy.ToGRPCMessage().(*netmapGRPC.PlacementPolicy)) + m.SetAttributes(AttributesToGRPC(c.attr)) + m.SetBasicAcl(c.basicACL) + m.SetNonce(c.nonce) + } + + return m +} + +func (c *Container) FromGRPCMessage(m grpc.Message) error { + v, ok := m.(*container.Container) + if !ok { + return message.NewUnexpectedMessageType(m, v) + } + + var err error + + version := v.GetVersion() + if version == nil { + c.version = nil + } else { + if c.version == nil { + c.version = new(refs.Version) + } + + err = c.version.FromGRPCMessage(version) + if err != nil { + return err + } + } + + ownerID := v.GetOwnerId() + if ownerID == nil { + c.ownerID = nil + } else { + if c.ownerID == nil { + c.ownerID = new(refs.OwnerID) + } + + err = c.ownerID.FromGRPCMessage(ownerID) + if err != nil { + return err + } + } + + policy := v.GetPlacementPolicy() + if policy == nil { + c.policy = nil + } else { + if c.policy == nil { + c.policy = new(netmap.PlacementPolicy) + } + + err = c.policy.FromGRPCMessage(policy) + if err != nil { + return err + } + } + + c.attr, err = AttributesFromGRPC(v.GetAttributes()) + if err != nil { + return err + } + + c.basicACL = v.GetBasicAcl() + c.nonce = v.GetNonce() + + return nil +} + +func toSignatureRFC6979(s *refs.Signature) *refsGRPC.SignatureRFC6979 { + var res *refsGRPC.SignatureRFC6979 + + if s != nil { + res = new(refsGRPC.SignatureRFC6979) + res.SetKey(s.GetKey()) + res.SetSign(s.GetSign()) + } + + return res +} + +func (r *PutRequestBody) ToGRPCMessage() grpc.Message { + var m *container.PutRequest_Body + + if r != nil { + m = new(container.PutRequest_Body) + + m.SetContainer(r.cnr.ToGRPCMessage().(*container.Container)) + m.SetSignature(toSignatureRFC6979(r.sig)) + } + + return m +} + +func (r *PutRequestBody) FromGRPCMessage(m grpc.Message) error { + v, ok := m.(*container.PutRequest_Body) + if !ok { + return message.NewUnexpectedMessageType(m, v) + } + + var err error + + cnr := v.GetContainer() + if cnr == nil { + r.cnr = nil + } else { + if r.cnr == nil { + r.cnr = new(Container) + } + + err = r.cnr.FromGRPCMessage(cnr) + if err != nil { + return err + } + } + + sig := v.GetSignature() + if sig == nil { + r.sig = nil + } else { + if r.sig == nil { + r.sig = new(refs.Signature) + } + + r.sig.SetKey(sig.GetKey()) + r.sig.SetSign(sig.GetSign()) + } + + return err +} + +func (r *PutRequest) ToGRPCMessage() grpc.Message { + var m *container.PutRequest + + if r != nil { + m = new(container.PutRequest) + + m.SetBody(r.body.ToGRPCMessage().(*container.PutRequest_Body)) + r.RequestHeaders.ToMessage(m) + } + + return m +} + +func (r *PutRequest) FromGRPCMessage(m grpc.Message) error { + v, ok := m.(*container.PutRequest) + if !ok { + return message.NewUnexpectedMessageType(m, v) + } + + var err error + + body := v.GetBody() + if body == nil { + r.body = nil + } else { + if r.body == nil { + r.body = new(PutRequestBody) + } + + err = r.body.FromGRPCMessage(body) + if err != nil { + return err + } + } + + return r.RequestHeaders.FromMessage(v) +} + +func (r *PutResponseBody) ToGRPCMessage() grpc.Message { + var m *container.PutResponse_Body + + if r != nil { + m = new(container.PutResponse_Body) + + m.SetContainerId(r.cid.ToGRPCMessage().(*refsGRPC.ContainerID)) + } + + return m +} + +func (r *PutResponseBody) FromGRPCMessage(m grpc.Message) error { + v, ok := m.(*container.PutResponse_Body) + if !ok { + return message.NewUnexpectedMessageType(m, v) + } + + var err error + + cid := v.GetContainerId() + if cid == nil { + r.cid = nil + } else { + if r.cid == nil { + r.cid = new(refs.ContainerID) + } + + err = r.cid.FromGRPCMessage(cid) + } + + return err +} + +func (r *PutResponse) ToGRPCMessage() grpc.Message { + var m *container.PutResponse + + if r != nil { + m = new(container.PutResponse) + + m.SetBody(r.body.ToGRPCMessage().(*container.PutResponse_Body)) + r.ResponseHeaders.ToMessage(m) + } + + return m +} + +func (r *PutResponse) FromGRPCMessage(m grpc.Message) error { + v, ok := m.(*container.PutResponse) + if !ok { + return message.NewUnexpectedMessageType(m, v) + } + + var err error + + body := v.GetBody() + if body == nil { + r.body = nil + } else { + if r.body == nil { + r.body = new(PutResponseBody) + } + + err = r.body.FromGRPCMessage(body) + if err != nil { + return err + } + } + + return r.ResponseHeaders.FromMessage(v) +} + +func (r *GetRequestBody) ToGRPCMessage() grpc.Message { + var m *container.GetRequest_Body + + if r != nil { + m = new(container.GetRequest_Body) + + m.SetContainerId(r.cid.ToGRPCMessage().(*refsGRPC.ContainerID)) + } + + return m +} + +func (r *GetRequestBody) FromGRPCMessage(m grpc.Message) error { + v, ok := m.(*container.GetRequest_Body) + if !ok { + return message.NewUnexpectedMessageType(m, v) + } + + var err error + + cid := v.GetContainerId() + if cid == nil { + r.cid = nil + } else { + if r.cid == nil { + r.cid = new(refs.ContainerID) + } + + err = r.cid.FromGRPCMessage(cid) + } + + return err +} + +func (r *GetRequest) ToGRPCMessage() grpc.Message { + var m *container.GetRequest + + if r != nil { + m = new(container.GetRequest) + + m.SetBody(r.body.ToGRPCMessage().(*container.GetRequest_Body)) + r.RequestHeaders.ToMessage(m) + } + + return m +} + +func (r *GetRequest) FromGRPCMessage(m grpc.Message) error { + v, ok := m.(*container.GetRequest) + if !ok { + return message.NewUnexpectedMessageType(m, v) + } + + var err error + + body := v.GetBody() + if body == nil { + r.body = nil + } else { + if r.body == nil { + r.body = new(GetRequestBody) + } + + err = r.body.FromGRPCMessage(body) + if err != nil { + return err + } + } + + return r.RequestHeaders.FromMessage(v) +} + +func (r *GetResponseBody) ToGRPCMessage() grpc.Message { + var m *container.GetResponse_Body + + if r != nil { + m = new(container.GetResponse_Body) + + m.SetContainer(r.cnr.ToGRPCMessage().(*container.Container)) + m.SetSessionToken(r.token.ToGRPCMessage().(*sessionGRPC.SessionToken)) + m.SetSignature(toSignatureRFC6979(r.sig)) + } + + return m +} + +func (r *GetResponseBody) FromGRPCMessage(m grpc.Message) error { + v, ok := m.(*container.GetResponse_Body) + if !ok { + return message.NewUnexpectedMessageType(m, v) + } + + var err error + + cnr := v.GetContainer() + if cnr == nil { + r.cnr = nil + } else { + if r.cnr == nil { + r.cnr = new(Container) + } + + err = r.cnr.FromGRPCMessage(cnr) + } + + sig := v.GetSignature() + if sig == nil { + r.sig = nil + } else { + if r.sig == nil { + r.sig = new(refs.Signature) + } + + r.sig.SetKey(sig.GetKey()) + r.sig.SetSign(sig.GetSign()) + } + + token := v.GetSessionToken() + if token == nil { + r.token = nil + } else { + if r.token == nil { + r.token = new(session.Token) + } + + err = r.token.FromGRPCMessage(token) + } + + return err +} + +func (r *GetResponse) ToGRPCMessage() grpc.Message { + var m *container.GetResponse + + if r != nil { + m = new(container.GetResponse) + + m.SetBody(r.body.ToGRPCMessage().(*container.GetResponse_Body)) + r.ResponseHeaders.ToMessage(m) + } + + return m +} + +func (r *GetResponse) FromGRPCMessage(m grpc.Message) error { + v, ok := m.(*container.GetResponse) + if !ok { + return message.NewUnexpectedMessageType(m, v) + } + + var err error + + body := v.GetBody() + if body == nil { + r.body = nil + } else { + if r.body == nil { + r.body = new(GetResponseBody) + } + + err = r.body.FromGRPCMessage(body) + if err != nil { + return err + } + } + + return r.ResponseHeaders.FromMessage(v) +} + +func (r *DeleteRequestBody) ToGRPCMessage() grpc.Message { + var m *container.DeleteRequest_Body + + if r != nil { + m = new(container.DeleteRequest_Body) + + m.SetContainerId(r.cid.ToGRPCMessage().(*refsGRPC.ContainerID)) + m.SetSignature(toSignatureRFC6979(r.sig)) + } + + return m +} + +func (r *DeleteRequestBody) FromGRPCMessage(m grpc.Message) error { + v, ok := m.(*container.DeleteRequest_Body) + if !ok { + return message.NewUnexpectedMessageType(m, v) + } + + var err error + + cid := v.GetContainerId() + if cid == nil { + r.cid = nil + } else { + if r.cid == nil { + r.cid = new(refs.ContainerID) + } + + err = r.cid.FromGRPCMessage(cid) + if err != nil { + return err + } + } + + sig := v.GetSignature() + if sig == nil { + r.sig = nil + } else { + if r.sig == nil { + r.sig = new(refs.Signature) + } + + r.sig.SetKey(sig.GetKey()) + r.sig.SetSign(sig.GetSign()) + } + + return err +} + +func (r *DeleteRequest) ToGRPCMessage() grpc.Message { + var m *container.DeleteRequest + + if r != nil { + m = new(container.DeleteRequest) + + m.SetBody(r.body.ToGRPCMessage().(*container.DeleteRequest_Body)) + r.RequestHeaders.ToMessage(m) + } + + return m +} + +func (r *DeleteRequest) FromGRPCMessage(m grpc.Message) error { + v, ok := m.(*container.DeleteRequest) + if !ok { + return message.NewUnexpectedMessageType(m, v) + } + + var err error + + body := v.GetBody() + if body == nil { + r.body = nil + } else { + if r.body == nil { + r.body = new(DeleteRequestBody) + } + + err = r.body.FromGRPCMessage(body) + if err != nil { + return err + } + } + + return r.RequestHeaders.FromMessage(v) +} + +func (r *DeleteResponseBody) ToGRPCMessage() grpc.Message { + var m *container.DeleteResponse_Body + + if r != nil { + m = new(container.DeleteResponse_Body) + } + + return m +} + +func (r *DeleteResponseBody) FromGRPCMessage(m grpc.Message) error { + v, ok := m.(*container.DeleteResponse_Body) + if !ok { + return message.NewUnexpectedMessageType(m, v) + } + + return nil +} + +func (r *DeleteResponse) ToGRPCMessage() grpc.Message { + var m *container.DeleteResponse + + if r != nil { + m = new(container.DeleteResponse) + + m.SetBody(r.body.ToGRPCMessage().(*container.DeleteResponse_Body)) + r.ResponseHeaders.ToMessage(m) + } + + return m +} + +func (r *DeleteResponse) FromGRPCMessage(m grpc.Message) error { + v, ok := m.(*container.DeleteResponse) + if !ok { + return message.NewUnexpectedMessageType(m, v) + } + + var err error + + body := v.GetBody() + if body == nil { + r.body = nil + } else { + if r.body == nil { + r.body = new(DeleteResponseBody) + } + + err = r.body.FromGRPCMessage(body) + if err != nil { + return err + } + } + + return r.ResponseHeaders.FromMessage(v) +} + +func (r *ListRequestBody) ToGRPCMessage() grpc.Message { + var m *container.ListRequest_Body + + if r != nil { + m = new(container.ListRequest_Body) + + m.SetOwnerId(r.ownerID.ToGRPCMessage().(*refsGRPC.OwnerID)) + } + + return m +} + +func (r *ListRequestBody) FromGRPCMessage(m grpc.Message) error { + v, ok := m.(*container.ListRequest_Body) + if !ok { + return message.NewUnexpectedMessageType(m, v) + } + + var err error + + ownerID := v.GetOwnerId() + if ownerID == nil { + r.ownerID = nil + } else { + if r.ownerID == nil { + r.ownerID = new(refs.OwnerID) + } + + err = r.ownerID.FromGRPCMessage(ownerID) + } + + return err +} + +func (r *ListRequest) ToGRPCMessage() grpc.Message { + var m *container.ListRequest + + if r != nil { + m = new(container.ListRequest) + + m.SetBody(r.body.ToGRPCMessage().(*container.ListRequest_Body)) + r.RequestHeaders.ToMessage(m) + } + + return m +} + +func (r *ListRequest) FromGRPCMessage(m grpc.Message) error { + v, ok := m.(*container.ListRequest) + if !ok { + return message.NewUnexpectedMessageType(m, v) + } + + var err error + + body := v.GetBody() + if body == nil { + r.body = nil + } else { + if r.body == nil { + r.body = new(ListRequestBody) + } + + err = r.body.FromGRPCMessage(body) + if err != nil { + return err + } + } + + return r.RequestHeaders.FromMessage(v) +} + +func (r *ListResponseBody) ToGRPCMessage() grpc.Message { + var m *container.ListResponse_Body + + if r != nil { + m = new(container.ListResponse_Body) + + m.SetContainerIds(refs.ContainerIDsToGRPCMessage(r.cidList)) + } + + return m +} + +func (r *ListResponseBody) FromGRPCMessage(m grpc.Message) error { + v, ok := m.(*container.ListResponse_Body) + if !ok { + return message.NewUnexpectedMessageType(m, v) + } + + var err error + + r.cidList, err = refs.ContainerIDsFromGRPCMessage(v.GetContainerIds()) + + return err +} + +func (r *ListResponse) ToGRPCMessage() grpc.Message { + var m *container.ListResponse + + if r != nil { + m = new(container.ListResponse) + + m.SetBody(r.body.ToGRPCMessage().(*container.ListResponse_Body)) + r.ResponseHeaders.ToMessage(m) + } + + return m +} + +func (r *ListResponse) FromGRPCMessage(m grpc.Message) error { + v, ok := m.(*container.ListResponse) + if !ok { + return message.NewUnexpectedMessageType(m, v) + } + + var err error + + body := v.GetBody() + if body == nil { + r.body = nil + } else { + if r.body == nil { + r.body = new(ListResponseBody) + } + + err = r.body.FromGRPCMessage(body) + if err != nil { + return err + } + } + + return r.ResponseHeaders.FromMessage(v) +} diff --git a/api/container/grpc/service_frostfs.pb.go b/api/container/grpc/service_frostfs.pb.go new file mode 100644 index 0000000000000000000000000000000000000000..07438240cec6d58629892f69a6f99c4f93142bbd GIT binary patch literal 76792 zcmeHQ>yH~ZlK-svE9e1|NWO_CPJlbC7d~JUXZL2|*n!8{{g4E^8mVRXT93Kzp4S@w z-?xg7V)4-rY1De5gCw@PS!A(TMOOV-WS^glAC{9uyiXQMHY$>dxOx!R*|J!UzlLXD zzh8bm&6asF&4=RU8}a(hJMrVoi+975lk3s= +)9N`d<1B%NO`vqJPwdjHCo z3vrq(#>+{%cz=#RPfvQ&c_B{WfAH7$+4WeQzE6vxsXkpym*?-Yo8*Vt@@DdCbak%Z ze4S4|0w3q2>-0QJrujM2>MEYCQAeKSdAeMfYOa^d*&4On6zS|7X@L_dueIwyVM8TG zSdXgb+|+^ l{Mp^!0G=n|~@j~>Uddv>`i*z;!$$nYPiLjCkJ22#pQ19x=cq0wS zK;Gx);<6ZB&63OX--*Z)5Yr+T#fL=X_+U8|_aa>=Q@^^Iis>?&kBTACDN-C5kZdmQ z(&B@n8_g3jO_SL~+>T~2h~mTOHnFd+(#0r)F@@@fC)1n7SoH5jAVtPie<1pjnRHPM zPzh3G>ctBIg2tac#kKh6r1$xxCrkd<3&rv9HRw<>xs>_?Y+gd6{by(5en{E_Lsd`m zEs~Jh0EuoADN$8Gih9JPilCrilA9&-WKk$-mgx}mDaodzaiuK0y8Dy75WoI@^-v@O z{U^#{E}lcTVXCp(^E3YXOH%ZK c{V1U=qFhw;J;-i zg@_Z`bS$i>|BTj&7LeQ*u-q;)F-sO$!$AD6_(s|k{5ZoZV63NOFr)iI63Iefav(5# z$%Jl90@m%}kJ$9Ie>xs57Rv%=mC-~jpb9-Lz 8R+@2;}k^>j06j*a4#JLkFVG>dsI>-5IuS z{`K EC|2c!K&6_oRzQ|Jr}^*)Pa1{14% ztxCc)I)X>|4*nZ27tqW##8l}$y5hhOjNj>R*2(_u^f|cth5^m8BESZi76(!WexL|6 z(=xBQ@$GAQB_%EA)e+)n5Tdu}mnt(Y0V}WbQYCS{phX^509d#|Y;vJd0oU*qMrNOI z>3c4)wA5^Bm6Y&HTs*XHXG)1-sjQk3!#ZFUf67}`TAaUIUcX9ilNst|x)@&M5ME@` z0R>Hz4_<>RgKg>oPc4PiKqvxpZ+ $#Zn#}2QcqWy;*OUGkoBHuhZ>m>NT8NNMcx;zQ8qIW}-Z47^;whGw z_KR-QDH%Yd<>;L4a1kxhgd?&h&XFEdC2=i3cxJ~FZ3<^DPdT6f*Nx9;cOGECumTAk zm?+o}FwJxNQ)2*EfHxp!0Jz8K|G;mRVZL0{o((l7;k;7xns`n`SjFFxES)~+O57^s zt8_dn0ASJubQQOZXiXSmbp*mvcuOux$A>yR3Ys0HxHOduS7?g8gd7HKqYbEvpwSJ1 z2CHC=Xud;mf-4(P>A9WZ6K_G8(HN5nt_~35%x1Yjph5ySI6DBSD;F5Vd=TIf7gH>N zr~!nB?*ZOPbNE}1fW~M6Wo&hqh^v4@1fh^2fwH67YzaD^OblG~R>Mo9#RCQ~Fkd4q za)S^EG=BFXg`jVeLaYcuBNE>UL xhaz!g?mBn!h6VM8lGSWH?`2n+OyB&|e)QN3KNr365B5MrQG^an!=zsV(7!fFmR z^-QV{j1KlDQ}`&rXm;SByNC(p8|qp)L^lwwHuX`D3?LQpxwSU(UTe3Q;0ve$^+yI~ zEG#hZ{m@`NIBnpgbccwf4aBBFG`Uc+fbf2ZPio*U2RgGe7Kel&rlJM0rZx?p$=KAH z%iQ|-ELT(GcUytu27;D8qpAmv#H|3rp_ZVRh!b;b_!~CjKul6w73${Uhp_JTP=g42 zlWH6e5c0cO*v;yB>+-NoIa0BYb=-g+d&Ff8WMm0#WI?LDNlASt0ZMMct;>ox=17ab zh{;R7cl`4}ZoG>2BnMFu{Hxc`QrL1J$ejpw($MPwo4Dr_DXquKf*Z8}KqZz&h|f1W zY(lLWAj=YvQnF}u%>a}U4XtF@*3eKez_z6^_++G2i4^b!tEMY__=C=b>QmV%URU#O zajKwWAqTpHoJx0*xm%^cm3G%^VJq^mBwn>!0G4 #Eb_rv<@3~uuZv# zQi~0vNk_n0Z?#3Isb|1p@11l~Y9L~q6JX-^&|zz=5e`aRNr#h?GEF$L)*tf4&9aU& zX_b!CR?cBqwh5=)?m4e8nFjj-@a&Ozc76@}AW3Nk+{Hw1w$cv4#Y>aPhFcLb+45qt zI*82LwJ@&>xKK<54gtZuQMv|zaycDuhM?$*_Vx{2<~?g0Vzyc^0aex>KEr=K!0pk1 zn$tJ0@&kj_oUUvdTw~+HLol{icQL7fvNY7@Z8Do6y?P|M36&;LI07IF)4<^ZTC86y zz*=m?Z~-J=TjTTvEkoU z6z2>J&Ahva<>H)>igXd(8W9~+#**NLPf9_w!e2HCPnbS$D>|u8w_!R(?z{O*qkJ1x z0Sgd4Y8JMZCfukx!hp}jO1BL8SvOc(BTlj;jYA}lgr dKg9n#*V;DaMh@eZoQOT+`ctz`}H%6yu1w0n0Vg-}}as1Zd z32el{CG_qA0uplXL5+mOr~^O(%eIF^utRUwKWtqfwkby}_Ax_iFv~#6$G#4PWM8!` zssl;CQ34CS^+0fZFQz@w!7dK`TakZbrXoK9hZ#mC#-+@-SsQ)|4r@aLTG~9r92~}` z)i?vl#T2!YX 7U z9zl;?HjZ2m^@tFz2GA=^ZD9W`=eNweY#e!>R-KOB_?b;(aO9Eh4#Ms7anK8d+zUD~ z8sug?J|9PoxAc1**f#1Ex0jKl*s!ao!;4AD@e4^;s)i?9B}bCg8P;14n1;t#-DbnZ zuY$+0a1&l7l{sh0GMivh4MrZGt)G 4T4(YzdqTWiRh?+Q4#Q;Cw>6AV&~ z_?NuLY9*LD _sD~Qm3Lfi_rj`}Kn@n~ zi*;sylY-QVxl1!MxG6^TxCXFm^XjUYILZ1C8n+93wgP;|_G)oK?2?SS`>gQlMZxbS zyD=A%Meh6pM+W|l15=@5ZYdQj*6<^evFL)XSiJza(e!3;AA3ckx!JQSy^P3L6eaYC zNDbN(w5%Z>D!8%yvY0G(4f9m+#DR`uIeWz%_|4>)wT%7b#yhCxmeh3z`EW5MB1d7% za7raFq|#kZ+`;C=DI|joqbxg*>_M+50d^72`>KfDkz(4SzzlYgFvnKfMJr14^g>NH zDR%15MrnHTs%{1unJP4~#hwAU*z&blPXE>dJ}l7+n1@G1^H?R^p^5et>FCn+aK7kn z$!dDhJChbm*K-Z>C=>R)E*?_Kp2zxefOuWn-p +F=#uv+HNs2Y^W0kLzVj28*5kGAJ}Go}b(Xo>Hjo?Nux(FrjS*7k_N>q} zDig+(DK|r6!GE@QXoc-mr0_X~w|SG6*KlbTpKmEBv#QIqym|r4 7BAhBlXz((F}+IftZsh&*#9*G{r)^){^7 z_#f^)^U>a{nEuwvMO9J8NXIGcKE%%;M!HsqCPjkV$6s9(;e%X#=>qNIKHn;)xHG${ z R4=d$7w6?3ShROOXJ?V0qjrXS!)iUo^ z*w `=*bFxWR9DKlmMf(~%Wp6!lG%7xts^Dn^vs z1+d4bKWEZ?6l(%s-y6#4iZ64qn}l7LgTa H-H!VfB`ne&h z4RXCr;9S?^@>Q|f@9S+P_g04+Y>2ZySUSER?l-MB!fj!F+9^1|WooY2@K;45@nY3c zF1}~5S_e60*u{{IZS46QDecRqzsC<@K>& 2KhbtUjs9r8lcj-2mu z*V*!of29@HeXhAUM=$B??EI^itnmhi&FqzOwosEDbz7S6rz`1jLybu3crWt-7i6uQ ze4>~5G|v|Bnzc}Jb*4ZmSv785O-2))4_wE|_6WK58C*Lvp#CPEXJ-HgvD-Sp;95;5 z2)cH#2wl*VO+$2SRNQ-S*o&H*Z`%B{R?vU4w}VE+yl$-kiLqh9wXM7JhTVA+N$8;H z*^~H|E^`T51xSO9pb3`Q-1t%l$E=hF0v?~65^%HY;JAz-a@LP1IvyX)N40`cp*eTU z!CW~bq##{1w?;I_RIwyn;S*92Lh+YPLML}Bqo{7 h4l5d~ILGLHVsq2JVD z`?q>fWGR})LmmZ96Xl_hrr66UsLS19*(hD^j#%$YTFe5EGH-|X?~rnKINwKr26WeP z?EE?ou2Hn~9SJJ20w954ed|aBHX?b#3tM||EM9LB;6)sE9P9=IzhMH4ww!eG?Q?N{ zF6 +-NoIa0BY0W&ZkJg39=_HCD*gF3|17Gm;>Brf2rg#NQTWuzbT>m0M zfJFdozRS2NGDyz`D>(h_^!9) =Sz~B;1WA2kJ(}W{i8tU@+;$~ULS-9rmz{EKW%QoRuQf=c?S#n%i z>J&q@2~tPg`l<7aAZ*z(si+^$YqVIQVn sc~S7DX p`&S{G$#$j3<)#B&Ffk;Di(Wwg9 zCcQJLT%f9zT_EyRyFi4L*+) au#brMl#o(otDc@yjsgE+1~DWWwuh!E-PHl zv2)sUlNYRo3Qtb(F3?72G9XUK__}=1$K-=Xz%?>#2S%f((&MW+&PRn|S@4iOGTKPX zQFO}TB3hyeM>GROIAYdT!j!xG1su7r8?nRM jp2?iV3@F^?vh#XmErVP|I^=p)H zZlUJV&)KOuLXg%Vjkj#KW?iJwf`#Rg`wP+MQn4hB@kNh>rmYHSQIqT^s@SE<;>6W` z#qM!mv12n0D3lcF%TY19O#hugkOO!bPG%bAy&pnpoT&PP;bk$pnkA~3#^qdUp@?!J zf#u~?B6C?HQ%vBQ9Xaq8t{#dc$4d$^Sh<9oGsf@<@gx^_YxGUF$L{fSI(Tr&vTFLJ z_{Jcf{Rvz$^6T#?w}Ji>$7zlik6bU8GvlHr+0Xb3WPoeXB$>iw7kU5lP0^o2fqM^z z9sUH1;^=*%d{?n#6cyczN(~9HWFRNEcu}@ukegnHE&o47N(h=1K8SE?ZLGY z|L< !y%ZDBB|%I@$y8VhQhiX&lP zRmXT`D|C&Mg`Ep|(4FLDx| TxZENy+>?@a#H8CLe(P0*7l5p z%!h0%+}<)wrj}oBM&%rD1KDQjtGffW851;a9iHu=pVCf>bb c;-c;)ubxyAW#qqz60G%vfT?# wnCXymZp z=Dl=b$H~wR8q)InwF0iC2L)G&R-rB3j)QI)EwLDmnabrl1nzl~Wr$GU>*o`@u+w-U z*MmRGl1$q+iahw@m2tgh-o(I$^U6S;gk1_igiQeoVVcq>MNLyWthG|a!2R(NZODS2 z=yn#?mU2Yi-4VsMOXa=PhGBZxcMySv!$Gs?mcy_LPDnMnSZs}0Oe *plk#HzJoL-uSpU`tYAi|Pc*Y~4xuGJI(IopB#dK*iJoy6h zGVZ&tLyBpd%qFXR9g>^jQKro}9XzD88He#>i0Qjq=5~I|oTF2;0#eL+h-ovdODn(& zYQMc@Yylf_fCYR2&@^U-Ft)q28QPT8!ZBMG-ZfqU`eL^y+5=z?DQiZ`t7P|tqgH&Y zq&>+&*wh41a*}3zv#5lvlo2;;gC))QZD_no8>f?H!irWT&3I~B$+K12rqTh AF*x!?cZbw=2{VYBLZ-y!$MhOcS;!x2fV5Zfgc&y8(&KedTK}bNzUCyTUa$$K zRp~l&awbbqVp`(ts_L*(<$Ts~WNYLJD2gv`nsuCoD_=Q>Vc90KlT_Qdc1{f}PlJYZ zL^u|O-UJ=xy_n>1_y+~gzXoISW+0id7^O0?CSN0YhP7=W;3nBKQNkv0b2%YRb8*l) zB-}gbnX|Zd2F-Mti4-@`v(e&x;(pCBN6o0e_wAR(_(L-O2xpSvbC6#;--FD5O@9X) zuyH|~i8qcs4jiR(`m({)oBH7n#&74zIF2mHY>xmbV;x%oFXK}+u0^fp IOBP-QgRh_cj$O2X$&@9@k_K!ozlRN~m3 zAZ(AUn=pX_^YMd5oSM&(-hrcF*_LN_I5C9lVbeN5=jC9WHktq9NMk?3^cYK&=5a(B zKx5{I)uaJAr!0PzM^@?=0aV%)Pse>DU4ZDp7n!CHC)~J(!E#>$56IFEFTW!%;2Ag% zH+jKP#BgZ!*hy^bXeY5k`q`x+tr6jJO9*2*fBliM8=e}O!NGyZq$%IqYj~8oz*!P- zlAplii(KE&1G=jXqlf2A{ANCZTH`-#v0|evm%v_6vOp^*{yGy={mwrm0d=V=U2}t` zk >hh-6ypkt4cB7^X(5NH4Z^)T##NlO5nXx$v zP^QbB+LPQVqgAvJ5+HE6$y2INE1*HR^Md|gt0)jQ;-EizzsF`zDZab(DcbP@*Sx!R zNPLl>o{3Yw!{w+Ee-RZ}$&Q-E1Kp2(?lOVO%tmnE09prT@$h9^ng+&6_FScS zCY34@>Qm*A@XA){9H$Gr7c$X)GL@0gKEbqIIu)9530?~G02m$47Tx_rhxP?eca|mh zbcdL8mr#X1GQgg#CRC|Owk)HHy#>q_{;5KTY586!rAoDAS6J0r)WNcan($;xLuDRk hw=CZ=1md3}M2_iRinnkHtl%*$+=N$2)oxnVe*iVtY3Kj| literal 0 HcmV?d00001 diff --git a/api/container/grpc/service_frostfs_fuzz.go b/api/container/grpc/service_frostfs_fuzz.go new file mode 100644 index 0000000..7e6d6e6 --- /dev/null +++ b/api/container/grpc/service_frostfs_fuzz.go @@ -0,0 +1,159 @@ +//go:build gofuzz +// +build gofuzz + +// Code generated by protoc-gen-go-frostfs. DO NOT EDIT. + +package container + +func DoFuzzProtoPutRequest(data []byte) int { + msg := new(PutRequest) + if err := msg.UnmarshalProtobuf(data); err != nil { + return 0 + } + _ = msg.MarshalProtobuf(nil) + return 1 +} +func DoFuzzJSONPutRequest(data []byte) int { + msg := new(PutRequest) + if err := msg.UnmarshalJSON(data); err != nil { + return 0 + } + _, err := msg.MarshalJSON() + if err != nil { + panic(err) + } + return 1 +} +func DoFuzzProtoPutResponse(data []byte) int { + msg := new(PutResponse) + if err := msg.UnmarshalProtobuf(data); err != nil { + return 0 + } + _ = msg.MarshalProtobuf(nil) + return 1 +} +func DoFuzzJSONPutResponse(data []byte) int { + msg := new(PutResponse) + if err := msg.UnmarshalJSON(data); err != nil { + return 0 + } + _, err := msg.MarshalJSON() + if err != nil { + panic(err) + } + return 1 +} +func DoFuzzProtoDeleteRequest(data []byte) int { + msg := new(DeleteRequest) + if err := msg.UnmarshalProtobuf(data); err != nil { + return 0 + } + _ = msg.MarshalProtobuf(nil) + return 1 +} +func DoFuzzJSONDeleteRequest(data []byte) int { + msg := new(DeleteRequest) + if err := msg.UnmarshalJSON(data); err != nil { + return 0 + } + _, err := msg.MarshalJSON() + if err != nil { + panic(err) + } + return 1 +} +func DoFuzzProtoDeleteResponse(data []byte) int { + msg := new(DeleteResponse) + if err := msg.UnmarshalProtobuf(data); err != nil { + return 0 + } + _ = msg.MarshalProtobuf(nil) + return 1 +} +func DoFuzzJSONDeleteResponse(data []byte) int { + msg := new(DeleteResponse) + if err := msg.UnmarshalJSON(data); err != nil { + return 0 + } + _, err := msg.MarshalJSON() + if err != nil { + panic(err) + } + return 1 +} +func DoFuzzProtoGetRequest(data []byte) int { + msg := new(GetRequest) + if err := msg.UnmarshalProtobuf(data); err != nil { + return 0 + } + _ = msg.MarshalProtobuf(nil) + return 1 +} +func DoFuzzJSONGetRequest(data []byte) int { + msg := new(GetRequest) + if err := msg.UnmarshalJSON(data); err != nil { + return 0 + } + _, err := msg.MarshalJSON() + if err != nil { + panic(err) + } + return 1 +} +func DoFuzzProtoGetResponse(data []byte) int { + msg := new(GetResponse) + if err := msg.UnmarshalProtobuf(data); err != nil { + return 0 + } + _ = msg.MarshalProtobuf(nil) + return 1 +} +func DoFuzzJSONGetResponse(data []byte) int { + msg := new(GetResponse) + if err := msg.UnmarshalJSON(data); err != nil { + return 0 + } + _, err := msg.MarshalJSON() + if err != nil { + panic(err) + } + return 1 +} +func DoFuzzProtoListRequest(data []byte) int { + msg := new(ListRequest) + if err := msg.UnmarshalProtobuf(data); err != nil { + return 0 + } + _ = msg.MarshalProtobuf(nil) + return 1 +} +func DoFuzzJSONListRequest(data []byte) int { + msg := new(ListRequest) + if err := msg.UnmarshalJSON(data); err != nil { + return 0 + } + _, err := msg.MarshalJSON() + if err != nil { + panic(err) + } + return 1 +} +func DoFuzzProtoListResponse(data []byte) int { + msg := new(ListResponse) + if err := msg.UnmarshalProtobuf(data); err != nil { + return 0 + } + _ = msg.MarshalProtobuf(nil) + return 1 +} +func DoFuzzJSONListResponse(data []byte) int { + msg := new(ListResponse) + if err := msg.UnmarshalJSON(data); err != nil { + return 0 + } + _, err := msg.MarshalJSON() + if err != nil { + panic(err) + } + return 1 +} diff --git a/api/container/grpc/service_frostfs_test.go b/api/container/grpc/service_frostfs_test.go new file mode 100644 index 0000000..804b89c --- /dev/null +++ b/api/container/grpc/service_frostfs_test.go @@ -0,0 +1,91 @@ +//go:build gofuzz +// +build gofuzz + +// Code generated by protoc-gen-go-frostfs. DO NOT EDIT. + +package container + +import ( + testing "testing" +) + +func FuzzProtoPutRequest(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzProtoPutRequest(data) + }) +} +func FuzzJSONPutRequest(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzJSONPutRequest(data) + }) +} +func FuzzProtoPutResponse(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzProtoPutResponse(data) + }) +} +func FuzzJSONPutResponse(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzJSONPutResponse(data) + }) +} +func FuzzProtoDeleteRequest(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzProtoDeleteRequest(data) + }) +} +func FuzzJSONDeleteRequest(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzJSONDeleteRequest(data) + }) +} +func FuzzProtoDeleteResponse(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzProtoDeleteResponse(data) + }) +} +func FuzzJSONDeleteResponse(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzJSONDeleteResponse(data) + }) +} +func FuzzProtoGetRequest(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzProtoGetRequest(data) + }) +} +func FuzzJSONGetRequest(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzJSONGetRequest(data) + }) +} +func FuzzProtoGetResponse(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzProtoGetResponse(data) + }) +} +func FuzzJSONGetResponse(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzJSONGetResponse(data) + }) +} +func FuzzProtoListRequest(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzProtoListRequest(data) + }) +} +func FuzzJSONListRequest(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzJSONListRequest(data) + }) +} +func FuzzProtoListResponse(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzProtoListResponse(data) + }) +} +func FuzzJSONListResponse(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzJSONListResponse(data) + }) +} diff --git a/api/container/grpc/service_grpc.pb.go b/api/container/grpc/service_grpc.pb.go new file mode 100644 index 0000000000000000000000000000000000000000..abb0fef28efa7bc1a88d27d920de117d4d04fafa GIT binary patch literal 11817 zcmeHNZExE+68;?h3Z4(>+JPhYvb!kYMQ{ilch|VBUF7Ay+#MW6qHLj(=oKkvgChU^ zo+0&SSx(h+eaS9Tv}tUT!{N-snP*5225cle&K5l8iKDp3?jG1G5lXlx`0ivOP8P|^ zwb{jtO=s6^d@;GUt-*k;c_IT5%OQO@X&A!Rf3v^0-&UJh_0Rf8`}CcCO6^jliOYw~ z6|r&xC^Dcq2a+f2z~#19$Ff$A`^j1GvX^BA(MlwWb*&>B%70g^ll{`Mjz|XUEQDBu z++K*#i5Ip=7Fsr%T`XJfY9HuU^peU^>F#c(xnuP#TF3P=kPLqv=oqbnke{d^;>?j) ziGs115{$<(O*m6aN8txLmzMW2g2doU8t LQM_753JLt-|X8ti%^Y>{Oe&A{;ys3lL;?EiD3}P;9U)t+an>?buYS)JkFdR0n ze?7n8Ay>RExV2O>yh-rSd@pq|yh-Y>fou@lT3s5RBdBhQQ5f)8yP6Q-nos8+C(MT% z>z!G?VE`h+(R=ifJaS?cxBz7$(%56J`kkc`prqeJApxKPk3GVXR0($?`VEkgeYOg@ zgPsZZc|y=#Dz%cs!C>`if&FBAd_8F1U4y?X=kClC?q7}mP&Bru{QgQ$U|Z^8#kaK2 zmx_Iy_>RljXX^-yxCKUTSrD(qCxH33So1BDk%PVO0$qvYs_)RKyG9y{dVL93X%b5< z2U}qUymJ;r5%&V@fbhX)=RRh`evSD(y9WWr{!;#7qVrcb5QfA6ED+<=8HP z5r^;>_<;$RlNce3U6SEq3M#o+5pxy^Z1zm@hCKLF>qYwO-9a-uo~I{-MfTRId~I8-Ivp$|0|X85U?Y$T!x0nK-U5aCR+Za(Gv^U1F_m*e?p_Tj^9+ABvK zACG3!>+{KUd^tbYYRxak)5-XPbpHPByB|umi;f$LmzHvYQw0dSE|-#dJcc7+8Mwbo z@TQK)1v=c||2}#07>yJX;le%>D fE@zh1! tG(k>7LE`z%Xtvfi|_>Yup10$Y=Z>d?FkqlmN9XUbcA$RXdw*}z8mGKW-E6d z1PUQ|O(mZk3J-hc60K!DlMp4_?JH_+<1Ev!=B-D|(4WFP8I{iHkJ-J%eO%Yj@{&SL za%yBjh@ZwT1E(sSx*@0g#OU5G>V-)SV`6WbwfpS4kHy?wmmN2E>#+l6Vls Vhu`rk(i-uO#Ido#EnQ1e7e2t~d$XXrte>T>EqGruTmALvf2EwrCwa(!JRn0)v zSoSp*qHzZ`5sIIB2&71ef(*G=7cb^ho1&SWBYombDdZ8VADq9SU}SPB($Hf(x&wo_ ziHm0LHP1_xH&y5Ockd)bI?Pc{5C-a@x%ABy&f`1T{ 6F*+>Ki|MWBxwt zWCysAjkA56{^#V7dz~J%)>U{rrA;KnG_}L&B19I>YmS+FXm14Wy8vwP2JOEX1oqGy zviDV*HbR5eEw;-ZdXw}<#m)Ycj?bHG2NpRGxPIB#b-1jcqX{Pbt-Ys93SDgIn`D{Y zm+M(@(}fZG`=F>RP!bV}6ON(;B@Xs=AGg!lwUdE5iYBqEqPh>lkc8vHCk?R~VqM$> zxK5_aZGBrJi{1akDocg1BcoKRF%?%ETd&zZUD|VI(_Qiftjv=g9dYkX&la*nG1ED5 zl0K8kddKp5 oWRp@N%s0=H%yLFowZF@FMV@~KWojywoVqcJUZ?x++R^|)neq=@t z^(zvH?&GWS@xgG*txYF=dZMLq2+gG}-FRt;VzKJ2-6qX*O1s5Etessm8snR_XL+e& z&fGTSbYJw5eKm@n0-K5RFBETiR(np!869%V9td@r7fNK(vlw6%r^U|zyy}t-_BI{y z4dI7${X(% !84aQAfsqA&@H;Yw%Vv{x<|0ajX}Lwko-N9=MSP2gaMW{|f Z3e1p=W9$#)}9_^u)jQ7%u=LSNd6= ze(k9R*LY~3h&MgjBLWYKjw%5%5+PR@3OOKzk?_o$v@T!c1s4%}5cJhU&ssa%uLU<& zls^GA6$C8?d@JL*l)i?n)8;{E2Wc*2yVrQ*H^c#8-5weH>!Jehq4x* p{+~pFt+14%7J%Xlp*^4P{Y$}sPJh07lc=BVgLXD literal 0 HcmV?d00001 diff --git a/api/container/grpc/types_frostfs.pb.go b/api/container/grpc/types_frostfs.pb.go new file mode 100644 index 0000000000000000000000000000000000000000..2aae748297b20c35e697cec5a074b0c4f17f6420 GIT binary patch literal 12213 zcmeHNZFAE|68>!eiosR8$efW0Zy$1%s{+Dau?Hb_0rmq+?aJ~f(MFbBlI$c$<-gz4 zJw2m0$98hKxV^d*5H#wSXL`DOrh7CVt5?}nt5cooJSg>4O|H~D&&n)(h_Q#K*~2K$ ziZUvEb@*N#z5k$IAHMnE_xtl8JP%H_3bV8fVsQ8S@ob*urSkf{KZ-0>1D%H1G)_;) z^f>7EqFJd1_y_+z&FA5s8l1+ZZ{^2nl#M^+3;im|7Sp%EWGrtU7SnS883*%toa%BG z%*VtoQtlum*HK|b%(E=HgNQ{LCu8CT3O3iBWst2Q#M YM>s@>vik`GTov4abIrXfN`(+(c|>KwXqM++?W+ z@}EI}*zcEDa}DPc@A>UT2}F}csTGJ8VX40Md;ic^O3^c%Nc~O-_6O(sYVdo%_bEsg zOj>)jq%pj_?)R5LPJZ~CvKH}=$ v;d$NdBl;yLa z^ucFX$?S_<7wRG|&kSEM(<+K}GF5EcLY)OmUA>ybX^>yBc)uSl(olJqYNb_c$h@KQ z7*L#+Lt5sTfE^qt7@i(`2Dy6F?_I-p2>M?R3~K*`eSzlZ?Tyr>4-dgw-AMD8XxLRi zC1Qz~Tm;sV6=K&QgCKz=dZyFTSnVd?0dAc~L8v#)xh3D57Nz?9WpY*Oq5P)#&eT4p zhg?#5%rQOvU6&qUPjv);UGe^SQF=2FT>7tPacL*v&HQ(|D3F?+|9L|a1KkG-FO3%y zmdYl0Ad{?L!KRj(df<{rN8_4h JcyeZ05h1&viOA%UoY#c!Fax z3wkqKW7V%^tzAf2L# z21(=>hOv@bSv)ZJXiReOAv`1HtoUd# ^LU|7_DAHZ5ufVsk30LqLa1hU67!6z>&oBmQgE)rqm|?Jw>=FnoXh7kS z3JlDJKazB?9`M8U@r8KDY>*vg!?T&Owwxl3r(G967;Kb&Vx%ucI=i4X-*O{2u<`V^ zLnp?sroSCNqoIY+T9q)?a`w+B?~mFM+}d@w4qHiyB$D YxvK}+|0!a)eV`(N*41{9hUrG}6LwLvr@qTaRA z3BsOV8%l(FfFXK^zhRa_%^aug_|p3Hfg1cuN4Q@H`#2X`)dJR#l4#N mlg& zIaRXnI?c|-e#|i$@!s6HDGnf;(pFavc9I9CQZSm0pB5jo`CGlz2?2{!|4o6zYR*R- z_!trl)i+VjsiU$$32I=bDFO_)sVDO#GSl(tE5yMzKPy z1)eBqKBh$wX|t)}(Ce`ylNPZ35~Ss;EXh( ;6l)AxHc?Ve z6XRWr>zaK$aq#ted^(V*q0wH1orTo!0oaIj+h0T!9CEY|57?0hw-;tptrJHg-TY-X z7{gFyM?ROemI?&*mEt2 6O2L^DHL<-6ia2}`CYEl24d%&X`{aHL+k~lK zy(Lap+ZKlE^B0)MbWLh?uSFWzO1L22;{=5U)g;GR5{Fl2mV17CShUWY)DVC~blYP1 z)+KPo(At8XQCj78i@~K3F3g3gQ7H? cbiOPCC2q}%mlrQrjMd$C*ulY44vW@UQrs0_m z&yl~Mjnt2I{?G9jC_rx6D8qYBs9Nq1UfH*ZUp8MkLD^cYeOkOzbw6zP4E}^Q{ZM_| zN*)~+N#V6h#SONa6Sl&a9)-6@y$RY=&;#;qMsLWG!3}aAS8_~e!FczICV@GLHzL65 zx4yoD>mw(7uN@NL+!PfpH{URiXVpA9US2QkJ1;lLeQxBoj-}hhMm;^REvkmfky^=r z?heQ;`W_MkbP{XzKE&1O_-)Q@@b$YlKtxZo^C0nuzK_1}>gOi}SC3+g+CoU&RaABt zick;#+G!x!#9HsrENp9LU;S`_mJy`0B1A*I2P2WYbCZakhDmHHshPwK9d=A2r2cMA z@`T$`oRwA>MA`$gT6gyjmftW0k5pYLE;gPr7Mtx>z>#ccO?7uXR!0Mp9c-a`#;B?@ z#R^>CHk2w=4S3Zs00!KTMyI+O9j&^x`g+aAiXsZL=F!INM-~6@;ocWnXbH *$(*(f8=C&Qww*fs5xpfFq#+5@)gHo zI@#2BZV~5F@pd}UzO_1*OHaMEDN)!kf%DXGrEl$1zqh#O%F&(<)l4hvIs2uyjq z9wz53fHa-hwdf?FZnI1 Y(+2Y{mj6sruJh%u?4)_Go1v4Yi{#ExET8 z;bJN~z8+eYSJ+^y8Vl>NR_t}OH~Tv9RxCigvXvnds0i0jp%u^8J3EPBts}IAY(1M6 zAEv|++)*xbYLUytzTw4$t1!R+b%cZBI>cym|F2v^EKSDV(C>G7wR6=2Q($94-a&Lx z^4!&BNoM+%8*xfv{cbo@YLjk2NqdZGJJmNtzC1y))6bupV{p})YTPl^E`
R zZM4evXy)DEhD=MQayY&aj@=t+-t3ESuvE9Xvf?(@;BHK2Cl`OUrpoua?yEs?TfzS? z`OLN8tuFaw)9K#LHL3=Rrykw+*^ 1%M%08+_gBz_~8js69`NRWqT@gZJC!XAdBN$=z2EbeH~b DhTZCW literal 0 HcmV?d00001 diff --git a/api/container/grpc/types_frostfs_fuzz.go b/api/container/grpc/types_frostfs_fuzz.go new file mode 100644 index 0000000..5551978 --- /dev/null +++ b/api/container/grpc/types_frostfs_fuzz.go @@ -0,0 +1,26 @@ +//go:build gofuzz +// +build gofuzz + +// Code generated by protoc-gen-go-frostfs. DO NOT EDIT. + +package container + +func DoFuzzProtoContainer(data []byte) int { + msg := new(Container) + if err := msg.UnmarshalProtobuf(data); err != nil { + return 0 + } + _ = msg.MarshalProtobuf(nil) + return 1 +} +func DoFuzzJSONContainer(data []byte) int { + msg := new(Container) + if err := msg.UnmarshalJSON(data); err != nil { + return 0 + } + _, err := msg.MarshalJSON() + if err != nil { + panic(err) + } + return 1 +} diff --git a/api/container/grpc/types_frostfs_test.go b/api/container/grpc/types_frostfs_test.go new file mode 100644 index 0000000..64d840e --- /dev/null +++ b/api/container/grpc/types_frostfs_test.go @@ -0,0 +1,21 @@ +//go:build gofuzz +// +build gofuzz + +// Code generated by protoc-gen-go-frostfs. DO NOT EDIT. + +package container + +import ( + testing "testing" +) + +func FuzzProtoContainer(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzProtoContainer(data) + }) +} +func FuzzJSONContainer(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzJSONContainer(data) + }) +} diff --git a/api/container/json.go b/api/container/json.go new file mode 100644 index 0000000..838fb8d --- /dev/null +++ b/api/container/json.go @@ -0,0 +1,22 @@ +package container + +import ( + container "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/container/grpc" + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/rpc/message" +) + +func (a *Attribute) MarshalJSON() ([]byte, error) { + return message.MarshalJSON(a) +} + +func (a *Attribute) UnmarshalJSON(data []byte) error { + return message.UnmarshalJSON(a, data, new(container.Container_Attribute)) +} + +func (c *Container) MarshalJSON() ([]byte, error) { + return message.MarshalJSON(c) +} + +func (c *Container) UnmarshalJSON(data []byte) error { + return message.UnmarshalJSON(c, data, new(container.Container)) +} diff --git a/api/container/marshal.go b/api/container/marshal.go new file mode 100644 index 0000000..2b16669 --- /dev/null +++ b/api/container/marshal.go @@ -0,0 +1,345 @@ +package container + +import ( + container "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/container/grpc" + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/rpc/message" + protoutil "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/util/proto" +) + +const ( + attributeKeyField = 1 + attributeValueField = 2 + + containerVersionField = 1 + containerOwnerField = 2 + containerNonceField = 3 + containerBasicACLField = 4 + containerAttributesField = 5 + containerPlacementField = 6 + + putReqBodyContainerField = 1 + putReqBodySignatureField = 2 + + putRespBodyIDField = 1 + + deleteReqBodyIDField = 1 + deleteReqBodySignatureField = 2 + + getReqBodyIDField = 1 + + getRespBodyContainerField = 1 + getRespBodySignatureField = 2 + getRespBodyTokenField = 3 + + listReqBodyOwnerField = 1 + + listRespBodyIDsField = 1 +) + +func (a *Attribute) StableMarshal(buf []byte) []byte { + if a == nil { + return []byte{} + } + + if buf == nil { + buf = make([]byte, a.StableSize()) + } + + var offset int + + offset += protoutil.StringMarshal(attributeKeyField, buf[offset:], a.key) + protoutil.StringMarshal(attributeValueField, buf[offset:], a.val) + + return buf +} + +func (a *Attribute) StableSize() (size int) { + if a == nil { + return 0 + } + + size += protoutil.StringSize(attributeKeyField, a.key) + size += protoutil.StringSize(attributeValueField, a.val) + + return size +} + +func (a *Attribute) Unmarshal(data []byte) error { + return message.Unmarshal(a, data, new(container.Container_Attribute)) +} + +func (c *Container) StableMarshal(buf []byte) []byte { + if c == nil { + return []byte{} + } + + if buf == nil { + buf = make([]byte, c.StableSize()) + } + + var offset int + + offset += protoutil.NestedStructureMarshal(containerVersionField, buf[offset:], c.version) + offset += protoutil.NestedStructureMarshal(containerOwnerField, buf[offset:], c.ownerID) + offset += protoutil.BytesMarshal(containerNonceField, buf[offset:], c.nonce) + offset += protoutil.UInt32Marshal(containerBasicACLField, buf[offset:], c.basicACL) + + for i := range c.attr { + offset += protoutil.NestedStructureMarshal(containerAttributesField, buf[offset:], &c.attr[i]) + } + + protoutil.NestedStructureMarshal(containerPlacementField, buf[offset:], c.policy) + + return buf +} + +func (c *Container) StableSize() (size int) { + if c == nil { + return 0 + } + + size += protoutil.NestedStructureSize(containerVersionField, c.version) + size += protoutil.NestedStructureSize(containerOwnerField, c.ownerID) + size += protoutil.BytesSize(containerNonceField, c.nonce) + size += protoutil.UInt32Size(containerBasicACLField, c.basicACL) + + for i := range c.attr { + size += protoutil.NestedStructureSize(containerAttributesField, &c.attr[i]) + } + + size += protoutil.NestedStructureSize(containerPlacementField, c.policy) + + return size +} + +func (c *Container) Unmarshal(data []byte) error { + return message.Unmarshal(c, data, new(container.Container)) +} + +func (r *PutRequestBody) StableMarshal(buf []byte) []byte { + if r == nil { + return []byte{} + } + + if buf == nil { + buf = make([]byte, r.StableSize()) + } + + var offset int + + offset += protoutil.NestedStructureMarshal(putReqBodyContainerField, buf[offset:], r.cnr) + protoutil.NestedStructureMarshal(putReqBodySignatureField, buf[offset:], r.sig) + + return buf +} + +func (r *PutRequestBody) StableSize() (size int) { + if r == nil { + return 0 + } + + size += protoutil.NestedStructureSize(putReqBodyContainerField, r.cnr) + size += protoutil.NestedStructureSize(putReqBodySignatureField, r.sig) + + return size +} + +func (r *PutRequestBody) Unmarshal(data []byte) error { + return message.Unmarshal(r, data, new(container.PutRequest_Body)) +} + +func (r *PutResponseBody) StableMarshal(buf []byte) []byte { + if r == nil { + return []byte{} + } + + if buf == nil { + buf = make([]byte, r.StableSize()) + } + + protoutil.NestedStructureMarshal(putRespBodyIDField, buf, r.cid) + + return buf +} + +func (r *PutResponseBody) StableSize() (size int) { + if r == nil { + return 0 + } + + size += protoutil.NestedStructureSize(putRespBodyIDField, r.cid) + + return size +} + +func (r *PutResponseBody) Unmarshal(data []byte) error { + return message.Unmarshal(r, data, new(container.PutResponse_Body)) +} + +func (r *DeleteRequestBody) StableMarshal(buf []byte) []byte { + if r == nil { + return []byte{} + } + + if buf == nil { + buf = make([]byte, r.StableSize()) + } + + var offset int + + offset += protoutil.NestedStructureMarshal(deleteReqBodyIDField, buf[offset:], r.cid) + protoutil.NestedStructureMarshal(deleteReqBodySignatureField, buf[offset:], r.sig) + + return buf +} + +func (r *DeleteRequestBody) StableSize() (size int) { + if r == nil { + return 0 + } + + size += protoutil.NestedStructureSize(deleteReqBodyIDField, r.cid) + size += protoutil.NestedStructureSize(deleteReqBodySignatureField, r.sig) + + return size +} + +func (r *DeleteRequestBody) Unmarshal(data []byte) error { + return message.Unmarshal(r, data, new(container.DeleteRequest_Body)) +} + +func (r *DeleteResponseBody) StableMarshal(_ []byte) []byte { + return nil +} + +func (r *DeleteResponseBody) StableSize() (size int) { + return 0 +} + +func (r *DeleteResponseBody) Unmarshal([]byte) error { + return nil +} + +func (r *GetRequestBody) StableMarshal(buf []byte) []byte { + if r == nil { + return []byte{} + } + + if buf == nil { + buf = make([]byte, r.StableSize()) + } + + protoutil.NestedStructureMarshal(getReqBodyIDField, buf, r.cid) + + return buf +} + +func (r *GetRequestBody) StableSize() (size int) { + if r == nil { + return 0 + } + + size += protoutil.NestedStructureSize(getReqBodyIDField, r.cid) + + return size +} + +func (r *GetRequestBody) Unmarshal(data []byte) error { + return message.Unmarshal(r, data, new(container.GetRequest_Body)) +} + +func (r *GetResponseBody) StableMarshal(buf []byte) []byte { + if r == nil { + return []byte{} + } + + if buf == nil { + buf = make([]byte, r.StableSize()) + } + + var offset int + + offset += protoutil.NestedStructureMarshal(getRespBodyContainerField, buf, r.cnr) + offset += protoutil.NestedStructureMarshal(getRespBodySignatureField, buf[offset:], r.sig) + protoutil.NestedStructureMarshal(getRespBodyTokenField, buf[offset:], r.token) + + return buf +} + +func (r *GetResponseBody) StableSize() (size int) { + if r == nil { + return 0 + } + + size += protoutil.NestedStructureSize(getRespBodyContainerField, r.cnr) + size += protoutil.NestedStructureSize(getRespBodySignatureField, r.sig) + size += protoutil.NestedStructureSize(getRespBodyTokenField, r.token) + + return size +} + +func (r *GetResponseBody) Unmarshal(data []byte) error { + return message.Unmarshal(r, data, new(container.GetResponse_Body)) +} + +func (r *ListRequestBody) StableMarshal(buf []byte) []byte { + if r == nil { + return []byte{} + } + + if buf == nil { + buf = make([]byte, r.StableSize()) + } + + protoutil.NestedStructureMarshal(listReqBodyOwnerField, buf, r.ownerID) + + return buf +} + +func (r *ListRequestBody) StableSize() (size int) { + if r == nil { + return 0 + } + + size += protoutil.NestedStructureSize(listReqBodyOwnerField, r.ownerID) + + return size +} + +func (r *ListRequestBody) Unmarshal(data []byte) error { + return message.Unmarshal(r, data, new(container.ListRequest_Body)) +} + +func (r *ListResponseBody) StableMarshal(buf []byte) []byte { + if r == nil { + return []byte{} + } + + if buf == nil { + buf = make([]byte, r.StableSize()) + } + + var offset int + + for i := range r.cidList { + offset += protoutil.NestedStructureMarshal(listRespBodyIDsField, buf[offset:], &r.cidList[i]) + } + + return buf +} + +func (r *ListResponseBody) StableSize() (size int) { + if r == nil { + return 0 + } + + for i := range r.cidList { + size += protoutil.NestedStructureSize(listRespBodyIDsField, &r.cidList[i]) + } + + return size +} + +func (r *ListResponseBody) Unmarshal(data []byte) error { + return message.Unmarshal(r, data, new(container.ListResponse_Body)) +} diff --git a/api/container/message_test.go b/api/container/message_test.go new file mode 100644 index 0000000..b32475d --- /dev/null +++ b/api/container/message_test.go @@ -0,0 +1,36 @@ +package container_test + +import ( + "testing" + + containertest "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/container/test" + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/rpc/message" + messagetest "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/rpc/message/test" +) + +func TestMessageConvert(t *testing.T) { + messagetest.TestRPCMessage(t, + func(empty bool) message.Message { return containertest.GenerateAttribute(empty) }, + func(empty bool) message.Message { return containertest.GenerateContainer(empty) }, + func(empty bool) message.Message { return containertest.GeneratePutRequestBody(empty) }, + func(empty bool) message.Message { return containertest.GeneratePutRequest(empty) }, + func(empty bool) message.Message { return containertest.GeneratePutResponseBody(empty) }, + func(empty bool) message.Message { return containertest.GeneratePutResponse(empty) }, + func(empty bool) message.Message { return containertest.GenerateGetRequestBody(empty) }, + func(empty bool) message.Message { return containertest.GenerateGetRequest(empty) }, + func(empty bool) message.Message { return containertest.GenerateGetResponseBody(empty) }, + func(empty bool) message.Message { return containertest.GenerateGetResponse(empty) }, + func(empty bool) message.Message { return containertest.GenerateDeleteRequestBody(empty) }, + func(empty bool) message.Message { return containertest.GenerateDeleteRequest(empty) }, + func(empty bool) message.Message { return containertest.GenerateDeleteResponseBody(empty) }, + func(empty bool) message.Message { return containertest.GenerateDeleteResponse(empty) }, + func(empty bool) message.Message { return containertest.GenerateListRequestBody(empty) }, + func(empty bool) message.Message { return containertest.GenerateListRequest(empty) }, + func(empty bool) message.Message { return containertest.GenerateListResponseBody(empty) }, + func(empty bool) message.Message { return containertest.GenerateListResponse(empty) }, + func(empty bool) message.Message { return containertest.GenerateGetRequestBody(empty) }, + func(empty bool) message.Message { return containertest.GenerateGetRequest(empty) }, + func(empty bool) message.Message { return containertest.GenerateGetResponseBody(empty) }, + func(empty bool) message.Message { return containertest.GenerateGetResponse(empty) }, + ) +} diff --git a/api/container/status.go b/api/container/status.go new file mode 100644 index 0000000..93b2983 --- /dev/null +++ b/api/container/status.go @@ -0,0 +1,33 @@ +package container + +import ( + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/status" + statusgrpc "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/status/grpc" +) + +// LocalizeFailStatus checks if passed global status.Code is related to container failure and: +// +// then localizes the code and returns true, +// else leaves the code unchanged and returns false. +// +// Arg must not be nil. +func LocalizeFailStatus(c *status.Code) bool { + return status.LocalizeIfInSection(c, uint32(statusgrpc.Section_SECTION_CONTAINER)) +} + +// GlobalizeFail globalizes local code of container failure. +// +// Arg must not be nil. +func GlobalizeFail(c *status.Code) { + c.GlobalizeSection(uint32(statusgrpc.Section_SECTION_CONTAINER)) +} + +const ( + // StatusNotFound is a local status.Code value for + // CONTAINER_NOT_FOUND container failure. + StatusNotFound status.Code = iota + + // StatusEACLNotFound is a local status.Code value for + // EACL_NOT_FOUND failure. + StatusEACLNotFound +) diff --git a/api/container/status_test.go b/api/container/status_test.go new file mode 100644 index 0000000..92c738d --- /dev/null +++ b/api/container/status_test.go @@ -0,0 +1,15 @@ +package container_test + +import ( + "testing" + + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/container" + statustest "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/status/test" +) + +func TestStatusCodes(t *testing.T) { + statustest.TestCodes(t, container.LocalizeFailStatus, container.GlobalizeFail, + container.StatusNotFound, 3072, + container.StatusEACLNotFound, 3073, + ) +} diff --git a/api/container/test/generate.go b/api/container/test/generate.go new file mode 100644 index 0000000..f804da5 --- /dev/null +++ b/api/container/test/generate.go @@ -0,0 +1,240 @@ +package containertest + +import ( + "crypto/rand" + + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/container" + netmaptest "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/netmap/test" + refstest "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/refs/test" + sessiontest "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/session/test" +) + +func GenerateAttribute(empty bool) *container.Attribute { + m := new(container.Attribute) + + if !empty { + m.SetKey("key") + m.SetValue("val") + } + + return m +} + +func GenerateAttributes(empty bool) []container.Attribute { + var res []container.Attribute + + if !empty { + res = append(res, + *GenerateAttribute(false), + *GenerateAttribute(false), + ) + } + + return res +} + +func GenerateContainer(empty bool) *container.Container { + m := new(container.Container) + + if !empty { + nonce := make([]byte, 16) + _, _ = rand.Read(nonce) + + m.SetBasicACL(12) + m.SetNonce(nonce) + m.SetOwnerID(refstest.GenerateOwnerID(false)) + m.SetAttributes(GenerateAttributes(false)) + m.SetPlacementPolicy(netmaptest.GeneratePlacementPolicy(false)) + } + + m.SetVersion(refstest.GenerateVersion(empty)) + + return m +} + +func GeneratePutRequestBody(empty bool) *container.PutRequestBody { + m := new(container.PutRequestBody) + + if !empty { + m.SetContainer(GenerateContainer(false)) + } + + m.SetSignature(refstest.GenerateSignature(empty)) + + return m +} + +func GeneratePutRequest(empty bool) *container.PutRequest { + m := new(container.PutRequest) + + if !empty { + m.SetBody(GeneratePutRequestBody(false)) + } + + m.SetMetaHeader(sessiontest.GenerateRequestMetaHeader(empty)) + m.SetVerificationHeader(sessiontest.GenerateRequestVerificationHeader(empty)) + + return m +} + +func GeneratePutResponseBody(empty bool) *container.PutResponseBody { + m := new(container.PutResponseBody) + + if !empty { + m.SetContainerID(refstest.GenerateContainerID(false)) + } + + return m +} + +func GeneratePutResponse(empty bool) *container.PutResponse { + m := new(container.PutResponse) + + if !empty { + m.SetBody(GeneratePutResponseBody(false)) + } + + m.SetMetaHeader(sessiontest.GenerateResponseMetaHeader(empty)) + m.SetVerificationHeader(sessiontest.GenerateResponseVerificationHeader(empty)) + + return m +} + +func GenerateGetRequestBody(empty bool) *container.GetRequestBody { + m := new(container.GetRequestBody) + + if !empty { + m.SetContainerID(refstest.GenerateContainerID(false)) + } + + return m +} + +func GenerateGetRequest(empty bool) *container.GetRequest { + m := new(container.GetRequest) + + if !empty { + m.SetBody(GenerateGetRequestBody(false)) + } + + m.SetMetaHeader(sessiontest.GenerateRequestMetaHeader(empty)) + m.SetVerificationHeader(sessiontest.GenerateRequestVerificationHeader(empty)) + + return m +} + +func GenerateGetResponseBody(empty bool) *container.GetResponseBody { + m := new(container.GetResponseBody) + + if !empty { + m.SetContainer(GenerateContainer(false)) + } + + m.SetSignature(refstest.GenerateSignature(empty)) + m.SetSessionToken(sessiontest.GenerateSessionToken(empty)) + + return m +} + +func GenerateGetResponse(empty bool) *container.GetResponse { + m := new(container.GetResponse) + + if !empty { + m.SetBody(GenerateGetResponseBody(false)) + } + + m.SetMetaHeader(sessiontest.GenerateResponseMetaHeader(empty)) + m.SetVerificationHeader(sessiontest.GenerateResponseVerificationHeader(empty)) + + return m +} + +func GenerateDeleteRequestBody(empty bool) *container.DeleteRequestBody { + m := new(container.DeleteRequestBody) + + if !empty { + m.SetContainerID(refstest.GenerateContainerID(false)) + } + + m.SetSignature(refstest.GenerateSignature(empty)) + + return m +} + +func GenerateDeleteRequest(empty bool) *container.DeleteRequest { + m := new(container.DeleteRequest) + + if !empty { + m.SetBody(GenerateDeleteRequestBody(false)) + } + + m.SetMetaHeader(sessiontest.GenerateRequestMetaHeader(empty)) + m.SetVerificationHeader(sessiontest.GenerateRequestVerificationHeader(empty)) + + return m +} + +func GenerateDeleteResponseBody(_ bool) *container.DeleteResponseBody { + m := new(container.DeleteResponseBody) + + return m +} + +func GenerateDeleteResponse(empty bool) *container.DeleteResponse { + m := new(container.DeleteResponse) + + if !empty { + m.SetBody(GenerateDeleteResponseBody(false)) + } + + m.SetMetaHeader(sessiontest.GenerateResponseMetaHeader(empty)) + m.SetVerificationHeader(sessiontest.GenerateResponseVerificationHeader(empty)) + + return m +} + +func GenerateListRequestBody(empty bool) *container.ListRequestBody { + m := new(container.ListRequestBody) + + if !empty { + m.SetOwnerID(refstest.GenerateOwnerID(false)) + } + + return m +} + +func GenerateListRequest(empty bool) *container.ListRequest { + m := new(container.ListRequest) + + if !empty { + m.SetBody(GenerateListRequestBody(false)) + } + + m.SetMetaHeader(sessiontest.GenerateRequestMetaHeader(empty)) + m.SetVerificationHeader(sessiontest.GenerateRequestVerificationHeader(empty)) + + return m +} + +func GenerateListResponseBody(empty bool) *container.ListResponseBody { + m := new(container.ListResponseBody) + + if !empty { + m.SetContainerIDs(refstest.GenerateContainerIDs(false)) + } + + return m +} + +func GenerateListResponse(empty bool) *container.ListResponse { + m := new(container.ListResponse) + + if !empty { + m.SetBody(GenerateListResponseBody(false)) + } + + m.SetMetaHeader(sessiontest.GenerateResponseMetaHeader(empty)) + m.SetVerificationHeader(sessiontest.GenerateResponseVerificationHeader(empty)) + + return m +} diff --git a/api/container/types.go b/api/container/types.go new file mode 100644 index 0000000..92c4706 --- /dev/null +++ b/api/container/types.go @@ -0,0 +1,446 @@ +package container + +import ( + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/netmap" + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/refs" + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/session" +) + +type Attribute struct { + key, val string +} + +type Container struct { + version *refs.Version + + ownerID *refs.OwnerID + + nonce []byte + + basicACL uint32 + + attr []Attribute + + policy *netmap.PlacementPolicy +} + +type PutRequestBody struct { + cnr *Container + + sig *refs.Signature +} +type PutRequest struct { + body *PutRequestBody + + session.RequestHeaders +} + +type PutResponseBody struct { + cid *refs.ContainerID +} + +type PutResponse struct { + body *PutResponseBody + + session.ResponseHeaders +} + +type GetRequestBody struct { + cid *refs.ContainerID +} + +type GetRequest struct { + body *GetRequestBody + + session.RequestHeaders +} + +type GetResponseBody struct { + cnr *Container + + token *session.Token + + sig *refs.Signature +} + +type GetResponse struct { + body *GetResponseBody + + session.ResponseHeaders +} + +type DeleteRequestBody struct { + cid *refs.ContainerID + + sig *refs.Signature +} + +type DeleteRequest struct { + body *DeleteRequestBody + + session.RequestHeaders +} + +type DeleteResponseBody struct{} + +type DeleteResponse struct { + body *DeleteResponseBody + + session.ResponseHeaders +} + +type ListRequestBody struct { + ownerID *refs.OwnerID +} + +type ListRequest struct { + body *ListRequestBody + + session.RequestHeaders +} + +type ListResponseBody struct { + cidList []refs.ContainerID +} + +type ListResponse struct { + body *ListResponseBody + + session.ResponseHeaders +} + +func (a *Attribute) GetKey() string { + if a != nil { + return a.key + } + + return "" +} + +func (a *Attribute) SetKey(v string) { + a.key = v +} + +func (a *Attribute) GetValue() string { + if a != nil { + return a.val + } + + return "" +} + +func (a *Attribute) SetValue(v string) { + a.val = v +} + +func (c *Container) GetVersion() *refs.Version { + if c != nil { + return c.version + } + + return nil +} + +func (c *Container) SetVersion(v *refs.Version) { + c.version = v +} + +func (c *Container) GetOwnerID() *refs.OwnerID { + if c != nil { + return c.ownerID + } + + return nil +} + +func (c *Container) SetOwnerID(v *refs.OwnerID) { + c.ownerID = v +} + +func (c *Container) GetNonce() []byte { + if c != nil { + return c.nonce + } + + return nil +} + +func (c *Container) SetNonce(v []byte) { + c.nonce = v +} + +func (c *Container) GetBasicACL() uint32 { + if c != nil { + return c.basicACL + } + + return 0 +} + +func (c *Container) SetBasicACL(v uint32) { + c.basicACL = v +} + +func (c *Container) GetAttributes() []Attribute { + if c != nil { + return c.attr + } + + return nil +} + +func (c *Container) SetAttributes(v []Attribute) { + c.attr = v +} + +func (c *Container) GetPlacementPolicy() *netmap.PlacementPolicy { + if c != nil { + return c.policy + } + + return nil +} + +func (c *Container) SetPlacementPolicy(v *netmap.PlacementPolicy) { + c.policy = v +} + +func (r *PutRequestBody) GetContainer() *Container { + if r != nil { + return r.cnr + } + + return nil +} + +func (r *PutRequestBody) SetContainer(v *Container) { + r.cnr = v +} + +func (r *PutRequestBody) GetSignature() *refs.Signature { + if r != nil { + return r.sig + } + + return nil +} + +func (r *PutRequestBody) SetSignature(v *refs.Signature) { + // TODO: (neofs-api-go#381) avoid this hack (e.g. create refs.SignatureRFC6979 type) + v.SetScheme(0) + r.sig = v +} + +func (r *PutRequest) GetBody() *PutRequestBody { + if r != nil { + return r.body + } + + return nil +} + +func (r *PutRequest) SetBody(v *PutRequestBody) { + r.body = v +} + +func (r *PutResponseBody) GetContainerID() *refs.ContainerID { + if r != nil { + return r.cid + } + + return nil +} + +func (r *PutResponseBody) SetContainerID(v *refs.ContainerID) { + r.cid = v +} + +func (r *PutResponse) GetBody() *PutResponseBody { + if r != nil { + return r.body + } + + return nil +} + +func (r *PutResponse) SetBody(v *PutResponseBody) { + r.body = v +} + +func (r *GetRequestBody) GetContainerID() *refs.ContainerID { + if r != nil { + return r.cid + } + + return nil +} + +func (r *GetRequestBody) SetContainerID(v *refs.ContainerID) { + r.cid = v +} + +func (r *GetRequest) GetBody() *GetRequestBody { + if r != nil { + return r.body + } + + return nil +} + +func (r *GetRequest) SetBody(v *GetRequestBody) { + r.body = v +} + +func (r *GetResponseBody) GetContainer() *Container { + if r != nil { + return r.cnr + } + + return nil +} + +func (r *GetResponseBody) SetContainer(v *Container) { + r.cnr = v +} + +// GetSessionToken returns token of the session within which requested +// container was created. +func (r *GetResponseBody) GetSessionToken() *session.Token { + if r != nil { + return r.token + } + + return nil +} + +// SetSessionToken sets token of the session within which requested +// container was created. +func (r *GetResponseBody) SetSessionToken(v *session.Token) { + r.token = v +} + +// GetSignature returns signature of the requested container. +func (r *GetResponseBody) GetSignature() *refs.Signature { + if r != nil { + return r.sig + } + + return nil +} + +// SetSignature sets signature of the requested container. +func (r *GetResponseBody) SetSignature(v *refs.Signature) { + // TODO: (neofs-api-go#381) avoid this hack (e.g. create refs.SignatureRFC6979 type) + v.SetScheme(0) + r.sig = v +} + +func (r *GetResponse) GetBody() *GetResponseBody { + if r != nil { + return r.body + } + + return nil +} + +func (r *GetResponse) SetBody(v *GetResponseBody) { + r.body = v +} + +func (r *DeleteRequestBody) GetContainerID() *refs.ContainerID { + if r != nil { + return r.cid + } + + return nil +} + +func (r *DeleteRequestBody) SetContainerID(v *refs.ContainerID) { + r.cid = v +} + +func (r *DeleteRequestBody) GetSignature() *refs.Signature { + if r != nil { + return r.sig + } + + return nil +} + +func (r *DeleteRequestBody) SetSignature(v *refs.Signature) { + // TODO: (neofs-api-go#381) avoid this hack (e.g. create refs.SignatureRFC6979 type) + v.SetScheme(0) + r.sig = v +} + +func (r *DeleteRequest) GetBody() *DeleteRequestBody { + if r != nil { + return r.body + } + + return nil +} + +func (r *DeleteRequest) SetBody(v *DeleteRequestBody) { + r.body = v +} + +func (r *DeleteResponse) GetBody() *DeleteResponseBody { + if r != nil { + return r.body + } + + return nil +} + +func (r *DeleteResponse) SetBody(v *DeleteResponseBody) { + r.body = v +} + +func (r *ListRequestBody) GetOwnerID() *refs.OwnerID { + if r != nil { + return r.ownerID + } + + return nil +} + +func (r *ListRequestBody) SetOwnerID(v *refs.OwnerID) { + r.ownerID = v +} + +func (r *ListRequest) GetBody() *ListRequestBody { + if r != nil { + return r.body + } + + return nil +} + +func (r *ListRequest) SetBody(v *ListRequestBody) { + r.body = v +} + +func (r *ListResponseBody) GetContainerIDs() []refs.ContainerID { + if r != nil { + return r.cidList + } + + return nil +} + +func (r *ListResponseBody) SetContainerIDs(v []refs.ContainerID) { + r.cidList = v +} + +func (r *ListResponse) GetBody() *ListResponseBody { + if r != nil { + return r.body + } + + return nil +} + +func (r *ListResponse) SetBody(v *ListResponseBody) { + r.body = v +} diff --git a/api/lock/grpc/types_frostfs.pb.go b/api/lock/grpc/types_frostfs.pb.go new file mode 100644 index 0000000000000000000000000000000000000000..19e45b09bf6c399209c29217c9c474d0a341aacd GIT binary patch literal 3858 zcmbtW?{C{S5dCcZ6&C?*<;; 7qm@99`D}Wy?11i&`C9ubSq1#g^@Fw-qTW7rb>^nc63`EWxA?OR>yRDNf(#b zbawjoIu3%RNEhN(QeLHtAW+4!(uSg7@U^Z=3T2sAGgaOu{22y=tS}VfAN==LFH;I{ zm5E!=s?4h7S~v0}ubSDpm?nPbsGcnVArVWJXqnXspM{HMRpme6qA@B@_$<5N9Dmpc zABK}Q%m3vkX}z#RQQvQn&6{bQRz>ocN=>DecqffkX`OUdIPo=?YpE@O*1QlZ*NwH} z>L<>GZC6^E?-@IrFc=1bxnD{;M}Mg{x=9T^1cP_7m`YvKryK4?d^!CpQ}gzezHq2< zSh(%*OYj&3cS18G{-!p=_&pcyozV4M ; z%bHfo%-uOrNXnGVXLKj>23vD+CpTMDRSJD?-N!-Jlqp4Pa+TSF5)CP`2vli?+zrJ@ zCKE#O_;cW(((7RG2+dWR6pbgOMG2v< FxYsC|Z-nhJR|PX6mxU~iE8TM>KrT{eB9(iI_Z&vE+JJY{ zdn1SbHz!?iimd7l6W;UJ@6tqoI+Gcg_Qd}C#zX}S*6~@POp7`yP^P-ZWb1ep4cQH1 zPY9cij*JDXz6ki>8ua?M>$af7PPcm@zR}F@w{p8Ty {2OCYaO~UOHpK<6c3R6**m4ULwj7qrh(uAg8)ZvIi z9!Vie#Y7*}P;olvXg1PCQ&=FWs3jebiJ9-F9&86QVZ?~41(OmdeBn>A%4D?o4zSA- z?Bx0cvZkGT9+`vX<5*-C8pzv aaaZqO%Z z6YSzY#R@kD9(hq%HhFGY>0r2-r8H(Ha=@)3gMmN6kfK))uK<}ffc_n1*3c?q$7sYx zgGZ7%gl4y$B{(Ut n#4;l(qz7HFqN# z`oy~d<+kc~Z^!tYADIt4#m2 RcXUK0hWzA7@HGC;jam> z+NhFsW69*NAP&xn*{i1|csRz271Ll~FVx+!&zp8VSN8d|b0^r%P3n&S1N(B^$sV+{ zf#E8Hx^{f 7lh+arI#ceK%O! TWX2M*(%oeU47LfvFQb0}-gwDh literal 0 HcmV?d00001 diff --git a/api/lock/grpc/types_frostfs_fuzz.go b/api/lock/grpc/types_frostfs_fuzz.go new file mode 100644 index 0000000..cb55151 --- /dev/null +++ b/api/lock/grpc/types_frostfs_fuzz.go @@ -0,0 +1,26 @@ +//go:build gofuzz +// +build gofuzz + +// Code generated by protoc-gen-go-frostfs. DO NOT EDIT. + +package lock + +func DoFuzzProtoLock(data []byte) int { + msg := new(Lock) + if err := msg.UnmarshalProtobuf(data); err != nil { + return 0 + } + _ = msg.MarshalProtobuf(nil) + return 1 +} +func DoFuzzJSONLock(data []byte) int { + msg := new(Lock) + if err := msg.UnmarshalJSON(data); err != nil { + return 0 + } + _, err := msg.MarshalJSON() + if err != nil { + panic(err) + } + return 1 +} diff --git a/api/lock/grpc/types_frostfs_test.go b/api/lock/grpc/types_frostfs_test.go new file mode 100644 index 0000000..7c69064 --- /dev/null +++ b/api/lock/grpc/types_frostfs_test.go @@ -0,0 +1,21 @@ +//go:build gofuzz +// +build gofuzz + +// Code generated by protoc-gen-go-frostfs. DO NOT EDIT. + +package lock + +import ( + testing "testing" +) + +func FuzzProtoLock(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzProtoLock(data) + }) +} +func FuzzJSONLock(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzJSONLock(data) + }) +} diff --git a/api/netmap/convert.go b/api/netmap/convert.go new file mode 100644 index 0000000..f8117a5 --- /dev/null +++ b/api/netmap/convert.go @@ -0,0 +1,916 @@ +package netmap + +import ( + netmap "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/netmap/grpc" + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/refs" + refsGRPC "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/refs/grpc" + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/rpc/grpc" + "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/api/rpc/message" +) + +func (f *Filter) ToGRPCMessage() grpc.Message { + var m *netmap.Filter + + if f != nil { + m = new(netmap.Filter) + + m.SetKey(f.key) + m.SetValue(f.value) + m.SetName(f.name) + m.SetOp(OperationToGRPCMessage(f.op)) + m.SetFilters(FiltersToGRPC(f.filters)) + } + + return m +} + +func (f *Filter) FromGRPCMessage(m grpc.Message) error { + v, ok := m.(*netmap.Filter) + if !ok { + return message.NewUnexpectedMessageType(m, v) + } + + var err error + + f.filters, err = FiltersFromGRPC(v.GetFilters()) + if err != nil { + return err + } + + f.key = v.GetKey() + f.value = v.GetValue() + f.name = v.GetName() + f.op = OperationFromGRPCMessage(v.GetOp()) + + return nil +} + +func FiltersToGRPC(fs []Filter) (res []netmap.Filter) { + if fs != nil { + res = make([]netmap.Filter, 0, len(fs)) + + for i := range fs { + res = append(res, *fs[i].ToGRPCMessage().(*netmap.Filter)) + } + } + + return +} + +func FiltersFromGRPC(fs []netmap.Filter) (res []Filter, err error) { + if fs != nil { + res = make([]Filter, len(fs)) + + for i := range fs { + err = res[i].FromGRPCMessage(&fs[i]) + if err != nil { + return + } + } + } + + return +} + +func (s *Selector) ToGRPCMessage() grpc.Message { + var m *netmap.Selector + + if s != nil { + m = new(netmap.Selector) + + m.SetName(s.name) + m.SetAttribute(s.attribute) + m.SetFilter(s.filter) + m.SetCount(s.count) + m.SetClause(ClauseToGRPCMessage(s.clause)) + } + + return m +} + +func (s *Selector) FromGRPCMessage(m grpc.Message) error { + v, ok := m.(*netmap.Selector) + if !ok { + return message.NewUnexpectedMessageType(m, v) + } + + s.name = v.GetName() + s.attribute = v.GetAttribute() + s.filter = v.GetFilter() + s.count = v.GetCount() + s.clause = ClauseFromGRPCMessage(v.GetClause()) + + return nil +} + +func SelectorsToGRPC(ss []Selector) (res []netmap.Selector) { + if ss != nil { + res = make([]netmap.Selector, 0, len(ss)) + + for i := range ss { + res = append(res, *ss[i].ToGRPCMessage().(*netmap.Selector)) + } + } + + return +} + +func SelectorsFromGRPC(ss []netmap.Selector) (res []Selector, err error) { + if ss != nil { + res = make([]Selector, len(ss)) + + for i := range ss { + err = res[i].FromGRPCMessage(&ss[i]) + if err != nil { + return + } + } + } + + return +} + +func (r *Replica) ToGRPCMessage() grpc.Message { + var m *netmap.Replica + + if r != nil { + m = new(netmap.Replica) + + m.SetSelector(r.selector) + m.SetCount(r.count) + m.EcDataCount = r.ecDataCount + m.EcParityCount = r.ecParityCount + } + + return m +} + +func (r *Replica) FromGRPCMessage(m grpc.Message) error { + v, ok := m.(*netmap.Replica) + if !ok { + return message.NewUnexpectedMessageType(m, v) + } + + r.selector = v.GetSelector() + r.count = v.GetCount() + r.ecDataCount = v.GetEcDataCount() + r.ecParityCount = v.GetEcParityCount() + + return nil +} + +func ReplicasToGRPC(rs []Replica) (res []netmap.Replica) { + if rs != nil { + res = make([]netmap.Replica, 0, len(rs)) + + for i := range rs { + res = append(res, *rs[i].ToGRPCMessage().(*netmap.Replica)) + } + } + + return +} + +func ReplicasFromGRPC(rs []netmap.Replica) (res []Replica, err error) { + if rs != nil { + res = make([]Replica, len(rs)) + + for i := range rs { + err = res[i].FromGRPCMessage(&rs[i]) + if err != nil { + return + } + } + } + + return +} + +func (p *PlacementPolicy) ToGRPCMessage() grpc.Message { + var m *netmap.PlacementPolicy + + if p != nil { + m = new(netmap.PlacementPolicy) + + m.SetFilters(FiltersToGRPC(p.filters)) + m.SetSelectors(SelectorsToGRPC(p.selectors)) + m.SetReplicas(ReplicasToGRPC(p.replicas)) + m.SetContainerBackupFactor(p.backupFactor) + m.SetUnique(p.unique) + } + + return m +} + +func (p *PlacementPolicy) FromGRPCMessage(m grpc.Message) error { + v, ok := m.(*netmap.PlacementPolicy) + if !ok { + return message.NewUnexpectedMessageType(m, v) + } + + var err error + + p.filters, err = FiltersFromGRPC(v.GetFilters()) + if err != nil { + return err + } + + p.selectors, err = SelectorsFromGRPC(v.GetSelectors()) + if err != nil { + return err + } + + p.replicas, err = ReplicasFromGRPC(v.GetReplicas()) + if err != nil { + return err + } + + p.backupFactor = v.GetContainerBackupFactor() + + p.unique = v.GetUnique() + + return nil +} + +func ClauseToGRPCMessage(n Clause) netmap.Clause { + return netmap.Clause(n) +} + +func ClauseFromGRPCMessage(n netmap.Clause) Clause { + return Clause(n) +} + +func OperationToGRPCMessage(n Operation) netmap.Operation { + return netmap.Operation(n) +} + +func OperationFromGRPCMessage(n netmap.Operation) Operation { + return Operation(n) +} + +func NodeStateToGRPCMessage(n NodeState) netmap.NodeInfo_State { + return netmap.NodeInfo_State(n) +} + +func NodeStateFromRPCMessage(n netmap.NodeInfo_State) NodeState { + return NodeState(n) +} + +func (a *Attribute) ToGRPCMessage() grpc.Message { + var m *netmap.NodeInfo_Attribute + + if a != nil { + m = new(netmap.NodeInfo_Attribute) + + m.SetKey(a.key) + m.SetValue(a.value) + m.SetParents(a.parents) + } + + return m +} + +func (a *Attribute) FromGRPCMessage(m grpc.Message) error { + v, ok := m.(*netmap.NodeInfo_Attribute) + if !ok { + return message.NewUnexpectedMessageType(m, v) + } + + a.key = v.GetKey() + a.value = v.GetValue() + a.parents = v.GetParents() + + return nil +} + +func AttributesToGRPC(as []Attribute) (res []netmap.NodeInfo_Attribute) { + if as != nil { + res = make([]netmap.NodeInfo_Attribute, 0, len(as)) + + for i := range as { + res = append(res, *as[i].ToGRPCMessage().(*netmap.NodeInfo_Attribute)) + } + } + + return +} + +func AttributesFromGRPC(as []netmap.NodeInfo_Attribute) (res []Attribute, err error) { + if as != nil { + res = make([]Attribute, len(as)) + + for i := range as { + err = res[i].FromGRPCMessage(&as[i]) + if err != nil { + return + } + } + } + + return +} + +func (ni *NodeInfo) ToGRPCMessage() grpc.Message { + var m *netmap.NodeInfo + + if ni != nil { + m = new(netmap.NodeInfo) + + m.SetPublicKey(ni.publicKey) + m.SetAddresses(ni.addresses) + m.SetState(NodeStateToGRPCMessage(ni.state)) + m.SetAttributes(AttributesToGRPC(ni.attributes)) + } + + return m +} + +func (ni *NodeInfo) FromGRPCMessage(m grpc.Message) error { + v, ok := m.(*netmap.NodeInfo) + if !ok { + return message.NewUnexpectedMessageType(m, v) + } + + var err error + + ni.attributes, err = AttributesFromGRPC(v.GetAttributes()) + if err != nil { + return err + } + + ni.publicKey = v.GetPublicKey() + ni.addresses = v.GetAddresses() + ni.state = NodeStateFromRPCMessage(v.GetState()) + + return nil +} + +func (l *LocalNodeInfoRequestBody) ToGRPCMessage() grpc.Message { + var m *netmap.LocalNodeInfoRequest_Body + + if l != nil { + m = new(netmap.LocalNodeInfoRequest_Body) + } + + return m +} + +func (l *LocalNodeInfoRequestBody) FromGRPCMessage(m grpc.Message) error { + v, ok := m.(*netmap.LocalNodeInfoRequest_Body) + if !ok { + return message.NewUnexpectedMessageType(m, v) + } + + return nil +} + +func (l *LocalNodeInfoRequest) ToGRPCMessage() grpc.Message { + var m *netmap.LocalNodeInfoRequest + + if l != nil { + m = new(netmap.LocalNodeInfoRequest) + + m.SetBody(l.body.ToGRPCMessage().(*netmap.LocalNodeInfoRequest_Body)) + l.RequestHeaders.ToMessage(m) + } + + return m +} + +func (l *LocalNodeInfoRequest) FromGRPCMessage(m grpc.Message) error { + v, ok := m.(*netmap.LocalNodeInfoRequest) + if !ok { + return message.NewUnexpectedMessageType(m, v) + } + + var err error + + body := v.GetBody() + if body == nil { + l.body = nil + } else { + if l.body == nil { + l.body = new(LocalNodeInfoRequestBody) + } + + err = l.body.FromGRPCMessage(body) + if err != nil { + return err + } + } + + return l.RequestHeaders.FromMessage(v) +} + +func (l *LocalNodeInfoResponseBody) ToGRPCMessage() grpc.Message { + var m *netmap.LocalNodeInfoResponse_Body + + if l != nil { + m = new(netmap.LocalNodeInfoResponse_Body) + + m.SetVersion(l.version.ToGRPCMessage().(*refsGRPC.Version)) + m.SetNodeInfo(l.nodeInfo.ToGRPCMessage().(*netmap.NodeInfo)) + } + + return m +} + +func (l *LocalNodeInfoResponseBody) FromGRPCMessage(m grpc.Message) error { + v, ok := m.(*netmap.LocalNodeInfoResponse_Body) + if !ok { + return message.NewUnexpectedMessageType(m, v) + } + + var err error + + version := v.GetVersion() + if version == nil { + l.version = nil + } else { + if l.version == nil { + l.version = new(refs.Version) + } + + err = l.version.FromGRPCMessage(version) + if err != nil { + return err + } + } + + nodeInfo := v.GetNodeInfo() + if nodeInfo == nil { + l.nodeInfo = nil + } else { + if l.nodeInfo == nil { + l.nodeInfo = new(NodeInfo) + } + + err = l.nodeInfo.FromGRPCMessage(nodeInfo) + } + + return err +} + +func (l *LocalNodeInfoResponse) ToGRPCMessage() grpc.Message { + var m *netmap.LocalNodeInfoResponse + + if l != nil { + m = new(netmap.LocalNodeInfoResponse) + + m.SetBody(l.body.ToGRPCMessage().(*netmap.LocalNodeInfoResponse_Body)) + l.ResponseHeaders.ToMessage(m) + } + + return m +} + +func (l *LocalNodeInfoResponse) FromGRPCMessage(m grpc.Message) error { + v, ok := m.(*netmap.LocalNodeInfoResponse) + if !ok { + return message.NewUnexpectedMessageType(m, v) + } + + var err error + + body := v.GetBody() + if body == nil { + l.body = nil + } else { + if l.body == nil { + l.body = new(LocalNodeInfoResponseBody) + } + + err = l.body.FromGRPCMessage(body) + if err != nil { + return err + } + } + + return l.ResponseHeaders.FromMessage(v) +} + +func (x *NetworkParameter) ToGRPCMessage() grpc.Message { + var m *netmap.NetworkConfig_Parameter + + if x != nil { + m = new(netmap.NetworkConfig_Parameter) + + m.SetKey(x.k) + m.SetValue(x.v) + } + + return m +} + +func (x *NetworkParameter) FromGRPCMessage(m grpc.Message) error { + v, ok := m.(*netmap.NetworkConfig_Parameter) + if !ok { + return message.NewUnexpectedMessageType(m, v) + } + + x.k = v.GetKey() + x.v = v.GetValue() + + return nil +} + +func (x *NetworkConfig) ToGRPCMessage() grpc.Message { + var m *netmap.NetworkConfig + + if x != nil { + m = new(netmap.NetworkConfig) + + var ps []netmap.NetworkConfig_Parameter + + if ln := len(x.ps); ln > 0 { + ps = make([]netmap.NetworkConfig_Parameter, 0, ln) + + for i := range ln { + ps = append(ps, *x.ps[i].ToGRPCMessage().(*netmap.NetworkConfig_Parameter)) + } + } + + m.SetParameters(ps) + } + + return m +} + +func (x *NetworkConfig) FromGRPCMessage(m grpc.Message) error { + v, ok := m.(*netmap.NetworkConfig) + if !ok { + return message.NewUnexpectedMessageType(m, v) + } + + var ( + ps []NetworkParameter + psV2 = v.GetParameters() + ) + + if psV2 != nil { + ln := len(psV2) + + ps = make([]NetworkParameter, ln) + + for i := range ln { + if err := ps[i].FromGRPCMessage(&psV2[i]); err != nil { + return err + } + } + } + + x.ps = ps + + return nil +} + +func (i *NetworkInfo) ToGRPCMessage() grpc.Message { + var m *netmap.NetworkInfo + + if i != nil { + m = new(netmap.NetworkInfo) + + m.SetMagicNumber(i.magicNum) + m.SetCurrentEpoch(i.curEpoch) + m.SetMsPerBlock(i.msPerBlock) + m.SetNetworkConfig(i.netCfg.ToGRPCMessage().(*netmap.NetworkConfig)) + } + + return m +} + +func (i *NetworkInfo) FromGRPCMessage(m grpc.Message) error { + v, ok := m.(*netmap.NetworkInfo) + if !ok { + return message.NewUnexpectedMessageType(m, v) + } + + var err error + + netCfg := v.GetNetworkConfig() + if netCfg == nil { + i.netCfg = nil + } else { + if i.netCfg == nil { + i.netCfg = new(NetworkConfig) + } + + err = i.netCfg.FromGRPCMessage(netCfg) + if err != nil { + return err + } + } + + i.magicNum = v.GetMagicNumber() + i.curEpoch = v.GetCurrentEpoch() + i.msPerBlock = v.GetMsPerBlock() + + return nil +} + +func (l *NetworkInfoRequestBody) ToGRPCMessage() grpc.Message { + var m *netmap.NetworkInfoRequest_Body + + if l != nil { + m = new(netmap.NetworkInfoRequest_Body) + } + + return m +} + +func (l *NetworkInfoRequestBody) FromGRPCMessage(m grpc.Message) error { + v, ok := m.(*netmap.NetworkInfoRequest_Body) + if !ok { + return message.NewUnexpectedMessageType(m, v) + } + + return nil +} + +func (l *NetworkInfoRequest) ToGRPCMessage() grpc.Message { + var m *netmap.NetworkInfoRequest + + if l != nil { + m = new(netmap.NetworkInfoRequest) + + m.SetBody(l.body.ToGRPCMessage().(*netmap.NetworkInfoRequest_Body)) + l.RequestHeaders.ToMessage(m) + } + + return m +} + +func (l *NetworkInfoRequest) FromGRPCMessage(m grpc.Message) error { + v, ok := m.(*netmap.NetworkInfoRequest) + if !ok { + return message.NewUnexpectedMessageType(m, v) + } + + var err error + + body := v.GetBody() + if body == nil { + l.body = nil + } else { + if l.body == nil { + l.body = new(NetworkInfoRequestBody) + } + + err = l.body.FromGRPCMessage(body) + if err != nil { + return err + } + } + + return l.RequestHeaders.FromMessage(v) +} + +func (i *NetworkInfoResponseBody) ToGRPCMessage() grpc.Message { + var m *netmap.NetworkInfoResponse_Body + + if i != nil { + m = new(netmap.NetworkInfoResponse_Body) + + m.SetNetworkInfo(i.netInfo.ToGRPCMessage().(*netmap.NetworkInfo)) + } + + return m +} + +func (i *NetworkInfoResponseBody) FromGRPCMessage(m grpc.Message) error { + v, ok := m.(*netmap.NetworkInfoResponse_Body) + if !ok { + return message.NewUnexpectedMessageType(m, v) + } + + var err error + + netInfo := v.GetNetworkInfo() + if netInfo == nil { + i.netInfo = nil + } else { + if i.netInfo == nil { + i.netInfo = new(NetworkInfo) + } + + err = i.netInfo.FromGRPCMessage(netInfo) + } + + return err +} + +func (l *NetworkInfoResponse) ToGRPCMessage() grpc.Message { + var m *netmap.NetworkInfoResponse + + if l != nil { + m = new(netmap.NetworkInfoResponse) + + m.SetBody(l.body.ToGRPCMessage().(*netmap.NetworkInfoResponse_Body)) + l.ResponseHeaders.ToMessage(m) + } + + return m +} + +func (l *NetworkInfoResponse) FromGRPCMessage(m grpc.Message) error { + v, ok := m.(*netmap.NetworkInfoResponse) + if !ok { + return message.NewUnexpectedMessageType(m, v) + } + + var err error + + body := v.GetBody() + if body == nil { + l.body = nil + } else { + if l.body == nil { + l.body = new(NetworkInfoResponseBody) + } + + err = l.body.FromGRPCMessage(body) + if err != nil { + return err + } + } + + return l.ResponseHeaders.FromMessage(v) +} + +func (x *NetMap) ToGRPCMessage() grpc.Message { + var m *netmap.Netmap + + if x != nil { + m = new(netmap.Netmap) + + m.SetEpoch(x.epoch) + + if x.nodes != nil { + nodes := make([]netmap.NodeInfo, len(x.nodes)) + + for i := range x.nodes { + nodes[i] = *x.nodes[i].ToGRPCMessage().(*netmap.NodeInfo) + } + + m.SetNodes(nodes) + } + } + + return m +} + +func (x *NetMap) FromGRPCMessage(m grpc.Message) error { + v, ok := m.(*netmap.Netmap) + if !ok { + return message.NewUnexpectedMessageType(m, v) + } + + var err error + + nodes := v.GetNodes() + if nodes == nil { + x.nodes = nil + } else { + x.nodes = make([]NodeInfo, len(nodes)) + + for i := range nodes { + err = x.nodes[i].FromGRPCMessage(&nodes[i]) + if err != nil { + return err + } + } + } + + x.epoch = v.GetEpoch() + + return nil +} + +func (x *SnapshotRequestBody) ToGRPCMessage() grpc.Message { + var m *netmap.NetmapSnapshotRequest_Body + + if x != nil { + m = new(netmap.NetmapSnapshotRequest_Body) + } + + return m +} + +func (x *SnapshotRequestBody) FromGRPCMessage(m grpc.Message) error { + v, ok := m.(*netmap.NetmapSnapshotRequest_Body) + if !ok { + return message.NewUnexpectedMessageType(m, v) + } + + return nil +} + +func (x *SnapshotRequest) ToGRPCMessage() grpc.Message { + var m *netmap.NetmapSnapshotRequest + + if x != nil { + m = new(netmap.NetmapSnapshotRequest) + + m.SetBody(x.body.ToGRPCMessage().(*netmap.NetmapSnapshotRequest_Body)) + x.RequestHeaders.ToMessage(m) + } + + return m +} + +func (x *SnapshotRequest) FromGRPCMessage(m grpc.Message) error { + v, ok := m.(*netmap.NetmapSnapshotRequest) + if !ok { + return message.NewUnexpectedMessageType(m, v) + } + + var err error + + body := v.GetBody() + if body == nil { + x.body = nil + } else { + if x.body == nil { + x.body = new(SnapshotRequestBody) + } + + err = x.body.FromGRPCMessage(body) + if err != nil { + return err + } + } + + return x.RequestHeaders.FromMessage(v) +} + +func (x *SnapshotResponseBody) ToGRPCMessage() grpc.Message { + var m *netmap.NetmapSnapshotResponse_Body + + if x != nil { + m = new(netmap.NetmapSnapshotResponse_Body) + + m.SetNetmap(x.netMap.ToGRPCMessage().(*netmap.Netmap)) + } + + return m +} + +func (x *SnapshotResponseBody) FromGRPCMessage(m grpc.Message) error { + v, ok := m.(*netmap.NetmapSnapshotResponse_Body) + if !ok { + return message.NewUnexpectedMessageType(m, v) + } + + var err error + + netMap := v.GetNetmap() + if netMap == nil { + x.netMap = nil + } else { + if x.netMap == nil { + x.netMap = new(NetMap) + } + + err = x.netMap.FromGRPCMessage(netMap) + } + + return err +} + +func (x *SnapshotResponse) ToGRPCMessage() grpc.Message { + var m *netmap.NetmapSnapshotResponse + + if x != nil { + m = new(netmap.NetmapSnapshotResponse) + + m.SetBody(x.body.ToGRPCMessage().(*netmap.NetmapSnapshotResponse_Body)) + x.ResponseHeaders.ToMessage(m) + } + + return m +} + +func (x *SnapshotResponse) FromGRPCMessage(m grpc.Message) error { + v, ok := m.(*netmap.NetmapSnapshotResponse) + if !ok { + return message.NewUnexpectedMessageType(m, v) + } + + var err error + + body := v.GetBody() + if body == nil { + x.body = nil + } else { + if x.body == nil { + x.body = new(SnapshotResponseBody) + } + + err = x.body.FromGRPCMessage(body) + if err != nil { + return err + } + } + + return x.ResponseHeaders.FromMessage(v) +} diff --git a/api/netmap/grpc/service_frostfs.pb.go b/api/netmap/grpc/service_frostfs.pb.go new file mode 100644 index 0000000000000000000000000000000000000000..7e2d7a367dce459c5086ab87413ab1afd9f69c85 GIT binary patch literal 55007 zcmeHQS(Dqw5q>6r#fVk5Nkt|t`%NjODz=
zvPrXLI%TiEV{g9ufqnhz^$*kGa2w3u1y>=9!)zUFhr?*S-6R Y*uxtaZt?82{Bo897@;BqGKe3~xaffTb~ z8_kk%na;4XY-5im(lAY )B7yJliBu@NJkS(L9~$Qc&afRd^RBTmV&k9Ym{S$4lWL&O{L`dz(aA zRbf$PG@J~F+5I+T-)!c=>J2#CYZ&Xd;s17Fn*I9aW^vEbEZNO7_I~(bIJ^lG^r2sw zQfB%++UVOLNw0$y^vT#uHh$`DY%-3c)uh &k$6s{KRCRo7`u {BtEKj)@dTTJzF#u9DLfNpfveI?@Bxj;3E@*URx}9>nn`gGm-FU{AZt z d^-(%{#zx*1X9Q!09z955md&qG1m69d-I_}Ww~W8TT(I5!-xuG#$y+|R@H8~H z^d@W=r=)OhZXaplTXb%@gin9wEBSkn@J{*R*RXf++8^H Zc zbUb;+ZiW3or#g-*MbjN@b{TtWOX6HPpxpFraC-r}R(v%c#k Q?Zqc{)NfYA2!=>!kch~;zjZFbqZTg!k0UkK>WfRusgUE zuH|^i)W9nPC<4 SM+ztP6W)Wo3Zun)c^BSG zX?~2;U>S;e1zXs78LZL}dx2^{2XXe*X0?fVHx6O3kQT$#Sg!+KV=#x+C5m^U6k5(> z>>L$?F)7r6Yeon3Er&s%0JX;l^gnMPWMZHJ4~bNy;NN;hiQjOocrm(!FlzMMZ~%c_ z@Gn@wc%1YUL*=QAw5&T;-L 1rSPv5qs}0PK2396Q`q$4@K#lUvcYNvnGIM(#ocdZ zoF2sYBCz4P%nsvJ$ggj&BM1@}5kwgfZeZnGhN%+lY&;C-n+7Cn5d3E0>Yj`E9fbVJ zZIp)3tRz5+4ePmR6cYnp9i+l0voKaN>CtMC3lM8n 7i7A83gHI7+OG|$_Iw(em*#SXdX1lCt9bS0!>)M><4#hM_S5*NW8^YF`6uhE4hSj zJ^lN38^#Nn*NvZJK8H~OwqRB^yB%}AqP5RyYjn-C+oN|XTKqz^m *gX8 zrfaEA&L-=tO8DkM8nWjv80;;$~mkSTZ>knJIR{W#XYK i6 zdc$EZe%s0l3TC4%v}ZW*GWa9+Td|v#LJ||ELy{7J6Q?Au>2%@Md%Bq24ST(ifzVA! zuRQzJ^Z2GJ)90!f=Y$m9O5DtyQf*AJ#+C$~Ozkmf=9y5~zqLKSWc8GzE|gcV=z*nb z(x&!Nl-AFyS` fO+gHAl4T!nfYuT4ETepl06(t4wf=v zDi6a4OA{86LBf?otsT(L5`XfQ2=j}Pnbf1ARPO!P63Jae^jR*CStE*4(aBpO7oWOT z JShS9ubZwxLq7C(91L*^B?O=d@x`nMG%uyQ @ z7R(Wjpq2vn9Ew+d1~p`IgB9%{Rlzt1#3L=6 X+)2BF0J&t-Sq^08hhl&o8wa`(0ev0Ff^XNyLW5$NKU4~;oc8#*iseE& z3^*>j;+9D}c8{^_fw9Ob 8TC=}q2zzPT$%Gw_KbrM8FVIC7U z7i?s1wQQ-=U>XX06?24w-FOlC_ELM6YmBg#aSh2-Da9w7kqt$w5RZG=hNR#6bVJoF zP$!LxD!3S4H52#Etx}Clv8I?S*UMBVG`ze3QZ-XnBCtMcC{d95QbEO{FP5rG+uBQ6 z+CV|wqA&(J*ST}M(6eUvbqjUYVrg9Ni~#3L*hniuT)x{L3u<2X#6$k#x_B$SMdG1a zwsbVy1#G;h9wJ-}w6gu*J#Rui6h6RXZVRT)aSur`9zIo|<+y%O&{{}jCbk1Cm&`)Z zk=c1Ip*&^kNMu?p%5_7y!&Q LjQMvttG*h!_e3q-J=WlgrCTn&&D(eXi zukdN20ih>&1Zk6<;1W>Aa2*2kG{C4)rJk1n$JK)o;JF$sQvHEQ mQHtVCi@sc%Y?HL(YRp_X^+j;&B^hg7AF^ z%|^ D{A$KA<`F@kRNdf`8~NV0Vr~X z?|KwIA@@3fo)D{Z0D?%z&!aK(pfo2Z)k f0B|?^ki8|N`Z5NC3>Dxw|Gj*$Mnv5ZtzG0WCUZl z)*%t>#KJ=4gAabNuu0}rSS4_7l!_uESf@QGga_|k|DKz0T=SHsfEt-wEnDg|xJtra zo@Y$dx99_<_AFPMtfUqw@ vA#VTq_=9Fx#` z#C>zCR2NgM8Fa`sGL;FLC*OVL(JD@R`B_V%{i64ks!2U@4@GJ9f~rMP9*wx9T%m(_ zX-{3^JF5dPhV7-NF4^D4yE#?rlA3@z<~=^{+>@7hh}+t6Dvs@nZ{I%Pxfp);N2$Pf zKt%3p8O=KE#j&f3!X}U;PgDfD&%be`3C3+h`0HSM5eM7!dXv2kIYUc?j_}dX-8_SE z1ZWBP^}q3O@GG;CY`H6NXc<3P20rC!&7jhD3wptUHtM!e35i$Az+0eQtEUk4Wat(^ z?$dT}ep1_1Bd@@xjOuxnYDhO~iv#49PfvqWhq*i&R-s*vwEG|jt$4)I`NT0_ z???xHP2+KoTk&eB-Tm<++-jM$-Ft3DBW?xE;lqC7))~l@aT`2i%j1eV)P@`#U&kZ$ zKP= OsBcRhaVDGoiwAxznAlwGszI^3V) z;FtmDN^)?idSWyOmkQ^V67^JvE>Im@ex~=)^UDIO62Oif%u*a3@l+L2L;6s T4OePt>VS~o zu|Hvc37o;oSHybyLxxWvgH;+r2!cr4d=sY`+a}>MI%)cYYZB@>xC`)bPl4b;Z<`c| zq7OJp5T0T@bZU(Tk;8=17K0_s#fsAjE95ajF-~i2l+WoP4cW*7iC^G7Jeg`Rz{mef z_LLrp$oFIGXcn)0#iDiWMmKyaY3o?O;6mj%5CajC@-tk(85F7>bHc-e@O09h!M2-` zB%Dn&!`A&x=oPUFVJ|k(VVs=`yEjpy{y{F_GT`%%O02LnJR2C%(yAe~qRvS!IJcj` z5*w{B8xiMnz_sI^-vR;{hL&XkJvlY3af+XV&ZqlC#RlM=Xa#qnwhwTD@vz=kl6!;L znZ=I+s3I=6heN@(2!Fh9a`;^c(sUu4JgNmsmL}q&@V>o?Gg_bzc{~gJ%bd)DT_5mJ z7HG~hZMv3f6Nf+RggfTpQf105=ZYL9iafxy0x!3yM2Gy$sOZ?^G `+WVwbx!QWZ*2|`3?an7sJ!7e7EFGS)B$f&At>`Ka7q5aWxQPA; zAx@2Ad@n^syLVIlCiL_o3ocioD26Yt%x1aTOoYfO xoOAk4g#mH6e5XEi$*-LyUOla~jMgVXtDoP_P?)Xw;tN znqzIH+$Eu%c;Ez44KUdD{k)W84wkZ)NY4_5lXDsmMy6lNayAT6m+{lK4{mWv;A1cq z<`+hbcT*#+gc<9oNL{L+WYHx{u~XVsV+56tByHjNC%8BVrJLaDS>vjgJ(0- yXSk4aMxhmQ qtts NZYnE8lZSeF`Tr<=2;~3( literal 0 HcmV?d00001 diff --git a/api/netmap/grpc/service_frostfs_fuzz.go b/api/netmap/grpc/service_frostfs_fuzz.go new file mode 100644 index 0000000..ebb59bc --- /dev/null +++ b/api/netmap/grpc/service_frostfs_fuzz.go @@ -0,0 +1,121 @@ +//go:build gofuzz +// +build gofuzz + +// Code generated by protoc-gen-go-frostfs. DO NOT EDIT. + +package netmap + +func DoFuzzProtoLocalNodeInfoRequest(data []byte) int { + msg := new(LocalNodeInfoRequest) + if err := msg.UnmarshalProtobuf(data); err != nil { + return 0 + } + _ = msg.MarshalProtobuf(nil) + return 1 +} +func DoFuzzJSONLocalNodeInfoRequest(data []byte) int { + msg := new(LocalNodeInfoRequest) + if err := msg.UnmarshalJSON(data); err != nil { + return 0 + } + _, err := msg.MarshalJSON() + if err != nil { + panic(err) + } + return 1 +} +func DoFuzzProtoLocalNodeInfoResponse(data []byte) int { + msg := new(LocalNodeInfoResponse) + if err := msg.UnmarshalProtobuf(data); err != nil { + return 0 + } + _ = msg.MarshalProtobuf(nil) + return 1 +} +func DoFuzzJSONLocalNodeInfoResponse(data []byte) int { + msg := new(LocalNodeInfoResponse) + if err := msg.UnmarshalJSON(data); err != nil { + return 0 + } + _, err := msg.MarshalJSON() + if err != nil { + panic(err) + } + return 1 +} +func DoFuzzProtoNetworkInfoRequest(data []byte) int { + msg := new(NetworkInfoRequest) + if err := msg.UnmarshalProtobuf(data); err != nil { + return 0 + } + _ = msg.MarshalProtobuf(nil) + return 1 +} +func DoFuzzJSONNetworkInfoRequest(data []byte) int { + msg := new(NetworkInfoRequest) + if err := msg.UnmarshalJSON(data); err != nil { + return 0 + } + _, err := msg.MarshalJSON() + if err != nil { + panic(err) + } + return 1 +} +func DoFuzzProtoNetworkInfoResponse(data []byte) int { + msg := new(NetworkInfoResponse) + if err := msg.UnmarshalProtobuf(data); err != nil { + return 0 + } + _ = msg.MarshalProtobuf(nil) + return 1 +} +func DoFuzzJSONNetworkInfoResponse(data []byte) int { + msg := new(NetworkInfoResponse) + if err := msg.UnmarshalJSON(data); err != nil { + return 0 + } + _, err := msg.MarshalJSON() + if err != nil { + panic(err) + } + return 1 +} +func DoFuzzProtoNetmapSnapshotRequest(data []byte) int { + msg := new(NetmapSnapshotRequest) + if err := msg.UnmarshalProtobuf(data); err != nil { + return 0 + } + _ = msg.MarshalProtobuf(nil) + return 1 +} +func DoFuzzJSONNetmapSnapshotRequest(data []byte) int { + msg := new(NetmapSnapshotRequest) + if err := msg.UnmarshalJSON(data); err != nil { + return 0 + } + _, err := msg.MarshalJSON() + if err != nil { + panic(err) + } + return 1 +} +func DoFuzzProtoNetmapSnapshotResponse(data []byte) int { + msg := new(NetmapSnapshotResponse) + if err := msg.UnmarshalProtobuf(data); err != nil { + return 0 + } + _ = msg.MarshalProtobuf(nil) + return 1 +} +func DoFuzzJSONNetmapSnapshotResponse(data []byte) int { + msg := new(NetmapSnapshotResponse) + if err := msg.UnmarshalJSON(data); err != nil { + return 0 + } + _, err := msg.MarshalJSON() + if err != nil { + panic(err) + } + return 1 +} diff --git a/api/netmap/grpc/service_frostfs_test.go b/api/netmap/grpc/service_frostfs_test.go new file mode 100644 index 0000000..5c9035f --- /dev/null +++ b/api/netmap/grpc/service_frostfs_test.go @@ -0,0 +1,71 @@ +//go:build gofuzz +// +build gofuzz + +// Code generated by protoc-gen-go-frostfs. DO NOT EDIT. + +package netmap + +import ( + testing "testing" +) + +func FuzzProtoLocalNodeInfoRequest(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzProtoLocalNodeInfoRequest(data) + }) +} +func FuzzJSONLocalNodeInfoRequest(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzJSONLocalNodeInfoRequest(data) + }) +} +func FuzzProtoLocalNodeInfoResponse(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzProtoLocalNodeInfoResponse(data) + }) +} +func FuzzJSONLocalNodeInfoResponse(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzJSONLocalNodeInfoResponse(data) + }) +} +func FuzzProtoNetworkInfoRequest(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzProtoNetworkInfoRequest(data) + }) +} +func FuzzJSONNetworkInfoRequest(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzJSONNetworkInfoRequest(data) + }) +} +func FuzzProtoNetworkInfoResponse(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzProtoNetworkInfoResponse(data) + }) +} +func FuzzJSONNetworkInfoResponse(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzJSONNetworkInfoResponse(data) + }) +} +func FuzzProtoNetmapSnapshotRequest(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzProtoNetmapSnapshotRequest(data) + }) +} +func FuzzJSONNetmapSnapshotRequest(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzJSONNetmapSnapshotRequest(data) + }) +} +func FuzzProtoNetmapSnapshotResponse(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzProtoNetmapSnapshotResponse(data) + }) +} +func FuzzJSONNetmapSnapshotResponse(f *testing.F) { + f.Fuzz(func(t *testing.T, data []byte) { + DoFuzzJSONNetmapSnapshotResponse(data) + }) +} diff --git a/api/netmap/grpc/service_grpc.pb.go b/api/netmap/grpc/service_grpc.pb.go new file mode 100644 index 0000000000000000000000000000000000000000..9ad7fefcd78b6a4cdd7017115730f1d8e6f93540 GIT binary patch literal 9140 zcmeHNZEqX75&rD`6|4%Rk|D3s6vY7!f+A4ttI&%bSWdqbg+p_>(!#6d9?6wX2>jps z47n?DrM0bVoF+gMH$+chh-vPzubyOWtZndwr* z?A;~1xV&a(? rnY+CTr%$40W{s7x+TWbXk16m%8^Gp^lP82@jKfV1hVJTwJ zl4K3f5q@I8IRuPc@WN&SbkeFS5+>|#tTLbz{hTX<(2r!15Ce?Wk}v2d1ltT*nM)2y zEmNt9baQJ ~Bp{#w0%qvcft1y3K4hwpY{A!T1z-u5 z8H~iTE=a**jV-iQAZ2-e=l-2br_I@0D9hHEUSf;+`OYn}ib%`e^YRl-N;5=DRhF%2 z!&C6 {;JHzL0>{6n893k#Wc>rw?hN zDji95RTSWagVYU8T{4r%))b->c6@yKkK<$3e>r5+v+?!h^5SOtaXdboP6u8j3<19- zH;BnZRv0 _DhYyD%mf#9V)%MpCT`pqimZzsQeygIuXU%r2T zc`<;6tyIt_s@I7;1_ddz55Wg^+!CV{Ck;kaX$_0x*qu>hp68b(v1Y*fyAw=_oF|7& zYOVCZnRx|+yI8RG_gpJu&!=?iP;8$aPf=-29IEwWyC6S5FT6D^DVIJBZXk3Bv@P*Y zV6o*3vvCXgmYF(VGL9G7Q&zK+nhdW(scw4X;la$^;PYKKi$1-|B Q0h@!!{O= zmPVAxkh7{faUpL(Epj+%{sR`zax$es5g%Q;ijM9#$hx^0cp6EBw5g#U#xW2&1jUy4 zqZoEb9*kfxcc(i0ALAEnzegbq z>y9$Ev9rmGKg!r2W$f!OE@OS^x-DUM%TZUq{wPy7w}2q&_xI}m<6yMTumoM}+jbVM z>-+!PGIWoAb)u|`@HU9S32`NNa72-aeEL+*Z89`&v@S^}$kVu_&iN7-|D8OH#?~f* z`}|%v7oJ|kH4~7Zo65ES)Ret~au1iCV(+kRNk>Fo>bfG$_X6WHx>}~7FWQT7%d{cu zEnI`FUEt#CjwVra*vX_ffNQQMKftqq{m-#uzQCuzk}vfrpklXEfxy8e{xa8s6+6B2 z`@`^5)_bf~{!r6tXIG$S;rc?j2=DF )5v;<``MAMn^gm$2(9O^ICS(&Y}JSWZQ&r~^hh+doPUHiV9>ywzx)_lmnttr?O zi^|wDD$-iy?|`LGME6X;lC#X9gV-g77B;;-)CHxN)Lajl(aSEE1{7%%H!ACS{TD6W zH}%)=q!CWj))s0!Q@B0sJ6E~Cqx+gm-KVuTUVpwHy6)`Wk(EfnXeajr(SdJ1(JclJ z3t3u}sm{ZX1=nj@R_e)i3};+0yJ=mKkBC+GW!2}e&7~4TFa%l#uMfFo;wl=N_Ga%& z3g6>?3nXv!Aa3PRe=o+W?&?adL^AiqzN@K%Ge=&>se2s43h3M(ZJ!iQ+kxH*%2tk^ z9xxME<+%?OE$JfA*;3T<*D#6?L)|3$-{m^A$9o+}9~+NJeR>3FH<%GSQZ>Zc>FD+# zIT3~Cet+|QQZulyN!8zgn1K}y#9xD!!7{tI>~BKKz#i|mr)2PT6O(5oWN=h|znlz= zY#{hHv tf9-1z z655lx1KB1$J+3*R;vvcd@m>97y?;pua1C<)3x^zy14y25%U!^TgB|}724>tx{{eLH BQ#Akp literal 0 HcmV?d00001 diff --git a/api/netmap/grpc/types_frostfs.pb.go b/api/netmap/grpc/types_frostfs.pb.go new file mode 100644 index 0000000000000000000000000000000000000000..9dc0de0e7b1cf4cfb60d5cbd33965cc37d36fe2b GIT binary patch literal 57425 zcmeHQYjaz-(f&;RiWAN>rSd9Ne2LRK<9Eif<&0Wea$~#w;Cft&I G>+Im^;p^tk&ZyNsYkkPGK|bxbMmsy*{%AO!W{sWN?~~ym ztLKCE@U%Pluup&1cWRyfG^^wP@at$e>}B;2-D%VE?hZP`{nz7J{-igYojz}!?3*v& zPfpJu#C~hk-Jeany?x3{EJ(z)c~E9Bu_9iG&%WhC3|o`WPO^`)lV*F^-~VT~Jspm_ ztrz)p+-* 3#A$KaiTtTVpb0{qTO!>gVW0 z82Ve*nRies*!CYgwb~C4v#g#}m--$)-=WV3|KQ*6((gwH{QEun{nyw0`+fdh(to7i zpG*1!{$0|4qTi2yWBR|)?`SFh }JyR8Ct#D$>w|6@GN`y z$WgMQiHbGOci;U?->HYj`E)iOWOMFR^9NXwgKIt;j{B|Y;b7W8hd#KEe0$lQA9i=X zs36~|BJXC;#>2jmW@4n-%}!`QS|}5%&fAHT=DTy)g7R$4cb$Y&jA6czomOv>Llp76 zPkOD{BrmvKNq+z2`QtaQ4g!a3GO)KbRIeVth;3M@;A9Qe)5BM<509Ss-a)DEdM0kj z%k8^fluWD*#zx}sA29B^E!9|`VP0x_3D(x`n02gaVsu)N|CuZ l!Wzb<+C!m47m?7wWW~W!pmo#g8 z(Hc)awtA>i1{-XACoFawgKiI27*oF)^p!qEO*L=&ETw7-cr&%naxpc$;Ql_mUhAZn zzv}*9o{7(zWYdp%Hlc)JCp(8Dkh%8Btdn)vjWv-^vQmf5#`z@s)SZ5meAIQFZr(f1 zxG^W$$JRV|pH8}i*7!5CZ#v&+TF4UyZn(iRz+dhfw+D~*C_^Lq-WD@F453W
he>s~r`XD%O9`w6YD|e&ce34HkSh%U$H>lC6Y|QU$86*o* zO*7yRY6Uy4TcgWD#{KL&UD*^0DbX`-aLfjJ`2ae=3YC3>5s!Pa-)}x1jq(9?A10iE z(xIO;8>!ghT%Hz6CUncayP|Q4T!;L%pqPGnqK1jQM$SN2k+D==ef^7i%U0iXTH{CR ztS?3<)fd^`E!RpVhr*QUrP5ipjmfyJn%T(5;|%{Bj=4|CPj}jx>-lHopq>n&_vcfo zH+3vH>}LOKIu(EH5esNsr` 9g Kzsi{q<3~m%G4 |1y`EzE%CMa4QV%1lTHZXB1yaBustLh{>gu&n24WbAAV#-Hg5wbc&=n?m zD!86>Ly(m^5q?Nj39SHzbH|W9H_f-8+>1nD<*(KZ#x$K23mZb1MonSOY>^~JVIhky z8Oj>Oko&E&K;h6XmiJtFo3=3@Oapl{z`wmZK8nZLXdO{DxSDAfaY2O|GT&ocF$3x+ zGDQRRHvh$l?;mK{b2~LV0LIL@zx@R3GWPJP`OB<>i6H*d?VbLDDdeO9803?TmMNVu zawxaqY?^%+7Mq;2fv)*m>yz=i^}(#yWA7w_aHDs+W6UN|3jsq+4ic*I=Rf`oq0xIh zlnNpJRSSsrh^Y1@%YmKnC|wguxG08GWJDL#dNj^E-E+H~eU#Pz%`4e|*B=4{w``5H zLQA} uOb&*1Y||uZ&|G2Xo*yF>P#rLoid{V;g05u;Jdj)%KV0& zGfw9BD)oZQC*;#*RCZ4I#FF|3+QHuGQu?7I_(uidV1QyyZ`o2x7Z=o3UAItCn-B(K zYNTW}xESN~unaDE8?*@-4iQrGs08mw0;C3sUyvxvW}Q$hbel?1bydR7kESKMB*8=j zXaKB)r!f*QYd)vI+8;Czd&pC4$awDNH|qDkr8$c6qGNHFn71WnCn~3A#Zu!tzcBHa zlh?!1^L(E7#Aup_6Tm=YUaM1243KaHrwQk8%)0AxP)ZFXSUyc=eV`Mz$&vQztcytj z2s}@V ^2(| zb(M}Vf={?|Tpn6J?Mm5Ez4d)5s_vKXub0HPQuCc)L43yYayub~{Ir?jqg?dSy!n!< z0TCPcgt$$1?=GnTejp2x`b%$09W)=Go*s~u(dysOsv=TG>J$7dlaITt(;K!Bs4>-F zm_>>a#-tP;As1%muTej8vlY8iTAcHx6-@fHpccj_7a5AHWQL>|y)J`yUMd>RSfMy+ zXm)I#@SSWKXgCr|GSg&_e!Rt)=W$Hib8)%=q`&sEj`@ay#gf%7P4B3GM_H7G%W-0| zCzqZi5n?t-bF}2(d0+{B7YKaU#W)pjIe3-#a@qtzh8_hXG*l49A)X9pgDE5IGoE_b zC_$UjDUKkq5TY=@83t_*vJu7ODeUZIHU;F)24`8cY`P0a5Sx%ldVZx~gso-Oa72+l zT1b#o=7+);3+Q!Z;;6`C`fC)F(3)IkSmF)OH96?OfG}$q^f-(bqyiJ7V9SFOs&ECa zuUG-OY(yNaC4$UMJxB&KWs1TA@mh10NP}zUi0RFVHMlBVi)h0+931(jVu0ced#OgC zS>!Wm&L=A;qeY3lte6ci;NTQaSO{|G)GNV5G_1ja2S>