mirror of https://github.com/micromata/borgbackup-butler.git

Kai Reinhard
24.52.2019 c50335dff01dffab42f4f25639eb14c11d80a70c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
package de.micromata.borgbutler;
 
import de.micromata.borgbutler.config.BorgRepoConfig;
import de.micromata.borgbutler.jobs.AbstractJob;
import de.micromata.borgbutler.jobs.JobQueue;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
 
import java.util.*;
 
/**
 * A queue is important because Borg doesn't support parallel calls for one repository.
 * For each repository one single queue is allocated.
 */
public class BorgQueueExecutor {
    private Logger log = LoggerFactory.getLogger(BorgQueueExecutor.class);
    private static final BorgQueueExecutor instance = new BorgQueueExecutor();
    private static final String NONE_REPO_QUEUE = "--NO_REPO--";
 
    public static BorgQueueExecutor getInstance() {
        return instance;
    }
 
    // key is the repo name.
    private Map<String, JobQueue<String>> queueMap = new HashMap<>();
 
    public BorgQueueStatistics getStatistics() {
        BorgQueueStatistics statistics = new BorgQueueStatistics();
        Iterator<JobQueue<String>> it = queueMap.values().iterator();
        while (it.hasNext()) {
            JobQueue<?> queue = it.next();
            statistics.totalNumberOfQueues++;
            int queueSize = queue.getQueueSize();
            if (queueSize > 0) {
                statistics.numberOfActiveQueues++;
                statistics.numberOfRunningAndQueuedJobs += queueSize;
            }
            int oldJobsSize = queue.getOldJobsSize();
            if (oldJobsSize > 0) {
                statistics.numberOfOldJobs += oldJobsSize;
            }
        }
        return statistics;
    }
 
    /**
     * @return A list of all repos with queues.
     */
    public List<String> getRepos() {
        List<String> list = new ArrayList<>();
        synchronized (queueMap) {
            list.addAll(queueMap.keySet());
        }
        Collections.sort(list);
        return list;
    }
 
    public void cancelJob(long uniqueJobNumber) {
        AbstractJob<?> job = null;
        JobQueue<?> queue = null;
        Iterator<JobQueue<String>> it = queueMap.values().iterator();
        while (it.hasNext()) {
            queue = it.next();
            job = queue.getQueuedJobByUniqueJobNumber(uniqueJobNumber);
            if (job != null) {
                break;
            }
        }
        if (job == null) {
            log.info("Can't cancel job #" + uniqueJobNumber + ". Not found as queued job (may-be already cancelled or finished). Nothing to do.");
            return;
        }
        job.cancel();
        queue.refreshQueue();
    }
 
    /**
     * For displaying purposes.
     *
     * @param repoConfig
     * @param oldJobs If false, the running and queued jobs are returned, otherwise the done ones.
     * @return A list of all jobs of the queue (as copies).
     */
    public List<BorgJob<?>> getJobListCopy(BorgRepoConfig repoConfig, boolean oldJobs) {
        JobQueue<String> origQueue = getQueue(repoConfig);
        List<BorgJob<?>> jobList = new ArrayList<>();
        if (origQueue == null) {
            return jobList;
        }
        synchronized (origQueue) {
            Iterator<AbstractJob<String>> it = oldJobs ? origQueue.getOldJobsIterator() : origQueue.getQueueIterator();
            while (it.hasNext()) {
                AbstractJob<String> origJob = it.next();
                if (!(origJob instanceof BorgJob)) {
                    log.error("Oups, only BorgJobs are supported. Ignoring unexpected job: " + origJob.getClass());
                    continue;
                }
                BorgJob<?> borgJob = ((BorgJob<?>) origJob).clone();
                jobList.add(borgJob);
            }
        }
        return jobList;
    }
 
    private JobQueue<String> getQueue(BorgRepoConfig repoConfig) {
        synchronized (queueMap) {
            return queueMap.get(getQueueName(repoConfig));
        }
    }
 
    private JobQueue<String> ensureAndGetQueue(BorgRepoConfig repoConfig) {
        synchronized (queueMap) {
            String queueName = getQueueName(repoConfig);
            JobQueue<String> queue = getQueue(repoConfig);
            if (queue == null) {
                queue = new JobQueue<>();
                queueMap.put(queueName, queue);
            }
            return queue;
        }
    }
 
    private String getQueueName(BorgRepoConfig repoConfig) {
        return repoConfig != null ? repoConfig.getId() : NONE_REPO_QUEUE;
    }
 
    public BorgJob<Void> execute(BorgCommand command) {
        BorgJob<Void> job = new BorgJob<Void>(command);
        return execute(job);
    }
 
    @SuppressWarnings("unchecked")
    public <T> BorgJob<T> execute(BorgJob<T> job) {
        return (BorgJob<T>) ensureAndGetQueue(job.getCommand().getRepoConfig()).append(job);
    }
 
    private BorgQueueExecutor() {
    }
}