github.com/containerd/Containerd@v1.4.13/pkg/process/init_state.go (about) 1 // +build !windows 2 3 /* 4 Copyright The containerd Authors. 5 6 Licensed under the Apache License, Version 2.0 (the "License"); 7 you may not use this file except in compliance with the License. 8 You may obtain a copy of the License at 9 10 http://www.apache.org/licenses/LICENSE-2.0 11 12 Unless required by applicable law or agreed to in writing, software 13 distributed under the License is distributed on an "AS IS" BASIS, 14 WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 15 See the License for the specific language governing permissions and 16 limitations under the License. 17 */ 18 19 package process 20 21 import ( 22 "context" 23 24 runc "github.com/containerd/go-runc" 25 google_protobuf "github.com/gogo/protobuf/types" 26 "github.com/pkg/errors" 27 "github.com/sirupsen/logrus" 28 ) 29 30 type initState interface { 31 Start(context.Context) error 32 Delete(context.Context) error 33 Pause(context.Context) error 34 Resume(context.Context) error 35 Update(context.Context, *google_protobuf.Any) error 36 Checkpoint(context.Context, *CheckpointConfig) error 37 Exec(context.Context, string, *ExecConfig) (Process, error) 38 Kill(context.Context, uint32, bool) error 39 SetExited(int) 40 Status(context.Context) (string, error) 41 } 42 43 type createdState struct { 44 p *Init 45 } 46 47 func (s *createdState) transition(name string) error { 48 switch name { 49 case "running": 50 s.p.initState = &runningState{p: s.p} 51 case "stopped": 52 s.p.initState = &stoppedState{p: s.p} 53 case "deleted": 54 s.p.initState = &deletedState{} 55 default: 56 return errors.Errorf("invalid state transition %q to %q", stateName(s), name) 57 } 58 return nil 59 } 60 61 func (s *createdState) Pause(ctx context.Context) error { 62 return errors.Errorf("cannot pause task in created state") 63 } 64 65 func (s *createdState) Resume(ctx context.Context) error { 66 return errors.Errorf("cannot resume task in created state") 67 } 68 69 func (s *createdState) Update(ctx context.Context, r *google_protobuf.Any) error { 70 return s.p.update(ctx, r) 71 } 72 73 func (s *createdState) Checkpoint(ctx context.Context, r *CheckpointConfig) error { 74 return errors.Errorf("cannot checkpoint a task in created state") 75 } 76 77 func (s *createdState) Start(ctx context.Context) error { 78 if err := s.p.start(ctx); err != nil { 79 return err 80 } 81 return s.transition("running") 82 } 83 84 func (s *createdState) Delete(ctx context.Context) error { 85 if err := s.p.delete(ctx); err != nil { 86 return err 87 } 88 return s.transition("deleted") 89 } 90 91 func (s *createdState) Kill(ctx context.Context, sig uint32, all bool) error { 92 return s.p.kill(ctx, sig, all) 93 } 94 95 func (s *createdState) SetExited(status int) { 96 s.p.setExited(status) 97 98 if err := s.transition("stopped"); err != nil { 99 panic(err) 100 } 101 } 102 103 func (s *createdState) Exec(ctx context.Context, path string, r *ExecConfig) (Process, error) { 104 return s.p.exec(ctx, path, r) 105 } 106 107 func (s *createdState) Status(ctx context.Context) (string, error) { 108 return "created", nil 109 } 110 111 type createdCheckpointState struct { 112 p *Init 113 opts *runc.RestoreOpts 114 } 115 116 func (s *createdCheckpointState) transition(name string) error { 117 switch name { 118 case "running": 119 s.p.initState = &runningState{p: s.p} 120 case "stopped": 121 s.p.initState = &stoppedState{p: s.p} 122 case "deleted": 123 s.p.initState = &deletedState{} 124 default: 125 return errors.Errorf("invalid state transition %q to %q", stateName(s), name) 126 } 127 return nil 128 } 129 130 func (s *createdCheckpointState) Pause(ctx context.Context) error { 131 return errors.Errorf("cannot pause task in created state") 132 } 133 134 func (s *createdCheckpointState) Resume(ctx context.Context) error { 135 return errors.Errorf("cannot resume task in created state") 136 } 137 138 func (s *createdCheckpointState) Update(ctx context.Context, r *google_protobuf.Any) error { 139 return s.p.update(ctx, r) 140 } 141 142 func (s *createdCheckpointState) Checkpoint(ctx context.Context, r *CheckpointConfig) error { 143 return errors.Errorf("cannot checkpoint a task in created state") 144 } 145 146 func (s *createdCheckpointState) Start(ctx context.Context) error { 147 p := s.p 148 sio := p.stdio 149 150 var ( 151 err error 152 socket *runc.Socket 153 ) 154 if sio.Terminal { 155 if socket, err = runc.NewTempConsoleSocket(); err != nil { 156 return errors.Wrap(err, "failed to create OCI runtime console socket") 157 } 158 defer socket.Close() 159 s.opts.ConsoleSocket = socket 160 } 161 162 if _, err := s.p.runtime.Restore(ctx, p.id, p.Bundle, s.opts); err != nil { 163 return p.runtimeError(err, "OCI runtime restore failed") 164 } 165 if sio.Stdin != "" { 166 if err := p.openStdin(sio.Stdin); err != nil { 167 return errors.Wrapf(err, "failed to open stdin fifo %s", sio.Stdin) 168 } 169 } 170 if socket != nil { 171 console, err := socket.ReceiveMaster() 172 if err != nil { 173 return errors.Wrap(err, "failed to retrieve console master") 174 } 175 console, err = p.Platform.CopyConsole(ctx, console, sio.Stdin, sio.Stdout, sio.Stderr, &p.wg) 176 if err != nil { 177 return errors.Wrap(err, "failed to start console copy") 178 } 179 p.console = console 180 } else { 181 if err := p.io.Copy(ctx, &p.wg); err != nil { 182 return errors.Wrap(err, "failed to start io pipe copy") 183 } 184 } 185 pid, err := runc.ReadPidFile(s.opts.PidFile) 186 if err != nil { 187 return errors.Wrap(err, "failed to retrieve OCI runtime container pid") 188 } 189 p.pid = pid 190 return s.transition("running") 191 } 192 193 func (s *createdCheckpointState) Delete(ctx context.Context) error { 194 if err := s.p.delete(ctx); err != nil { 195 return err 196 } 197 return s.transition("deleted") 198 } 199 200 func (s *createdCheckpointState) Kill(ctx context.Context, sig uint32, all bool) error { 201 return s.p.kill(ctx, sig, all) 202 } 203 204 func (s *createdCheckpointState) SetExited(status int) { 205 s.p.setExited(status) 206 207 if err := s.transition("stopped"); err != nil { 208 panic(err) 209 } 210 } 211 212 func (s *createdCheckpointState) Exec(ctx context.Context, path string, r *ExecConfig) (Process, error) { 213 return nil, errors.Errorf("cannot exec in a created state") 214 } 215 216 func (s *createdCheckpointState) Status(ctx context.Context) (string, error) { 217 return "created", nil 218 } 219 220 type runningState struct { 221 p *Init 222 } 223 224 func (s *runningState) transition(name string) error { 225 switch name { 226 case "stopped": 227 s.p.initState = &stoppedState{p: s.p} 228 case "paused": 229 s.p.initState = &pausedState{p: s.p} 230 default: 231 return errors.Errorf("invalid state transition %q to %q", stateName(s), name) 232 } 233 return nil 234 } 235 236 func (s *runningState) Pause(ctx context.Context) error { 237 s.p.pausing.set(true) 238 // NOTE "pausing" will be returned in the short window 239 // after `transition("paused")`, before `pausing` is reset 240 // to false. That doesn't break the state machine, just 241 // delays the "paused" state a little bit. 242 defer s.p.pausing.set(false) 243 244 if err := s.p.runtime.Pause(ctx, s.p.id); err != nil { 245 return s.p.runtimeError(err, "OCI runtime pause failed") 246 } 247 248 return s.transition("paused") 249 } 250 251 func (s *runningState) Resume(ctx context.Context) error { 252 return errors.Errorf("cannot resume a running process") 253 } 254 255 func (s *runningState) Update(ctx context.Context, r *google_protobuf.Any) error { 256 return s.p.update(ctx, r) 257 } 258 259 func (s *runningState) Checkpoint(ctx context.Context, r *CheckpointConfig) error { 260 return s.p.checkpoint(ctx, r) 261 } 262 263 func (s *runningState) Start(ctx context.Context) error { 264 return errors.Errorf("cannot start a running process") 265 } 266 267 func (s *runningState) Delete(ctx context.Context) error { 268 return errors.Errorf("cannot delete a running process") 269 } 270 271 func (s *runningState) Kill(ctx context.Context, sig uint32, all bool) error { 272 return s.p.kill(ctx, sig, all) 273 } 274 275 func (s *runningState) SetExited(status int) { 276 s.p.setExited(status) 277 278 if err := s.transition("stopped"); err != nil { 279 panic(err) 280 } 281 } 282 283 func (s *runningState) Exec(ctx context.Context, path string, r *ExecConfig) (Process, error) { 284 return s.p.exec(ctx, path, r) 285 } 286 287 func (s *runningState) Status(ctx context.Context) (string, error) { 288 return "running", nil 289 } 290 291 type pausedState struct { 292 p *Init 293 } 294 295 func (s *pausedState) transition(name string) error { 296 switch name { 297 case "running": 298 s.p.initState = &runningState{p: s.p} 299 case "stopped": 300 s.p.initState = &stoppedState{p: s.p} 301 default: 302 return errors.Errorf("invalid state transition %q to %q", stateName(s), name) 303 } 304 return nil 305 } 306 307 func (s *pausedState) Pause(ctx context.Context) error { 308 return errors.Errorf("cannot pause a paused container") 309 } 310 311 func (s *pausedState) Resume(ctx context.Context) error { 312 if err := s.p.runtime.Resume(ctx, s.p.id); err != nil { 313 return s.p.runtimeError(err, "OCI runtime resume failed") 314 } 315 316 return s.transition("running") 317 } 318 319 func (s *pausedState) Update(ctx context.Context, r *google_protobuf.Any) error { 320 return s.p.update(ctx, r) 321 } 322 323 func (s *pausedState) Checkpoint(ctx context.Context, r *CheckpointConfig) error { 324 return s.p.checkpoint(ctx, r) 325 } 326 327 func (s *pausedState) Start(ctx context.Context) error { 328 return errors.Errorf("cannot start a paused process") 329 } 330 331 func (s *pausedState) Delete(ctx context.Context) error { 332 return errors.Errorf("cannot delete a paused process") 333 } 334 335 func (s *pausedState) Kill(ctx context.Context, sig uint32, all bool) error { 336 return s.p.kill(ctx, sig, all) 337 } 338 339 func (s *pausedState) SetExited(status int) { 340 s.p.setExited(status) 341 342 if err := s.p.runtime.Resume(context.Background(), s.p.id); err != nil { 343 logrus.WithError(err).Error("resuming exited container from paused state") 344 } 345 346 if err := s.transition("stopped"); err != nil { 347 panic(err) 348 } 349 } 350 351 func (s *pausedState) Exec(ctx context.Context, path string, r *ExecConfig) (Process, error) { 352 return nil, errors.Errorf("cannot exec in a paused state") 353 } 354 355 func (s *pausedState) Status(ctx context.Context) (string, error) { 356 return "paused", nil 357 } 358 359 type stoppedState struct { 360 p *Init 361 } 362 363 func (s *stoppedState) transition(name string) error { 364 switch name { 365 case "deleted": 366 s.p.initState = &deletedState{} 367 default: 368 return errors.Errorf("invalid state transition %q to %q", stateName(s), name) 369 } 370 return nil 371 } 372 373 func (s *stoppedState) Pause(ctx context.Context) error { 374 return errors.Errorf("cannot pause a stopped container") 375 } 376 377 func (s *stoppedState) Resume(ctx context.Context) error { 378 return errors.Errorf("cannot resume a stopped container") 379 } 380 381 func (s *stoppedState) Update(ctx context.Context, r *google_protobuf.Any) error { 382 return errors.Errorf("cannot update a stopped container") 383 } 384 385 func (s *stoppedState) Checkpoint(ctx context.Context, r *CheckpointConfig) error { 386 return errors.Errorf("cannot checkpoint a stopped container") 387 } 388 389 func (s *stoppedState) Start(ctx context.Context) error { 390 return errors.Errorf("cannot start a stopped process") 391 } 392 393 func (s *stoppedState) Delete(ctx context.Context) error { 394 if err := s.p.delete(ctx); err != nil { 395 return err 396 } 397 return s.transition("deleted") 398 } 399 400 func (s *stoppedState) Kill(ctx context.Context, sig uint32, all bool) error { 401 return s.p.kill(ctx, sig, all) 402 } 403 404 func (s *stoppedState) SetExited(status int) { 405 // no op 406 } 407 408 func (s *stoppedState) Exec(ctx context.Context, path string, r *ExecConfig) (Process, error) { 409 return nil, errors.Errorf("cannot exec in a stopped state") 410 } 411 412 func (s *stoppedState) Status(ctx context.Context) (string, error) { 413 return "stopped", nil 414 }