]> git.immae.eu Git - github/Chocobozzz/PeerTube.git/blame - server/models/pod.js
Update bittorrent-tracker and standard to v9
[github/Chocobozzz/PeerTube.git] / server / models / pod.js
CommitLineData
9f10b292
C
1'use strict'
2
9e167724 3const each = require('async/each')
00057e85 4const map = require('lodash/map')
9e167724 5const waterfall = require('async/waterfall')
9f10b292 6
f0f5567b 7const constants = require('../initializers/constants')
9e167724 8const logger = require('../helpers/logger')
67bf9b96 9const customPodsValidators = require('../helpers/custom-validators').pods
9f10b292
C
10
11// ---------------------------------------------------------------------------
12
feb4bdfd
C
13module.exports = function (sequelize, DataTypes) {
14 const Pod = sequelize.define('Pod',
15 {
16 host: {
67bf9b96
C
17 type: DataTypes.STRING,
18 allowNull: false,
19 validate: {
20 isHost: function (value) {
21 const res = customPodsValidators.isHostValid(value)
22 if (res === false) throw new Error('Host not valid.')
23 }
24 }
feb4bdfd
C
25 },
26 publicKey: {
67bf9b96
C
27 type: DataTypes.STRING(5000),
28 allowNull: false
feb4bdfd
C
29 },
30 score: {
31 type: DataTypes.INTEGER,
67bf9b96
C
32 defaultValue: constants.FRIEND_SCORE.BASE,
33 allowNull: false,
34 validate: {
35 isInt: true,
36 max: constants.FRIEND_SCORE.MAX
37 }
4793c343
C
38 },
39 email: {
40 type: DataTypes.STRING(400),
ad4a8a1c
C
41 allowNull: false,
42 validate: {
43 isEmail: true
44 }
feb4bdfd 45 }
feb4bdfd
C
46 },
47 {
319d072e
C
48 indexes: [
49 {
5d67f289
C
50 fields: [ 'host' ],
51 unique: true
319d072e
C
52 },
53 {
54 fields: [ 'score' ]
55 }
56 ],
feb4bdfd
C
57 classMethods: {
58 associate,
59
60 countAll,
61 incrementScores,
62 list,
63 listAllIds,
bd14d16a 64 listRandomPodIdsWithRequest,
feb4bdfd
C
65 listBadPods,
66 load,
67 loadByHost,
9e167724 68 updatePodsScore,
feb4bdfd
C
69 removeAll
70 },
71 instanceMethods: {
72 toFormatedJSON
73 }
74 }
75 )
76
77 return Pod
53572423
C
78}
79
53572423
C
80// ------------------------------ METHODS ------------------------------
81
82function toFormatedJSON () {
83 const json = {
feb4bdfd 84 id: this.id,
49abbbbe 85 host: this.host,
4793c343 86 email: this.email,
53572423 87 score: this.score,
feb4bdfd 88 createdAt: this.createdAt
53572423
C
89 }
90
91 return json
92}
93
a3ee6fa2
C
94// ------------------------------ Statics ------------------------------
95
feb4bdfd
C
96function associate (models) {
97 this.belongsToMany(models.Request, {
98 foreignKey: 'podId',
99 through: models.RequestToPod,
7920c273 100 onDelete: 'cascade'
feb4bdfd
C
101 })
102}
103
a3ee6fa2 104function countAll (callback) {
feb4bdfd 105 return this.count().asCallback(callback)
9f10b292 106}
45239549 107
9f10b292
C
108function incrementScores (ids, value, callback) {
109 if (!callback) callback = function () {}
feb4bdfd
C
110
111 const update = {
112 score: this.sequelize.literal('score +' + value)
113 }
114
67bf9b96 115 const options = {
feb4bdfd
C
116 where: {
117 id: {
118 $in: ids
119 }
67bf9b96
C
120 },
121 // In this case score is a literal and not an integer so we do not validate it
122 validate: false
feb4bdfd
C
123 }
124
67bf9b96 125 return this.update(update, options).asCallback(callback)
9f10b292 126}
45239549 127
9f10b292 128function list (callback) {
feb4bdfd 129 return this.findAll().asCallback(callback)
9f10b292 130}
8c308c2b 131
ed04d94f
C
132function listAllIds (transaction, callback) {
133 if (!callback) {
134 callback = transaction
135 transaction = null
136 }
137
feb4bdfd
C
138 const query = {
139 attributes: [ 'id' ]
140 }
141
ed04d94f
C
142 if (transaction) query.transaction = transaction
143
feb4bdfd 144 return this.findAll(query).asCallback(function (err, pods) {
00057e85
C
145 if (err) return callback(err)
146
feb4bdfd 147 return callback(null, map(pods, 'id'))
00057e85 148 })
528a9efa
C
149}
150
e4c87ec2
C
151function listRandomPodIdsWithRequest (limit, tableWithPods, tableWithPodsJoins, callback) {
152 if (!callback) {
153 callback = tableWithPodsJoins
154 tableWithPodsJoins = ''
155 }
156
bd14d16a
C
157 const self = this
158
159 self.count().asCallback(function (err, count) {
160 if (err) return callback(err)
161
162 // Optimization...
163 if (count === 0) return callback(null, [])
164
165 let start = Math.floor(Math.random() * count) - limit
166 if (start < 0) start = 0
167
168 const query = {
169 attributes: [ 'id' ],
170 order: [
171 [ 'id', 'ASC' ]
172 ],
173 offset: start,
174 limit: limit,
175 where: {
176 id: {
177 $in: [
e4c87ec2 178 this.sequelize.literal(`SELECT DISTINCT "${tableWithPods}"."podId" FROM "${tableWithPods}" ${tableWithPodsJoins}`)
bd14d16a
C
179 ]
180 }
181 }
182 }
183
184 return this.findAll(query).asCallback(function (err, pods) {
185 if (err) return callback(err)
186
187 return callback(null, map(pods, 'id'))
188 })
189 })
190}
191
a3ee6fa2 192function listBadPods (callback) {
feb4bdfd
C
193 const query = {
194 where: {
195 score: { $lte: 0 }
196 }
197 }
198
199 return this.findAll(query).asCallback(callback)
9f10b292 200}
8c308c2b 201
a3ee6fa2 202function load (id, callback) {
feb4bdfd 203 return this.findById(id).asCallback(callback)
9f10b292 204}
c45f7f84 205
49abbbbe 206function loadByHost (host, callback) {
feb4bdfd
C
207 const query = {
208 where: {
209 host: host
210 }
211 }
212
213 return this.findOne(query).asCallback(callback)
9f10b292 214}
c45f7f84 215
a3ee6fa2 216function removeAll (callback) {
feb4bdfd 217 return this.destroy().asCallback(callback)
a3ee6fa2 218}
9e167724
C
219
220function updatePodsScore (goodPods, badPods) {
221 const self = this
222
223 logger.info('Updating %d good pods and %d bad pods scores.', goodPods.length, badPods.length)
224
225 if (goodPods.length !== 0) {
226 this.incrementScores(goodPods, constants.PODS_SCORE.BONUS, function (err) {
227 if (err) logger.error('Cannot increment scores of good pods.', { error: err })
228 })
229 }
230
231 if (badPods.length !== 0) {
232 this.incrementScores(badPods, constants.PODS_SCORE.MALUS, function (err) {
233 if (err) logger.error('Cannot decrement scores of bad pods.', { error: err })
234 removeBadPods.call(self)
235 })
236 }
237}
238
239// ---------------------------------------------------------------------------
240
241// Remove pods with a score of 0 (too many requests where they were unreachable)
242function removeBadPods () {
243 const self = this
244
245 waterfall([
246 function findBadPods (callback) {
247 self.sequelize.models.Pod.listBadPods(function (err, pods) {
248 if (err) {
249 logger.error('Cannot find bad pods.', { error: err })
250 return callback(err)
251 }
252
253 return callback(null, pods)
254 })
255 },
256
257 function removeTheseBadPods (pods, callback) {
258 each(pods, function (pod, callbackEach) {
259 pod.destroy().asCallback(callbackEach)
260 }, function (err) {
261 return callback(err, pods.length)
262 })
263 }
264 ], function (err, numberOfPodsRemoved) {
265 if (err) {
266 logger.error('Cannot remove bad pods.', { error: err })
267 } else if (numberOfPodsRemoved) {
268 logger.info('Removed %d pods.', numberOfPodsRemoved)
269 } else {
270 logger.info('No need to remove bad pods.')
271 }
272 })
273}