-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathuser-data-loader.lua
209 lines (172 loc) · 6.2 KB
/
user-data-loader.lua
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
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
--[[
A simple script that attempts to make it easier to configure
the `user-data` property using config files.
The script has two features:
- user-data properties can be set on startup using config files
- user-data can be set during runtime using script-opts
See the README for documentation.
This file is under the MIT license, see Github for the license file.
Project available at: https://github.com/CogentRedTester/mpv-user-data-loader
]]
local mp = require 'mp'
local msg = require 'mp.msg'
local utils = require 'mp.utils'
-- A map that stores all the currently applied script-opts.
-- The key is the user-data path of the field, and the value is
-- the result of the set_value call that modified the field.
local overrides = {}
local config_path = mp.command_native({'expand-path', '~~/script-opts/user-data.conf'})
local json_config_file = mp.command_native({'expand-path', '~~/script-opts/user-data.json'})
-- tests if two complex values are equal
local function is_equal(a,b)
return utils.to_string(a) == utils.to_string(b)
end
-- parse trailing options from script-opts
local function parse_value_opts(trail)
local opts = { restore = 'copy' } -- defaults
for key, value in string.gmatch(trail, '([%w_%-]+)%s*=%s*([%w_%-]+)') do
opts[key] = value
end
return opts
end
-- substitute special characters
-- extended from RFC 6901
local function substitute_chars(str)
return string.gsub(str, '~%d', {
['~0'] = '~',
['~1'] = '/',
['~2'] = '\n',
['~3'] = '\r',
['~4'] = '=',
})
end
-- Sets a field of the user-data property.
-- Appends `user-data/` to `key` and passes value through natively.
local function set_value_native(key, ud_value)
local ud_key = 'user-data/'..key
local prev_val = mp.get_property_native(ud_key)
if not is_equal(prev_val, ud_value) then
msg.verbose('setting', ud_key, utils.to_string(ud_value))
mp.set_property_native(ud_key, ud_value)
end
return true, {
key = ud_key,
value = ud_value,
prev_value = prev_val,
}
end
-- Sets a field of the user-data property.
-- Appends `user-data/` to `key` and parses `value` as a json string.
local function set_value(key, value)
local ud_value, err, trail = utils.parse_json(value, true)
if (err) then
msg.error(err)
msg.warn('failed to parse user-data JSON string:', value)
return false
end
key = substitute_chars(key)
local success, vars = set_value_native(key, ud_value)
vars.trail = trail
vars.opts = parse_value_opts(trail)
return success, vars
end
-- restores the user-data for the given key using the override parameters
local function restore(key, override)
msg.verbose('restoring', key)
local prev_value = override.prev_value
local restore_opt = override.opts.restore
if restore_opt == 'no' then return end
if restore_opt == 'copy-equal' and not is_equal(override.value, mp.get_property_native(key)) then
msg.verbose(key, 'not equal to original value', utils.to_string(override.value), '- not restoring')
return
end
if restore_opt == 'copy-equal' or restore_opt == 'copy' then
if prev_value ~= nil then
msg.verbose('setting', key, utils.to_string(prev_value))
mp.set_property_native(key, prev_value)
else
msg.verbose('deleting', key)
mp.del_property(key)
end
end
end
-- detects changes to the script-opts property that requires `user-data` updates
local function main(_, script_opts)
local current_overrides = {}
-- Finds any user-data value overrides in `script-opts`.
for key, value in pairs(script_opts) do
if string.find(key, '^user[_-]data/.') then
local succeeds, vars = set_value(string.match(key, '^user[_-]data/(.+)') or '', value)
if succeeds and vars then
current_overrides[vars.key] = true
msg.debug('script-opt values:', utils.to_string(vars))
if not overrides[vars.key] and vars.opts.restore ~= 'no' then
msg.debug('setting override')
overrides[vars.key] = vars
end
end
end
end
-- If any values have been removed from script-opts we want to reset them to the original values.
-- This is to add synergy with conditional auto profiles.
for key, vars in pairs(overrides) do
if not current_overrides[key] then
restore(key, vars)
overrides[key] = nil
end
end
end
-- loading the values from user-data.conf
-- uses normal script-opts syntax
local function setup_config()
local file = io.open(config_path, 'r')
if not file then
msg.debug('could not read', config_path)
return
end
msg.debug('reading values from', config_path)
local i = 1
for line in file:lines() do
if line ~= '' and string.sub(line, 1, 1) ~= '#' then
local key, value = string.match(line, '^([^=]+)=(.+)')
if not key or not value then
msg.error(('line %d invalid in %s: "%s"'):format(i, config_path, line))
else
set_value(key, value)
end
end
i = i + 1
end
file:close()
end
-- loading values from user-data.json
-- uses JSON syntax
local function setup_json()
local file = io.open(json_config_file, 'r')
if not file then
msg.debug('could not read', json_config_file)
return
end
msg.debug('reading values from', json_config_file)
local json, err = utils.parse_json(file:read("*a"))
file:close()
if err then
msg.error(err, 'failed to parse JSON file', json_config_file)
msg.warn('check the syntax!')
elseif type(json) ~= 'table' then
msg.warn(json_config_file)
msg.error('json file must be a an object')
else
for key, value in pairs(json) do
set_value_native(key, value)
end
end
end
-- Loading initial user-data values from config files
local function setup()
msg.debug('reading initial values from config files')
setup_json()
setup_config()
end
setup()
mp.observe_property('script-opts', 'native', main)