You can not select more than 25 topics Topics must start with a chinese character,a letter or number, can include dashes ('-') and can be up to 35 characters long.

ec.go 5.4 kB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223
  1. package ops2
  2. import (
  3. "context"
  4. "fmt"
  5. "io"
  6. "github.com/samber/lo"
  7. "gitlink.org.cn/cloudream/common/pkgs/future"
  8. "gitlink.org.cn/cloudream/common/pkgs/ioswitch/dag"
  9. "gitlink.org.cn/cloudream/common/pkgs/ioswitch/exec"
  10. "gitlink.org.cn/cloudream/common/pkgs/ioswitch/utils"
  11. cdssdk "gitlink.org.cn/cloudream/common/sdks/storage"
  12. "gitlink.org.cn/cloudream/common/utils/io2"
  13. "gitlink.org.cn/cloudream/common/utils/sync2"
  14. "gitlink.org.cn/cloudream/storage/common/pkgs/ec"
  15. "gitlink.org.cn/cloudream/storage/common/pkgs/ec/lrc"
  16. )
  17. func init() {
  18. exec.UseOp[*GalMultiply]()
  19. }
  20. type GalMultiply struct {
  21. Coef [][]byte `json:"coef"`
  22. Inputs []*exec.StreamVar `json:"inputs"`
  23. Outputs []*exec.StreamVar `json:"outputs"`
  24. ChunkSize int `json:"chunkSize"`
  25. }
  26. func (o *GalMultiply) Execute(ctx context.Context, e *exec.Executor) error {
  27. err := exec.BindArrayVars(e, ctx, o.Inputs)
  28. if err != nil {
  29. return err
  30. }
  31. defer func() {
  32. for _, s := range o.Inputs {
  33. s.Stream.Close()
  34. }
  35. }()
  36. outputWrs := make([]*io.PipeWriter, len(o.Outputs))
  37. for i := range o.Outputs {
  38. rd, wr := io.Pipe()
  39. o.Outputs[i].Stream = rd
  40. outputWrs[i] = wr
  41. }
  42. fut := future.NewSetVoid()
  43. go func() {
  44. mul := ec.GaloisMultiplier().BuildGalois()
  45. inputChunks := make([][]byte, len(o.Inputs))
  46. for i := range o.Inputs {
  47. inputChunks[i] = make([]byte, o.ChunkSize)
  48. }
  49. outputChunks := make([][]byte, len(o.Outputs))
  50. for i := range o.Outputs {
  51. outputChunks[i] = make([]byte, o.ChunkSize)
  52. }
  53. for {
  54. err := sync2.ParallelDo(o.Inputs, func(s *exec.StreamVar, i int) error {
  55. _, err := io.ReadFull(s.Stream, inputChunks[i])
  56. return err
  57. })
  58. if err == io.EOF {
  59. fut.SetVoid()
  60. return
  61. }
  62. if err != nil {
  63. fut.SetError(err)
  64. return
  65. }
  66. err = mul.Multiply(o.Coef, inputChunks, outputChunks)
  67. if err != nil {
  68. fut.SetError(err)
  69. return
  70. }
  71. for i := range o.Outputs {
  72. err := io2.WriteAll(outputWrs[i], outputChunks[i])
  73. if err != nil {
  74. fut.SetError(err)
  75. return
  76. }
  77. }
  78. }
  79. }()
  80. exec.PutArrayVars(e, o.Outputs)
  81. err = fut.Wait(ctx)
  82. if err != nil {
  83. for _, wr := range outputWrs {
  84. wr.CloseWithError(err)
  85. }
  86. return err
  87. }
  88. for _, wr := range outputWrs {
  89. wr.Close()
  90. }
  91. return nil
  92. }
  93. func (o *GalMultiply) String() string {
  94. return fmt.Sprintf(
  95. "ECMultiply(coef=%v) (%v) -> (%v)",
  96. o.Coef,
  97. utils.FormatVarIDs(o.Inputs),
  98. utils.FormatVarIDs(o.Outputs),
  99. )
  100. }
  101. type LRCConstructAnyNode struct {
  102. dag.NodeBase
  103. LRC cdssdk.LRCRedundancy
  104. InputIndexes []int
  105. OutputIndexes []int
  106. }
  107. func (b *GraphNodeBuilder) NewLRCConstructAny(lrc cdssdk.LRCRedundancy) *LRCConstructAnyNode {
  108. node := &LRCConstructAnyNode{
  109. LRC: lrc,
  110. }
  111. b.AddNode(node)
  112. return node
  113. }
  114. func (t *LRCConstructAnyNode) AddInput(str *dag.StreamVar, dataIndex int) {
  115. t.InputIndexes = append(t.InputIndexes, dataIndex)
  116. idx := t.InputStreams().EnlargeOne()
  117. str.Connect(t, idx)
  118. }
  119. func (t *LRCConstructAnyNode) RemoveAllInputs() {
  120. for i, in := range t.InputStreams().RawArray() {
  121. in.Disconnect(t, i)
  122. }
  123. t.InputStreams().Resize(0)
  124. t.InputIndexes = nil
  125. }
  126. func (t *LRCConstructAnyNode) NewOutput(dataIndex int) *dag.StreamVar {
  127. t.OutputIndexes = append(t.OutputIndexes, dataIndex)
  128. output := t.Graph().NewStreamVar()
  129. t.OutputStreams().SetupNew(t, output)
  130. return output
  131. }
  132. func (t *LRCConstructAnyNode) GenerateOp() (exec.Op, error) {
  133. l, err := lrc.New(t.LRC.N, t.LRC.K, t.LRC.Groups)
  134. if err != nil {
  135. return nil, err
  136. }
  137. coef, err := l.GenerateMatrix(t.InputIndexes, t.OutputIndexes)
  138. if err != nil {
  139. return nil, err
  140. }
  141. return &GalMultiply{
  142. Coef: coef,
  143. Inputs: lo.Map(t.InputStreams().RawArray(), func(v *dag.StreamVar, idx int) *exec.StreamVar { return v.Var }),
  144. Outputs: lo.Map(t.OutputStreams().RawArray(), func(v *dag.StreamVar, idx int) *exec.StreamVar { return v.Var }),
  145. ChunkSize: t.LRC.ChunkSize,
  146. }, nil
  147. }
  148. // func (t *LRCConstructAnyType) String() string {
  149. // return fmt.Sprintf("LRCAny[]%v%v", formatStreamIO(node), formatValueIO(node))
  150. // }
  151. type LRCConstructGroupNode struct {
  152. dag.NodeBase
  153. LRC cdssdk.LRCRedundancy
  154. TargetBlockIndex int
  155. }
  156. func (b *GraphNodeBuilder) NewLRCConstructGroup(lrc cdssdk.LRCRedundancy) *LRCConstructGroupNode {
  157. node := &LRCConstructGroupNode{
  158. LRC: lrc,
  159. }
  160. b.AddNode(node)
  161. return node
  162. }
  163. func (t *LRCConstructGroupNode) SetupForTarget(blockIdx int, inputs []*dag.StreamVar) *dag.StreamVar {
  164. t.TargetBlockIndex = blockIdx
  165. t.InputStreams().Resize(0)
  166. for _, in := range inputs {
  167. idx := t.InputStreams().EnlargeOne()
  168. in.Connect(t, idx)
  169. }
  170. output := t.Graph().NewStreamVar()
  171. t.OutputStreams().Setup(t, output, 0)
  172. return output
  173. }
  174. func (t *LRCConstructGroupNode) GenerateOp() (exec.Op, error) {
  175. l, err := lrc.New(t.LRC.N, t.LRC.K, t.LRC.Groups)
  176. if err != nil {
  177. return nil, err
  178. }
  179. coef, err := l.GenerateGroupMatrix(t.TargetBlockIndex)
  180. if err != nil {
  181. return nil, err
  182. }
  183. return &GalMultiply{
  184. Coef: coef,
  185. Inputs: lo.Map(t.InputStreams().RawArray(), func(v *dag.StreamVar, idx int) *exec.StreamVar { return v.Var }),
  186. Outputs: lo.Map(t.OutputStreams().RawArray(), func(v *dag.StreamVar, idx int) *exec.StreamVar { return v.Var }),
  187. ChunkSize: t.LRC.ChunkSize,
  188. }, nil
  189. }
  190. // func (t *LRCConstructGroupType) String() string {
  191. // return fmt.Sprintf("LRCGroup[]%v%v", formatStreamIO(node), formatValueIO(node))
  192. // }

本项目旨在将云际存储公共基础设施化,使个人及企业可低门槛使用高效的云际存储服务(安装开箱即用云际存储客户端即可,无需关注其他组件的部署),同时支持用户灵活便捷定制云际存储的功能细节。