SnapshotVmJob.php 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105
  1. <?php
  2. namespace ModulesGarden\ProxmoxAddon\App\Jobs\Vps;
  3. use MGProvision\Proxmox\v2\models\Kvm;
  4. use MGProvision\Proxmox\v2\models\Snapshot;
  5. use MGProvision\Proxmox\v2\repository\SnapshotRepository;
  6. use ModulesGarden\ProxmoxAddon\App\Models\SnapshotJob;
  7. use ModulesGarden\ProxmoxAddon\App\Services\Vps\ProductService;
  8. use ModulesGarden\ProxmoxAddon\App\Enum\Vps\ConfigurableOption;
  9. class SnapshotVmJob extends BaseJob {
  10. use ProductService;
  11. /**
  12. * @var SnapshotJob
  13. */
  14. private $snapshotJob;
  15. public function handle($text = null) {
  16. openlog("ProxmoxSnapshot", LOG_PID,LOG_SYSLOG);
  17. syslog(LOG_NOTICE,"Snapshot Create for " . $this->model->id );
  18. $this->initParams();
  19. $this->setHostingId($this->getWhmcsParamByKey("serviceid"));
  20. if ($this->isDone()) {
  21. return true;
  22. } elseif ($this->isTaskRunning()) {
  23. //sleep
  24. $this->sleep(5);
  25. return false;
  26. }
  27. $snapshot = new Snapshot();
  28. $snapshot->setApi($this->api());
  29. $snapshot->setPath($this->vm()->getPath() . "/snapshot");
  30. $snapshot->setAttributes([
  31. // Changed by Roland at 2021-08-28 because duplicate snappshot names which causes the error: TASK ERROR: snapshot name 'Secure_280' already used
  32. // Added ' "_" . date("m") . "_" . date("d") . "_" . date("H"),' to make the snapshot-name more unique
  33. "name" => $this->getSnapshotJob()->name . "_" . $this->model->id . "_" . date("m") . "_" . date("d") . "_" . date("H"),
  34. "description" => $this->snapshotJob->description,
  35. ]);
  36. syslog(LOG_NOTICE," Creating Snapshot for " . $this->getSnapshotJob()->name . "_" . $this->model->id . "_" . date("m") . "_" . date("d") . "_" . date("H"));
  37. if (!is_null($this->snapshotJob->vmstate ) && $this->vm() instanceof Kvm) {
  38. $snapshot->setVmstate($this->snapshotJob->vmstate );
  39. }
  40. try {
  41. $taskId = $snapshot->create();
  42. } catch(\Exception $ex) {
  43. $message = $ex->getMessage();
  44. if (strpos($message,"snapshot") === false && strpos($message,"already") === false && strpos($message,"used") === false ) {
  45. throw $ex;
  46. }
  47. }
  48. //save task id
  49. $this->putModelDataAndSave(["taskId" => $taskId, "node" => $this->vm()->getNode()]);
  50. //sleep
  51. $this->sleep(25);
  52. $this->clean();
  53. $this->sleep(5);
  54. $stackTrace = implode("\n", debug_backtrace());
  55. syslog(LOG_NOTICE,"Stack Trace " . $stackTrace );
  56. return false;
  57. }
  58. /**
  59. * @return SnapshotJob
  60. */
  61. private function getSnapshotJob(){
  62. return $this->snapshotJob = SnapshotJob::findOrFail($this->getModelData()['snapshotJobId']);
  63. }
  64. private function clean() {
  65. syslog(LOG_NOTICE,"Snapshot Clean for " . $this->model->id );
  66. $limit = $this->getWhmcsConfigOption(ConfigurableOption::SNAPSHOTS, $this->configuration()->getSnapshotMaxFiles());
  67. syslog(LOG_NOTICE," Checking Snapshot Limit (" . $limit . ") for " . $this->model->id );
  68. if($limit=="-1"){
  69. return;
  70. }
  71. if(!$limit || !is_numeric($limit)){
  72. throw new \InvalidArgumentException("Snapshot limit cannot be empty");
  73. }
  74. $snapshotRepository = new SnapshotRepository();
  75. $snapshotRepository->setApi($this->api());
  76. $snapshotRepository->findByVm($this->vm());
  77. $snapshotRepository->ignoreCurrent(true);
  78. if($snapshotRepository->count() < $limit){
  79. return;
  80. }
  81. $toDelete = (int) $snapshotRepository->count() - $limit;
  82. syslog(LOG_NOTICE," Snapshots: Existing: " . $snapshotRepository->count() . " Limit: " . $limit );
  83. syslog(LOG_NOTICE," Number of Snapshots to delete: " . $toDelete );
  84. foreach ($snapshotRepository->sortBySnaptime()->fetch() as $entity) {
  85. syslog(LOG_NOTICE," Checking Snapshot: " . $entity->getName() . " (" . $entity->getPath() . ")" );
  86. if($toDelete <=0){
  87. break;
  88. }
  89. syslog(LOG_NOTICE," ->Delete Snapshot: " . $entity->getName() . " (" . $entity->getPath() . ")" );
  90. $entity->delete();
  91. $toDelete --;
  92. }
  93. }
  94. }