solve.go 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564
  1. package client
  2. import (
  3. "context"
  4. "encoding/base64"
  5. "encoding/json"
  6. "io"
  7. "os"
  8. "path/filepath"
  9. "strings"
  10. "time"
  11. "github.com/containerd/containerd/content"
  12. contentlocal "github.com/containerd/containerd/content/local"
  13. controlapi "github.com/moby/buildkit/api/services/control"
  14. "github.com/moby/buildkit/client/llb"
  15. "github.com/moby/buildkit/client/ociindex"
  16. "github.com/moby/buildkit/exporter/containerimage/exptypes"
  17. "github.com/moby/buildkit/identity"
  18. "github.com/moby/buildkit/session"
  19. sessioncontent "github.com/moby/buildkit/session/content"
  20. "github.com/moby/buildkit/session/filesync"
  21. "github.com/moby/buildkit/session/grpchijack"
  22. "github.com/moby/buildkit/solver/pb"
  23. spb "github.com/moby/buildkit/sourcepolicy/pb"
  24. "github.com/moby/buildkit/util/bklog"
  25. "github.com/moby/buildkit/util/entitlements"
  26. ocispecs "github.com/opencontainers/image-spec/specs-go/v1"
  27. "github.com/pkg/errors"
  28. "github.com/tonistiigi/fsutil"
  29. fstypes "github.com/tonistiigi/fsutil/types"
  30. "go.opentelemetry.io/otel/trace"
  31. "golang.org/x/sync/errgroup"
  32. )
  33. type SolveOpt struct {
  34. Exports []ExportEntry
  35. LocalDirs map[string]string // Deprecated: use LocalMounts
  36. LocalMounts map[string]fsutil.FS
  37. OCIStores map[string]content.Store
  38. SharedKey string
  39. Frontend string
  40. FrontendAttrs map[string]string
  41. FrontendInputs map[string]llb.State
  42. CacheExports []CacheOptionsEntry
  43. CacheImports []CacheOptionsEntry
  44. Session []session.Attachable
  45. AllowedEntitlements []entitlements.Entitlement
  46. SharedSession *session.Session // TODO: refactor to better session syncing
  47. SessionPreInitialized bool // TODO: refactor to better session syncing
  48. Internal bool
  49. SourcePolicy *spb.Policy
  50. Ref string
  51. }
  52. type ExportEntry struct {
  53. Type string
  54. Attrs map[string]string
  55. Output filesync.FileOutputFunc // for ExporterOCI and ExporterDocker
  56. OutputDir string // for ExporterLocal
  57. }
  58. type CacheOptionsEntry struct {
  59. Type string
  60. Attrs map[string]string
  61. }
  62. // Solve calls Solve on the controller.
  63. // def must be nil if (and only if) opt.Frontend is set.
  64. func (c *Client) Solve(ctx context.Context, def *llb.Definition, opt SolveOpt, statusChan chan *SolveStatus) (*SolveResponse, error) {
  65. defer func() {
  66. if statusChan != nil {
  67. close(statusChan)
  68. }
  69. }()
  70. if opt.Frontend == "" && def == nil {
  71. return nil, errors.New("invalid empty definition")
  72. }
  73. if opt.Frontend != "" && def != nil {
  74. return nil, errors.Errorf("invalid definition for frontend %s", opt.Frontend)
  75. }
  76. return c.solve(ctx, def, nil, opt, statusChan)
  77. }
  78. type runGatewayCB func(ref string, s *session.Session, opts map[string]string) error
  79. func (c *Client) solve(ctx context.Context, def *llb.Definition, runGateway runGatewayCB, opt SolveOpt, statusChan chan *SolveStatus) (*SolveResponse, error) {
  80. if def != nil && runGateway != nil {
  81. return nil, errors.New("invalid with def and cb")
  82. }
  83. mounts, err := prepareMounts(&opt)
  84. if err != nil {
  85. return nil, err
  86. }
  87. syncedDirs, err := prepareSyncedFiles(def, mounts)
  88. if err != nil {
  89. return nil, err
  90. }
  91. ref := identity.NewID()
  92. if opt.Ref != "" {
  93. ref = opt.Ref
  94. }
  95. eg, ctx := errgroup.WithContext(ctx)
  96. statusContext, cancelStatus := context.WithCancelCause(context.Background())
  97. defer cancelStatus(errors.WithStack(context.Canceled))
  98. if span := trace.SpanFromContext(ctx); span.SpanContext().IsValid() {
  99. statusContext = trace.ContextWithSpan(statusContext, span)
  100. }
  101. s := opt.SharedSession
  102. if s == nil {
  103. if opt.SessionPreInitialized {
  104. return nil, errors.Errorf("no session provided for preinitialized option")
  105. }
  106. s, err = session.NewSession(statusContext, defaultSessionName(), opt.SharedKey)
  107. if err != nil {
  108. return nil, errors.Wrap(err, "failed to create session")
  109. }
  110. }
  111. cacheOpt, err := parseCacheOptions(ctx, runGateway != nil, opt)
  112. if err != nil {
  113. return nil, err
  114. }
  115. storesToUpdate := []string{}
  116. if !opt.SessionPreInitialized {
  117. if len(syncedDirs) > 0 {
  118. s.Allow(filesync.NewFSSyncProvider(syncedDirs))
  119. }
  120. for _, a := range opt.Session {
  121. s.Allow(a)
  122. }
  123. contentStores := map[string]content.Store{}
  124. for key, store := range cacheOpt.contentStores {
  125. contentStores[key] = store
  126. }
  127. for key, store := range opt.OCIStores {
  128. key2 := "oci:" + key
  129. if _, ok := contentStores[key2]; ok {
  130. return nil, errors.Errorf("oci store key %q already exists", key)
  131. }
  132. contentStores[key2] = store
  133. }
  134. var syncTargets []filesync.FSSyncTarget
  135. for exID, ex := range opt.Exports {
  136. var supportFile bool
  137. var supportDir bool
  138. switch ex.Type {
  139. case ExporterLocal:
  140. supportDir = true
  141. case ExporterTar:
  142. supportFile = true
  143. case ExporterOCI, ExporterDocker:
  144. supportDir = ex.OutputDir != ""
  145. supportFile = ex.Output != nil
  146. }
  147. if supportFile && supportDir {
  148. return nil, errors.Errorf("both file and directory output is not supported by %s exporter", ex.Type)
  149. }
  150. if !supportFile && ex.Output != nil {
  151. return nil, errors.Errorf("output file writer is not supported by %s exporter", ex.Type)
  152. }
  153. if !supportDir && ex.OutputDir != "" {
  154. return nil, errors.Errorf("output directory is not supported by %s exporter", ex.Type)
  155. }
  156. if supportFile {
  157. if ex.Output == nil {
  158. return nil, errors.Errorf("output file writer is required for %s exporter", ex.Type)
  159. }
  160. syncTargets = append(syncTargets, filesync.WithFSSync(exID, ex.Output))
  161. }
  162. if supportDir {
  163. if ex.OutputDir == "" {
  164. return nil, errors.Errorf("output directory is required for %s exporter", ex.Type)
  165. }
  166. switch ex.Type {
  167. case ExporterOCI, ExporterDocker:
  168. if err := os.MkdirAll(ex.OutputDir, 0755); err != nil {
  169. return nil, err
  170. }
  171. cs, err := contentlocal.NewStore(ex.OutputDir)
  172. if err != nil {
  173. return nil, err
  174. }
  175. contentStores["export"] = cs
  176. storesToUpdate = append(storesToUpdate, ex.OutputDir)
  177. default:
  178. syncTargets = append(syncTargets, filesync.WithFSSyncDir(exID, ex.OutputDir))
  179. }
  180. }
  181. }
  182. if len(contentStores) > 0 {
  183. s.Allow(sessioncontent.NewAttachable(contentStores))
  184. }
  185. if len(syncTargets) > 0 {
  186. s.Allow(filesync.NewFSSyncTarget(syncTargets...))
  187. }
  188. eg.Go(func() error {
  189. sd := c.sessionDialer
  190. if sd == nil {
  191. sd = grpchijack.Dialer(c.ControlClient())
  192. }
  193. return s.Run(statusContext, sd)
  194. })
  195. }
  196. frontendAttrs := map[string]string{}
  197. for k, v := range opt.FrontendAttrs {
  198. frontendAttrs[k] = v
  199. }
  200. for k, v := range cacheOpt.frontendAttrs {
  201. frontendAttrs[k] = v
  202. }
  203. solveCtx, cancelSolve := context.WithCancelCause(ctx)
  204. var res *SolveResponse
  205. eg.Go(func() error {
  206. ctx := solveCtx
  207. defer cancelSolve(errors.WithStack(context.Canceled))
  208. defer func() { // make sure the Status ends cleanly on build errors
  209. go func() {
  210. <-time.After(3 * time.Second)
  211. cancelStatus(errors.WithStack(context.Canceled))
  212. }()
  213. if !opt.SessionPreInitialized {
  214. bklog.G(ctx).Debugf("stopping session")
  215. s.Close()
  216. }
  217. }()
  218. var pbd *pb.Definition
  219. if def != nil {
  220. pbd = def.ToPB()
  221. }
  222. frontendInputs := make(map[string]*pb.Definition)
  223. for key, st := range opt.FrontendInputs {
  224. def, err := st.Marshal(ctx)
  225. if err != nil {
  226. return err
  227. }
  228. frontendInputs[key] = def.ToPB()
  229. }
  230. exports := make([]*controlapi.Exporter, 0, len(opt.Exports))
  231. exportDeprecated := ""
  232. exportAttrDeprecated := map[string]string{}
  233. for i, exp := range opt.Exports {
  234. if i == 0 {
  235. exportDeprecated = exp.Type
  236. exportAttrDeprecated = exp.Attrs
  237. }
  238. exports = append(exports, &controlapi.Exporter{
  239. Type: exp.Type,
  240. Attrs: exp.Attrs,
  241. })
  242. }
  243. resp, err := c.ControlClient().Solve(ctx, &controlapi.SolveRequest{
  244. Ref: ref,
  245. Definition: pbd,
  246. Exporters: exports,
  247. ExporterDeprecated: exportDeprecated,
  248. ExporterAttrsDeprecated: exportAttrDeprecated,
  249. Session: s.ID(),
  250. Frontend: opt.Frontend,
  251. FrontendAttrs: frontendAttrs,
  252. FrontendInputs: frontendInputs,
  253. Cache: cacheOpt.options,
  254. Entitlements: opt.AllowedEntitlements,
  255. Internal: opt.Internal,
  256. SourcePolicy: opt.SourcePolicy,
  257. })
  258. if err != nil {
  259. return errors.Wrap(err, "failed to solve")
  260. }
  261. res = &SolveResponse{
  262. ExporterResponse: resp.ExporterResponse,
  263. }
  264. return nil
  265. })
  266. if runGateway != nil {
  267. eg.Go(func() error {
  268. err := runGateway(ref, s, frontendAttrs)
  269. if err == nil {
  270. return nil
  271. }
  272. // If the callback failed then the main
  273. // `Solve` (called above) should error as
  274. // well. However as a fallback we wait up to
  275. // 5s for that to happen before failing this
  276. // goroutine.
  277. select {
  278. case <-solveCtx.Done():
  279. case <-time.After(5 * time.Second):
  280. cancelSolve(errors.WithStack(context.Canceled))
  281. }
  282. return err
  283. })
  284. }
  285. eg.Go(func() error {
  286. stream, err := c.ControlClient().Status(statusContext, &controlapi.StatusRequest{
  287. Ref: ref,
  288. })
  289. if err != nil {
  290. return errors.Wrap(err, "failed to get status")
  291. }
  292. for {
  293. resp, err := stream.Recv()
  294. if err != nil {
  295. if err == io.EOF {
  296. return nil
  297. }
  298. return errors.Wrap(err, "failed to receive status")
  299. }
  300. if statusChan != nil {
  301. statusChan <- NewSolveStatus(resp)
  302. }
  303. }
  304. })
  305. if err := eg.Wait(); err != nil {
  306. return nil, err
  307. }
  308. // Update index.json of exported cache content store
  309. // FIXME(AkihiroSuda): dedupe const definition of cache/remotecache.ExporterResponseManifestDesc = "cache.manifest"
  310. if manifestDescJSON := res.ExporterResponse["cache.manifest"]; manifestDescJSON != "" {
  311. var manifestDesc ocispecs.Descriptor
  312. if err = json.Unmarshal([]byte(manifestDescJSON), &manifestDesc); err != nil {
  313. return nil, err
  314. }
  315. for storePath, tag := range cacheOpt.storesToUpdate {
  316. idx := ociindex.NewStoreIndex(storePath)
  317. if err := idx.Put(tag, manifestDesc); err != nil {
  318. return nil, err
  319. }
  320. }
  321. }
  322. if manifestDescDt := res.ExporterResponse[exptypes.ExporterImageDescriptorKey]; manifestDescDt != "" {
  323. manifestDescDt, err := base64.StdEncoding.DecodeString(manifestDescDt)
  324. if err != nil {
  325. return nil, err
  326. }
  327. var manifestDesc ocispecs.Descriptor
  328. if err = json.Unmarshal([]byte(manifestDescDt), &manifestDesc); err != nil {
  329. return nil, err
  330. }
  331. for _, storePath := range storesToUpdate {
  332. tag := "latest"
  333. if t, ok := res.ExporterResponse["image.name"]; ok {
  334. tag = t
  335. }
  336. idx := ociindex.NewStoreIndex(storePath)
  337. if err := idx.Put(tag, manifestDesc); err != nil {
  338. return nil, err
  339. }
  340. }
  341. }
  342. return res, nil
  343. }
  344. func prepareSyncedFiles(def *llb.Definition, localMounts map[string]fsutil.FS) (filesync.StaticDirSource, error) {
  345. resetUIDAndGID := func(p string, st *fstypes.Stat) fsutil.MapResult {
  346. st.Uid = 0
  347. st.Gid = 0
  348. return fsutil.MapResultKeep
  349. }
  350. result := make(filesync.StaticDirSource, len(localMounts))
  351. if def == nil {
  352. for name, mount := range localMounts {
  353. mount, err := fsutil.NewFilterFS(mount, &fsutil.FilterOpt{
  354. Map: resetUIDAndGID,
  355. })
  356. if err != nil {
  357. return nil, err
  358. }
  359. result[name] = mount
  360. }
  361. } else {
  362. for _, dt := range def.Def {
  363. var op pb.Op
  364. if err := (&op).Unmarshal(dt); err != nil {
  365. return nil, errors.Wrap(err, "failed to parse llb proto op")
  366. }
  367. if src := op.GetSource(); src != nil {
  368. if strings.HasPrefix(src.Identifier, "local://") {
  369. name := strings.TrimPrefix(src.Identifier, "local://")
  370. mount, ok := localMounts[name]
  371. if !ok {
  372. return nil, errors.Errorf("local directory %s not enabled", name)
  373. }
  374. mount, err := fsutil.NewFilterFS(mount, &fsutil.FilterOpt{
  375. Map: resetUIDAndGID,
  376. })
  377. if err != nil {
  378. return nil, err
  379. }
  380. result[name] = mount
  381. }
  382. }
  383. }
  384. }
  385. return result, nil
  386. }
  387. func defaultSessionName() string {
  388. wd, err := os.Getwd()
  389. if err != nil {
  390. return "unknown"
  391. }
  392. return filepath.Base(wd)
  393. }
  394. type cacheOptions struct {
  395. options controlapi.CacheOptions
  396. contentStores map[string]content.Store // key: ID of content store ("local:" + csDir)
  397. storesToUpdate map[string]string // key: path to content store, value: tag
  398. frontendAttrs map[string]string
  399. }
  400. func parseCacheOptions(ctx context.Context, isGateway bool, opt SolveOpt) (*cacheOptions, error) {
  401. var (
  402. cacheExports []*controlapi.CacheOptionsEntry
  403. cacheImports []*controlapi.CacheOptionsEntry
  404. )
  405. contentStores := make(map[string]content.Store)
  406. storesToUpdate := make(map[string]string)
  407. frontendAttrs := make(map[string]string)
  408. for _, ex := range opt.CacheExports {
  409. if ex.Type == "local" {
  410. csDir := ex.Attrs["dest"]
  411. if csDir == "" {
  412. return nil, errors.New("local cache exporter requires dest")
  413. }
  414. if err := os.MkdirAll(csDir, 0755); err != nil {
  415. return nil, err
  416. }
  417. cs, err := contentlocal.NewStore(csDir)
  418. if err != nil {
  419. return nil, err
  420. }
  421. contentStores["local:"+csDir] = cs
  422. tag := "latest"
  423. if t, ok := ex.Attrs["tag"]; ok {
  424. tag = t
  425. }
  426. // TODO(AkihiroSuda): support custom index JSON path and tag
  427. storesToUpdate[csDir] = tag
  428. }
  429. if ex.Type == "registry" {
  430. regRef := ex.Attrs["ref"]
  431. if regRef == "" {
  432. return nil, errors.New("registry cache exporter requires ref")
  433. }
  434. }
  435. cacheExports = append(cacheExports, &controlapi.CacheOptionsEntry{
  436. Type: ex.Type,
  437. Attrs: ex.Attrs,
  438. })
  439. }
  440. for _, im := range opt.CacheImports {
  441. if im.Type == "local" {
  442. csDir := im.Attrs["src"]
  443. if csDir == "" {
  444. return nil, errors.New("local cache importer requires src")
  445. }
  446. cs, err := contentlocal.NewStore(csDir)
  447. if err != nil {
  448. bklog.G(ctx).Warning("local cache import at " + csDir + " not found due to err: " + err.Error())
  449. continue
  450. }
  451. // if digest is not specified, attempt to load from tag
  452. if im.Attrs["digest"] == "" {
  453. tag := "latest"
  454. if t, ok := im.Attrs["tag"]; ok {
  455. tag = t
  456. }
  457. idx := ociindex.NewStoreIndex(csDir)
  458. desc, err := idx.Get(tag)
  459. if err != nil {
  460. bklog.G(ctx).Warning("local cache import at " + csDir + " not found due to err: " + err.Error())
  461. continue
  462. }
  463. if desc != nil {
  464. im.Attrs["digest"] = desc.Digest.String()
  465. }
  466. }
  467. if im.Attrs["digest"] == "" {
  468. return nil, errors.New("local cache importer requires either explicit digest, \"latest\" tag or custom tag on index.json")
  469. }
  470. contentStores["local:"+csDir] = cs
  471. }
  472. if im.Type == "registry" {
  473. regRef := im.Attrs["ref"]
  474. if regRef == "" {
  475. return nil, errors.New("registry cache importer requires ref")
  476. }
  477. }
  478. cacheImports = append(cacheImports, &controlapi.CacheOptionsEntry{
  479. Type: im.Type,
  480. Attrs: im.Attrs,
  481. })
  482. }
  483. if opt.Frontend != "" || isGateway {
  484. if len(cacheImports) > 0 {
  485. s, err := json.Marshal(cacheImports)
  486. if err != nil {
  487. return nil, err
  488. }
  489. frontendAttrs["cache-imports"] = string(s)
  490. }
  491. }
  492. res := cacheOptions{
  493. options: controlapi.CacheOptions{
  494. Exports: cacheExports,
  495. Imports: cacheImports,
  496. },
  497. contentStores: contentStores,
  498. storesToUpdate: storesToUpdate,
  499. frontendAttrs: frontendAttrs,
  500. }
  501. return &res, nil
  502. }
  503. func prepareMounts(opt *SolveOpt) (map[string]fsutil.FS, error) {
  504. // merge local mounts and fallback local directories together
  505. mounts := make(map[string]fsutil.FS)
  506. for k, mount := range opt.LocalMounts {
  507. mounts[k] = mount
  508. }
  509. for k, dir := range opt.LocalDirs {
  510. mount, err := fsutil.NewFS(dir)
  511. if err != nil {
  512. return nil, err
  513. }
  514. if _, ok := mounts[k]; ok {
  515. return nil, errors.Errorf("local mount %s already exists", k)
  516. }
  517. mounts[k] = mount
  518. }
  519. return mounts, nil
  520. }