SnapshotVmJob.php 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128
  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. {
  11. use ProductService;
  12. /**
  13. * @var SnapshotJob
  14. */
  15. private $snapshotJob;
  16. public function handle($text = null)
  17. {
  18. openlog("ProxmoxSnapshot", LOG_PID,LOG_SYSLOG);
  19. $this->initParams();
  20. $this->setHostingId($this->getWhmcsParamByKey("serviceid"));
  21. if ($this->isDone())
  22. {
  23. return true;
  24. }
  25. elseif ($this->isTaskRunning())
  26. {
  27. //sleep
  28. $this->sleep(25);
  29. return false;
  30. }
  31. $this->resourceGuard()->snapshotLimit();
  32. syslog(LOG_NOTICE,"Snapshot Create for " . $this->model->id );
  33. $snapshot = new Snapshot();
  34. $snapshot->setApi($this->api());
  35. $snapshot->setPath($this->vm()->getPath() . "/snapshot");
  36. $snapshot->setAttributes([
  37. "name" => $this->getSnapshotJob()->name . "_" . date("m") . "_" . date("d") . "_" . date("H") . "_" . date("i"),
  38. "description" => $this->snapshotJob->description,
  39. ]);
  40. syslog(LOG_NOTICE," Creating Snapshot for " . $this->getSnapshotJob()->name . "_" . $this->model->id . "_" . date("m") . "_" . date("d") . "_" . date("H"));
  41. if (!is_null($this->snapshotJob->vmstate ) && $this->vm() instanceof Kvm)
  42. {
  43. $snapshot->setVmstate($this->snapshotJob->vmstate );
  44. }
  45. try {
  46. $taskId = $snapshot->create();
  47. } catch(\Exception $ex) {
  48. $message = $ex->getMessage();
  49. syslog(LOG_NOTICE,"!!Snapshot throw Exception: " . $message );
  50. if (strpos($message,"snapshot") === false && strpos($message,"already") === false && strpos($message,"used") === false ) {
  51. throw $ex;
  52. }
  53. }
  54. $this->sleep(25);
  55. $this->clean();
  56. $this->sleep(5);
  57. //save task id
  58. $this->putModelDataAndSave(["taskId" => $taskId, "node" => $this->vm()->getNode()]);
  59. //sleep
  60. syslog(LOG_NOTICE,"Snapshot Processing DONE" );
  61. return false;
  62. }
  63. /**
  64. * @return SnapshotJob
  65. */
  66. private function getSnapshotJob(){
  67. return $this->snapshotJob = SnapshotJob::findOrFail($this->getModelData()['snapshotJobId']);
  68. }
  69. private function clean(){
  70. syslog(LOG_NOTICE,"Snapshot Clean for " . $this->model->id );
  71. $limit = $this->getWhmcsConfigOption(ConfigurableOption::SNAPSHOTS, $this->configuration()->getSnapshotMaxFiles());
  72. syslog(LOG_NOTICE," Checking Snapshot Limit (" . $limit . ") for " . $this->model->id );
  73. if($limit=="-1"){
  74. return;
  75. }
  76. if(!$limit || !is_numeric($limit)){
  77. throw new \InvalidArgumentException("Snapshot limit cannot be empty");
  78. }
  79. $snapshotRepository = new SnapshotRepository();
  80. $snapshotRepository->setApi($this->api());
  81. $snapshotRepository->findByVm($this->vm());
  82. $snapshotRepository->ignoreCurrent(true);
  83. if($snapshotRepository->count() < $limit){
  84. return;
  85. }
  86. $toDelete = (int) $snapshotRepository->count() - $limit;
  87. syslog(LOG_NOTICE," Snapshots: Existing: " . $snapshotRepository->count() . " Limit: " . $limit );
  88. syslog(LOG_NOTICE," Number of Snapshots to delete: " . $toDelete );
  89. foreach ($snapshotRepository->sortBySnaptime()->fetch() as $entity) {
  90. syslog(LOG_NOTICE," Checking Snapshot: " . $entity->getName() . " (" . $entity->getPath() . ")" );
  91. if($toDelete <=0){
  92. break;
  93. }
  94. $errorCount = 0;
  95. $errorMessage = "NOT EMPTY";
  96. while(!empty($errorMessage) && $errorCount < 18) {
  97. syslog(LOG_NOTICE," ->Delete Snapshot: " . $entity->getName() );
  98. try {
  99. $result = $entity->delete();
  100. //syslog(LOG_NOTICE," ->RESULT " . print_r($result,true) );
  101. $errorMessage = "";
  102. syslog(LOG_NOTICE," ->SUCCESS " . $result );
  103. sleep(45);
  104. } catch(\Exception $ex) {
  105. $errorMessage = $ex->getMessage();
  106. syslog(LOG_NOTICE," ->ERROR " . $errorCount . ": " . $errorMessage );
  107. $errorCount ++;
  108. sleep(45);
  109. }
  110. }
  111. $toDelete --;
  112. }
  113. syslog(LOG_NOTICE,"Snapshot Clean DONE" );
  114. }
  115. }