]>
Commit | Line | Data |
---|---|---|
6dd9de95 C |
1 | import { IConfig } from 'config' |
2 | import { dirname, join } from 'path' | |
b764380a | 3 | import { VideosRedundancyStrategy } from '../../shared/models' |
6dd9de95 | 4 | // Do not use barrels, remain constants as independent as possible |
8f0bc73d | 5 | import { buildPath, parseBytes, parseDurationToMs, root } from '../helpers/core-utils' |
6dd9de95 C |
6 | import { NSFWPolicyType } from '../../shared/models/videos/nsfw-policy.type' |
7 | import * as bytes from 'bytes' | |
8c9e7875 | 8 | import { VideoRedundancyConfigFilter } from '@shared/models/redundancy/video-redundancy-config-filter.type' |
72c33e71 | 9 | import { BroadcastMessageLevel } from '@shared/models/server' |
6dd9de95 C |
10 | |
11 | // Use a variable to reload the configuration if we need | |
12 | let config: IConfig = require('config') | |
13 | ||
14 | const configChangedHandlers: Function[] = [] | |
15 | ||
16 | const CONFIG = { | |
17 | CUSTOM_FILE: getLocalConfigFilePath(), | |
18 | LISTEN: { | |
19 | PORT: config.get<number>('listen.port'), | |
20 | HOSTNAME: config.get<string>('listen.hostname') | |
21 | }, | |
22 | DATABASE: { | |
6ee72211 | 23 | DBNAME: config.has('database.name') ? config.get<string>('database.name') : 'peertube' + config.get<string>('database.suffix'), |
6dd9de95 C |
24 | HOSTNAME: config.get<string>('database.hostname'), |
25 | PORT: config.get<number>('database.port'), | |
b0ce7c39 | 26 | SSL: config.get<boolean>('database.ssl'), |
6dd9de95 C |
27 | USERNAME: config.get<string>('database.username'), |
28 | PASSWORD: config.get<string>('database.password'), | |
29 | POOL: { | |
30 | MAX: config.get<number>('database.pool.max') | |
31 | } | |
32 | }, | |
33 | REDIS: { | |
34 | HOSTNAME: config.has('redis.hostname') ? config.get<string>('redis.hostname') : null, | |
35 | PORT: config.has('redis.port') ? config.get<number>('redis.port') : null, | |
36 | SOCKET: config.has('redis.socket') ? config.get<string>('redis.socket') : null, | |
37 | AUTH: config.has('redis.auth') ? config.get<string>('redis.auth') : null, | |
38 | DB: config.has('redis.db') ? config.get<number>('redis.db') : null | |
39 | }, | |
40 | SMTP: { | |
ed3f089c IB |
41 | TRANSPORT: config.has('smtp.transport') ? config.get<string>('smtp.transport') : 'smtp', |
42 | SENDMAIL: config.has('smtp.sendmail') ? config.get<string>('smtp.sendmail') : null, | |
6dd9de95 C |
43 | HOSTNAME: config.get<string>('smtp.hostname'), |
44 | PORT: config.get<number>('smtp.port'), | |
45 | USERNAME: config.get<string>('smtp.username'), | |
46 | PASSWORD: config.get<string>('smtp.password'), | |
47 | TLS: config.get<boolean>('smtp.tls'), | |
48 | DISABLE_STARTTLS: config.get<boolean>('smtp.disable_starttls'), | |
49 | CA_FILE: config.get<string>('smtp.ca_file'), | |
50 | FROM_ADDRESS: config.get<string>('smtp.from_address') | |
51 | }, | |
b5bfadf0 YB |
52 | EMAIL: { |
53 | BODY: { | |
54 | SIGNATURE: config.get<string>('email.body.signature') | |
55 | }, | |
916937d7 C |
56 | SUBJECT: { |
57 | PREFIX: config.get<string>('email.subject.prefix') + ' ' | |
b5bfadf0 YB |
58 | } |
59 | }, | |
6dd9de95 C |
60 | STORAGE: { |
61 | TMP_DIR: buildPath(config.get<string>('storage.tmp')), | |
62 | AVATARS_DIR: buildPath(config.get<string>('storage.avatars')), | |
63 | LOG_DIR: buildPath(config.get<string>('storage.logs')), | |
64 | VIDEOS_DIR: buildPath(config.get<string>('storage.videos')), | |
65 | STREAMING_PLAYLISTS_DIR: buildPath(config.get<string>('storage.streaming_playlists')), | |
66 | REDUNDANCY_DIR: buildPath(config.get<string>('storage.redundancy')), | |
67 | THUMBNAILS_DIR: buildPath(config.get<string>('storage.thumbnails')), | |
68 | PREVIEWS_DIR: buildPath(config.get<string>('storage.previews')), | |
69 | CAPTIONS_DIR: buildPath(config.get<string>('storage.captions')), | |
70 | TORRENTS_DIR: buildPath(config.get<string>('storage.torrents')), | |
345da516 | 71 | CACHE_DIR: buildPath(config.get<string>('storage.cache')), |
caf2aaf4 K |
72 | PLUGINS_DIR: buildPath(config.get<string>('storage.plugins')), |
73 | CLIENT_OVERRIDES_DIR: buildPath(config.get<string>('storage.client_overrides')) | |
6dd9de95 C |
74 | }, |
75 | WEBSERVER: { | |
76 | SCHEME: config.get<boolean>('webserver.https') === true ? 'https' : 'http', | |
77 | WS: config.get<boolean>('webserver.https') === true ? 'wss' : 'ws', | |
78 | HOSTNAME: config.get<string>('webserver.hostname'), | |
79 | PORT: config.get<number>('webserver.port') | |
80 | }, | |
c342726a | 81 | RATES_LIMIT: { |
c1340a6a C |
82 | API: { |
83 | WINDOW_MS: parseDurationToMs(config.get<string>('rates_limit.api.window')), | |
84 | MAX: config.get<number>('rates_limit.api.max') | |
85 | }, | |
86 | SIGNUP: { | |
87 | WINDOW_MS: parseDurationToMs(config.get<string>('rates_limit.signup.window')), | |
88 | MAX: config.get<number>('rates_limit.signup.max') | |
89 | }, | |
c342726a YB |
90 | LOGIN: { |
91 | WINDOW_MS: parseDurationToMs(config.get<string>('rates_limit.login.window')), | |
92 | MAX: config.get<number>('rates_limit.login.max') | |
93 | }, | |
94 | ASK_SEND_EMAIL: { | |
95 | WINDOW_MS: parseDurationToMs(config.get<string>('rates_limit.ask_send_email.window')), | |
96 | MAX: config.get<number>('rates_limit.ask_send_email.max') | |
97 | } | |
98 | }, | |
6dd9de95 C |
99 | TRUST_PROXY: config.get<string[]>('trust_proxy'), |
100 | LOG: { | |
fcf4569f | 101 | LEVEL: config.get<string>('log.level'), |
2f6b5e2d C |
102 | ROTATION: { |
103 | ENABLED: config.get<boolean>('log.rotation.enabled'), | |
104 | MAX_FILE_SIZE: bytes.parse(config.get<string>('log.rotation.maxFileSize')), | |
105 | MAX_FILES: config.get<number>('log.rotation.maxFiles') | |
106 | }, | |
12c1e38d | 107 | ANONYMIZE_IP: config.get<boolean>('log.anonymizeIP'), |
1e743faa C |
108 | LOG_PING_REQUESTS: config.get<boolean>('log.log_ping_requests'), |
109 | PRETTIFY_SQL: config.get<boolean>('log.prettify_sql') | |
6dd9de95 | 110 | }, |
6dd9de95 C |
111 | TRENDING: { |
112 | VIDEOS: { | |
ba5d4a84 RK |
113 | INTERVAL_DAYS: config.get<number>('trending.videos.interval_days'), |
114 | ALGORITHMS: { | |
115 | get ENABLED () { return config.get<string[]>('trending.videos.algorithms.enabled') }, | |
116 | get DEFAULT () { return config.get<string>('trending.videos.algorithms.default') } | |
117 | } | |
6dd9de95 C |
118 | } |
119 | }, | |
120 | REDUNDANCY: { | |
121 | VIDEOS: { | |
8f0bc73d | 122 | CHECK_INTERVAL: parseDurationToMs(config.get<string>('redundancy.videos.check_interval')), |
6dd9de95 C |
123 | STRATEGIES: buildVideosRedundancy(config.get<any[]>('redundancy.videos.strategies')) |
124 | } | |
125 | }, | |
8c9e7875 C |
126 | REMOTE_REDUNDANCY: { |
127 | VIDEOS: { | |
128 | ACCEPT_FROM: config.get<VideoRedundancyConfigFilter>('remote_redundancy.videos.accept_from') | |
129 | } | |
130 | }, | |
6dd9de95 C |
131 | CSP: { |
132 | ENABLED: config.get<boolean>('csp.enabled'), | |
133 | REPORT_ONLY: config.get<boolean>('csp.report_only'), | |
50fcdebd | 134 | REPORT_URI: config.get<string>('csp.report_uri') |
6dd9de95 C |
135 | }, |
136 | TRACKER: { | |
137 | ENABLED: config.get<boolean>('tracker.enabled'), | |
138 | PRIVATE: config.get<boolean>('tracker.private'), | |
139 | REJECT_TOO_MANY_ANNOUNCES: config.get<boolean>('tracker.reject_too_many_announces') | |
140 | }, | |
8f0bc73d C |
141 | HISTORY: { |
142 | VIDEOS: { | |
143 | MAX_AGE: parseDurationToMs(config.get('history.videos.max_age')) | |
144 | } | |
145 | }, | |
cda03765 C |
146 | VIEWS: { |
147 | VIDEOS: { | |
148 | REMOTE: { | |
149 | MAX_AGE: parseDurationToMs(config.get('views.videos.remote.max_age')) | |
150 | } | |
151 | } | |
152 | }, | |
6702a1b2 C |
153 | PLUGINS: { |
154 | INDEX: { | |
155 | ENABLED: config.get<boolean>('plugins.index.enabled'), | |
d133f385 C |
156 | CHECK_LATEST_VERSIONS_INTERVAL: parseDurationToMs(config.get<string>('plugins.index.check_latest_versions_interval')), |
157 | URL: config.get<string>('plugins.index.url') | |
6702a1b2 C |
158 | } |
159 | }, | |
3092e9bb LB |
160 | FEDERATION: { |
161 | VIDEOS: { | |
162 | FEDERATE_UNLISTED: config.get<boolean>('federation.videos.federate_unlisted') | |
163 | } | |
164 | }, | |
6dd9de95 C |
165 | ADMIN: { |
166 | get EMAIL () { return config.get<string>('admin.email') } | |
167 | }, | |
168 | CONTACT_FORM: { | |
169 | get ENABLED () { return config.get<boolean>('contact_form.enabled') } | |
170 | }, | |
171 | SIGNUP: { | |
172 | get ENABLED () { return config.get<boolean>('signup.enabled') }, | |
173 | get LIMIT () { return config.get<number>('signup.limit') }, | |
174 | get REQUIRES_EMAIL_VERIFICATION () { return config.get<boolean>('signup.requires_email_verification') }, | |
175 | FILTERS: { | |
176 | CIDR: { | |
177 | get WHITELIST () { return config.get<string[]>('signup.filters.cidr.whitelist') }, | |
178 | get BLACKLIST () { return config.get<string[]>('signup.filters.cidr.blacklist') } | |
179 | } | |
180 | } | |
181 | }, | |
182 | USER: { | |
183 | get VIDEO_QUOTA () { return parseBytes(config.get<number>('user.video_quota')) }, | |
184 | get VIDEO_QUOTA_DAILY () { return parseBytes(config.get<number>('user.video_quota_daily')) } | |
185 | }, | |
186 | TRANSCODING: { | |
187 | get ENABLED () { return config.get<boolean>('transcoding.enabled') }, | |
188 | get ALLOW_ADDITIONAL_EXTENSIONS () { return config.get<boolean>('transcoding.allow_additional_extensions') }, | |
536598cf | 189 | get ALLOW_AUDIO_FILES () { return config.get<boolean>('transcoding.allow_audio_files') }, |
6dd9de95 | 190 | get THREADS () { return config.get<number>('transcoding.threads') }, |
1896bca0 | 191 | get PROFILE () { return config.get<string>('transcoding.profile') }, |
6dd9de95 | 192 | RESOLUTIONS: { |
5c7d6508 | 193 | get '0p' () { return config.get<boolean>('transcoding.resolutions.0p') }, |
6dd9de95 C |
194 | get '240p' () { return config.get<boolean>('transcoding.resolutions.240p') }, |
195 | get '360p' () { return config.get<boolean>('transcoding.resolutions.360p') }, | |
196 | get '480p' () { return config.get<boolean>('transcoding.resolutions.480p') }, | |
197 | get '720p' () { return config.get<boolean>('transcoding.resolutions.720p') }, | |
00aa1f0d | 198 | get '1080p' () { return config.get<boolean>('transcoding.resolutions.1080p') }, |
b7085c71 | 199 | get '1440p' () { return config.get<boolean>('transcoding.resolutions.1440p') }, |
00aa1f0d | 200 | get '2160p' () { return config.get<boolean>('transcoding.resolutions.2160p') } |
6dd9de95 C |
201 | }, |
202 | HLS: { | |
203 | get ENABLED () { return config.get<boolean>('transcoding.hls.enabled') } | |
d7a25329 C |
204 | }, |
205 | WEBTORRENT: { | |
206 | get ENABLED () { return config.get<boolean>('transcoding.webtorrent.enabled') } | |
6dd9de95 C |
207 | } |
208 | }, | |
c6c0fa6c C |
209 | LIVE: { |
210 | get ENABLED () { return config.get<boolean>('live.enabled') }, | |
fb719404 C |
211 | |
212 | get MAX_DURATION () { return parseDurationToMs(config.get<string>('live.max_duration')) }, | |
a056ca48 C |
213 | get MAX_INSTANCE_LIVES () { return config.get<number>('live.max_instance_lives') }, |
214 | get MAX_USER_LIVES () { return config.get<number>('live.max_user_lives') }, | |
215 | ||
fb719404 | 216 | get ALLOW_REPLAY () { return config.get<boolean>('live.allow_replay') }, |
c6c0fa6c C |
217 | |
218 | RTMP: { | |
219 | get PORT () { return config.get<number>('live.rtmp.port') } | |
220 | }, | |
221 | ||
222 | TRANSCODING: { | |
223 | get ENABLED () { return config.get<boolean>('live.transcoding.enabled') }, | |
224 | get THREADS () { return config.get<number>('live.transcoding.threads') }, | |
1896bca0 | 225 | get PROFILE () { return config.get<string>('live.transcoding.profile') }, |
c6c0fa6c C |
226 | |
227 | RESOLUTIONS: { | |
228 | get '240p' () { return config.get<boolean>('live.transcoding.resolutions.240p') }, | |
229 | get '360p' () { return config.get<boolean>('live.transcoding.resolutions.360p') }, | |
230 | get '480p' () { return config.get<boolean>('live.transcoding.resolutions.480p') }, | |
231 | get '720p' () { return config.get<boolean>('live.transcoding.resolutions.720p') }, | |
232 | get '1080p' () { return config.get<boolean>('live.transcoding.resolutions.1080p') }, | |
b7085c71 | 233 | get '1440p' () { return config.get<boolean>('live.transcoding.resolutions.1440p') }, |
c6c0fa6c C |
234 | get '2160p' () { return config.get<boolean>('live.transcoding.resolutions.2160p') } |
235 | } | |
236 | } | |
237 | }, | |
6dd9de95 C |
238 | IMPORT: { |
239 | VIDEOS: { | |
240 | HTTP: { | |
be7ca0c6 | 241 | get ENABLED () { return config.get<boolean>('import.videos.http.enabled') }, |
e0409585 | 242 | get FORCE_IPV4 () { return config.get<boolean>('import.videos.http.force_ipv4') }, |
be7ca0c6 C |
243 | PROXY: { |
244 | get ENABLED () { return config.get<boolean>('import.videos.http.proxy.enabled') }, | |
245 | get URL () { return config.get<string>('import.videos.http.proxy.url') } | |
246 | } | |
6dd9de95 C |
247 | }, |
248 | TORRENT: { | |
249 | get ENABLED () { return config.get<boolean>('import.videos.torrent.enabled') } | |
250 | } | |
251 | } | |
252 | }, | |
253 | AUTO_BLACKLIST: { | |
254 | VIDEOS: { | |
255 | OF_USERS: { | |
256 | get ENABLED () { return config.get<boolean>('auto_blacklist.videos.of_users.enabled') } | |
257 | } | |
258 | } | |
259 | }, | |
260 | CACHE: { | |
261 | PREVIEWS: { | |
262 | get SIZE () { return config.get<number>('cache.previews.size') } | |
263 | }, | |
264 | VIDEO_CAPTIONS: { | |
265 | get SIZE () { return config.get<number>('cache.captions.size') } | |
266 | } | |
267 | }, | |
268 | INSTANCE: { | |
269 | get NAME () { return config.get<string>('instance.name') }, | |
270 | get SHORT_DESCRIPTION () { return config.get<string>('instance.short_description') }, | |
271 | get DESCRIPTION () { return config.get<string>('instance.description') }, | |
272 | get TERMS () { return config.get<string>('instance.terms') }, | |
ccc00cb2 C |
273 | get CODE_OF_CONDUCT () { return config.get<string>('instance.code_of_conduct') }, |
274 | ||
8ae03c37 C |
275 | get CREATION_REASON () { return config.get<string>('instance.creation_reason') }, |
276 | ||
ccc00cb2 C |
277 | get MODERATION_INFORMATION () { return config.get<string>('instance.moderation_information') }, |
278 | get ADMINISTRATOR () { return config.get<string>('instance.administrator') }, | |
279 | get MAINTENANCE_LIFETIME () { return config.get<string>('instance.maintenance_lifetime') }, | |
280 | get BUSINESS_MODEL () { return config.get<string>('instance.business_model') }, | |
be04c6fd | 281 | get HARDWARE_INFORMATION () { return config.get<string>('instance.hardware_information') }, |
ccc00cb2 C |
282 | |
283 | get LANGUAGES () { return config.get<string[]>('instance.languages') || [] }, | |
284 | get CATEGORIES () { return config.get<number[]>('instance.categories') || [] }, | |
285 | ||
6dd9de95 | 286 | get IS_NSFW () { return config.get<boolean>('instance.is_nsfw') }, |
6dd9de95 | 287 | get DEFAULT_NSFW_POLICY () { return config.get<NSFWPolicyType>('instance.default_nsfw_policy') }, |
3da68f0a RK |
288 | |
289 | get DEFAULT_CLIENT_ROUTE () { return config.get<string>('instance.default_client_route') }, | |
3da68f0a | 290 | |
6dd9de95 C |
291 | CUSTOMIZATIONS: { |
292 | get JAVASCRIPT () { return config.get<string>('instance.customizations.javascript') }, | |
293 | get CSS () { return config.get<string>('instance.customizations.css') } | |
294 | }, | |
295 | get ROBOTS () { return config.get<string>('instance.robots') }, | |
296 | get SECURITYTXT () { return config.get<string>('instance.securitytxt') }, | |
297 | get SECURITYTXT_CONTACT () { return config.get<string>('admin.email') } | |
298 | }, | |
299 | SERVICES: { | |
300 | TWITTER: { | |
301 | get USERNAME () { return config.get<string>('services.twitter.username') }, | |
302 | get WHITELISTED () { return config.get<boolean>('services.twitter.whitelisted') } | |
303 | } | |
304 | }, | |
305 | FOLLOWERS: { | |
306 | INSTANCE: { | |
307 | get ENABLED () { return config.get<boolean>('followers.instance.enabled') }, | |
308 | get MANUAL_APPROVAL () { return config.get<boolean>('followers.instance.manual_approval') } | |
309 | } | |
7cd4d2ba | 310 | }, |
8424c402 C |
311 | FOLLOWINGS: { |
312 | INSTANCE: { | |
313 | AUTO_FOLLOW_BACK: { | |
314 | get ENABLED () { | |
315 | return config.get<boolean>('followings.instance.auto_follow_back.enabled') | |
316 | } | |
317 | }, | |
318 | AUTO_FOLLOW_INDEX: { | |
319 | get ENABLED () { | |
320 | return config.get<boolean>('followings.instance.auto_follow_index.enabled') | |
321 | }, | |
322 | get INDEX_URL () { | |
323 | return config.get<string>('followings.instance.auto_follow_index.index_url') | |
324 | } | |
325 | } | |
326 | } | |
327 | }, | |
7cd4d2ba C |
328 | THEME: { |
329 | get DEFAULT () { return config.get<string>('theme.default') } | |
72c33e71 C |
330 | }, |
331 | BROADCAST_MESSAGE: { | |
332 | get ENABLED () { return config.get<boolean>('broadcast_message.enabled') }, | |
333 | get MESSAGE () { return config.get<string>('broadcast_message.message') }, | |
334 | get LEVEL () { return config.get<BroadcastMessageLevel>('broadcast_message.level') }, | |
335 | get DISMISSABLE () { return config.get<boolean>('broadcast_message.dismissable') } | |
5fb2e288 C |
336 | }, |
337 | SEARCH: { | |
338 | REMOTE_URI: { | |
339 | USERS: config.get<boolean>('search.remote_uri.users'), | |
340 | ANONYMOUS: config.get<boolean>('search.remote_uri.anonymous') | |
341 | }, | |
342 | SEARCH_INDEX: { | |
343 | get ENABLED () { return config.get<boolean>('search.search_index.enabled') }, | |
344 | get URL () { return config.get<string>('search.search_index.url') }, | |
345 | get DISABLE_LOCAL_SEARCH () { return config.get<boolean>('search.search_index.disable_local_search') }, | |
346 | get IS_DEFAULT_SEARCH () { return config.get<boolean>('search.search_index.is_default_search') } | |
347 | } | |
6dd9de95 C |
348 | } |
349 | } | |
350 | ||
351 | function registerConfigChangedHandler (fun: Function) { | |
352 | configChangedHandlers.push(fun) | |
353 | } | |
354 | ||
4c1c1709 | 355 | function isEmailEnabled () { |
448487a6 C |
356 | if (CONFIG.SMTP.TRANSPORT === 'sendmail' && CONFIG.SMTP.SENDMAIL) return true |
357 | ||
358 | if (CONFIG.SMTP.TRANSPORT === 'smtp' && CONFIG.SMTP.HOSTNAME && CONFIG.SMTP.PORT) return true | |
359 | ||
360 | return false | |
4c1c1709 C |
361 | } |
362 | ||
6dd9de95 C |
363 | // --------------------------------------------------------------------------- |
364 | ||
365 | export { | |
366 | CONFIG, | |
4c1c1709 C |
367 | registerConfigChangedHandler, |
368 | isEmailEnabled | |
6dd9de95 C |
369 | } |
370 | ||
371 | // --------------------------------------------------------------------------- | |
372 | ||
373 | function getLocalConfigFilePath () { | |
374 | const configSources = config.util.getConfigSources() | |
375 | if (configSources.length === 0) throw new Error('Invalid config source.') | |
376 | ||
377 | let filename = 'local' | |
378 | if (process.env.NODE_ENV) filename += `-${process.env.NODE_ENV}` | |
379 | if (process.env.NODE_APP_INSTANCE) filename += `-${process.env.NODE_APP_INSTANCE}` | |
380 | ||
a1587156 | 381 | return join(dirname(configSources[0].name), filename + '.json') |
6dd9de95 C |
382 | } |
383 | ||
b764380a | 384 | function buildVideosRedundancy (objs: any[]): VideosRedundancyStrategy[] { |
6dd9de95 C |
385 | if (!objs) return [] |
386 | ||
387 | if (!Array.isArray(objs)) return objs | |
388 | ||
389 | return objs.map(obj => { | |
390 | return Object.assign({}, obj, { | |
8f0bc73d | 391 | minLifetime: parseDurationToMs(obj.min_lifetime), |
6dd9de95 C |
392 | size: bytes.parse(obj.size), |
393 | minViews: obj.min_views | |
394 | }) | |
395 | }) | |
396 | } | |
397 | ||
398 | export function reloadConfig () { | |
399 | ||
400 | function directory () { | |
401 | if (process.env.NODE_CONFIG_DIR) { | |
402 | return process.env.NODE_CONFIG_DIR | |
403 | } | |
404 | ||
405 | return join(root(), 'config') | |
406 | } | |
407 | ||
408 | function purge () { | |
409 | for (const fileName in require.cache) { | |
bdd428a6 | 410 | if (fileName.includes(directory()) === false) { |
6dd9de95 C |
411 | continue |
412 | } | |
413 | ||
414 | delete require.cache[fileName] | |
415 | } | |
416 | ||
417 | delete require.cache[require.resolve('config')] | |
418 | } | |
419 | ||
420 | purge() | |
421 | ||
422 | config = require('config') | |
423 | ||
424 | for (const configChangedHandler of configChangedHandlers) { | |
425 | configChangedHandler() | |
426 | } | |
427 | } |