SnapshotVmJob.php 4.7 KB

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