Proyectos de Subversion Moodle

Rev

| Ultima modificación | Ver Log |

Rev Autor Línea Nro. Línea
1 efrain 1
<?php
2
 
3
// This file is part of Moodle - http://moodle.org/
4
//
5
// Moodle is free software: you can redistribute it and/or modify
6
// it under the terms of the GNU General Public License as published by
7
// the Free Software Foundation, either version 3 of the License, or
8
// (at your option) any later version.
9
//
10
// Moodle is distributed in the hope that it will be useful,
11
// but WITHOUT ANY WARRANTY; without even the implied warranty of
12
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13
// GNU General Public License for more details.
14
//
15
// You should have received a copy of the GNU General Public License
16
// along with Moodle.  If not, see <http://www.gnu.org/licenses/>.
17
 
18
/**
19
 * @package    portfolio
20
 * @subpackage flickr
21
 * @copyright  2008 Nicolas Connault
22
 * @license    http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
23
 */
24
 
25
defined('MOODLE_INTERNAL') || die();
26
 
27
require_once($CFG->libdir.'/portfolio/plugin.php');
28
require_once($CFG->libdir.'/filelib.php');
29
require_once($CFG->libdir.'/flickrclient.php');
30
 
31
class portfolio_plugin_flickr extends portfolio_plugin_push_base {
32
 
33
    /** @var flickr_client */
34
    private $flickr;
35
    private $raw_sets;
36
 
37
    public function supported_formats() {
38
        return array(PORTFOLIO_FORMAT_IMAGE);
39
    }
40
 
41
    public static function get_name() {
42
        return get_string('pluginname', 'portfolio_flickr');
43
    }
44
 
45
    public function prepare_package() {
46
 
47
    }
48
 
49
    public function send_package() {
50
        foreach ($this->exporter->get_tempfiles() as $file) {
51
            // @TODO get max size from flickr people_getUploadStatus
52
            $filesize = $file->get_filesize();
53
 
54
            if ($this->is_valid_image($file)) {
55
                $photoid = $this->flickr->upload($file, [
56
                    'title' => $this->get_export_config('title'),
57
                    'description' => $this->get_export_config('description'),
58
                    'tags' => $this->get_export_config('tags'),
59
                    'is_public' => $this->get_export_config('is_public'),
60
                    'is_friend' => $this->get_export_config('is_friend'),
61
                    'is_family' => $this->get_export_config('is_family'),
62
                    'safety_level' => $this->get_export_config('safety_level'),
63
                    'content_type' => $this->get_export_config('content_type'),
64
                    'hidden' => $this->get_export_config('hidden'),
65
                ]);
66
 
67
                if ($photoid === false) {
68
                    $this->set_user_config([
69
                        'accesstoken' => null,
70
                        'accesstokensecret' => null,
71
                    ]);
72
                    throw new portfolio_plugin_exception('uploadfailed', 'portfolio_flickr', '', 'Authentication failed');
73
                }
74
 
75
                // Attach photo to a set if requested.
76
                if ($this->get_export_config('set')) {
77
                    $result = $this->flickr->call('photosets.addPhoto', [
78
                        'photoset_id' => $this->get_export_config('set'),
79
                        'photo_id' => $photoid,
80
                    ], 'POST');
81
                }
82
            }
83
        }
84
    }
85
 
86
    public static function allows_multiple_instances() {
87
        return false;
88
    }
89
 
90
    public function get_interactive_continue_url() {
91
        return 'https://www.flickr.com/photos/organize';
92
    }
93
 
94
    public function expected_time($callertime) {
95
        return $callertime;
96
    }
97
 
98
    public static function get_allowed_config() {
99
        return array('apikey', 'sharedsecret');
100
    }
101
 
102
    public static function has_admin_config() {
103
        return true;
104
    }
105
 
106
    public static function admin_config_form(&$mform) {
107
        global $CFG;
108
 
109
        $strrequired = get_string('required');
110
        $mform->addElement('text', 'apikey', get_string('apikey', 'portfolio_flickr'), array('size' => 30));
111
        $mform->addRule('apikey', $strrequired, 'required', null, 'client');
112
        $mform->setType('apikey', PARAM_RAW_TRIMMED);
113
        $mform->addElement('text', 'sharedsecret', get_string('sharedsecret', 'portfolio_flickr'));
114
        $mform->addRule('sharedsecret', $strrequired, 'required', null, 'client');
115
        $mform->setType('sharedsecret', PARAM_RAW_TRIMMED);
116
        $a = new stdClass();
117
        $a->applyurl = 'http://www.flickr.com/services/api/keys/apply/';
118
        $a->keysurl = 'http://www.flickr.com/services/api/keys/';
119
        $a->callbackurl = $CFG->wwwroot . '/portfolio/add.php?postcontrol=1&type=flickr';
120
        $mform->addElement('static', 'setupinfo', get_string('setupinfo', 'portfolio_flickr'),
121
            get_string('setupinfodetails', 'portfolio_flickr', $a));
122
    }
123
 
124
    public function has_export_config() {
125
        return true;
126
    }
127
 
128
    public function get_allowed_user_config() {
129
        return array('accesstoken', 'accesstokensecret');
130
    }
131
 
132
    public function steal_control($stage) {
133
        if ($stage != PORTFOLIO_STAGE_CONFIG) {
134
            return false;
135
        }
136
 
137
        $accesstoken = $this->get_user_config('accesstoken');
138
        $accesstokensecret = $this->get_user_config('accesstokensecret');
139
 
140
        $callbackurl = new moodle_url('/portfolio/add.php', ['postcontrol' => 1, 'type' => 'flickr']);
141
        $this->flickr = new flickr_client($this->get_config('apikey'), $this->get_config('sharedsecret'), $callbackurl);
142
 
143
        if (!empty($accesstoken) && !empty($accesstokensecret)) {
144
            // The user has authenticated us already.
145
            $this->flickr->set_access_token($accesstoken, $accesstokensecret);
146
            return false;
147
        }
148
 
149
        $reqtoken = $this->flickr->request_token();
150
        $this->flickr->set_request_token_secret(['caller' => 'portfolio_flickr'], $reqtoken['oauth_token_secret']);
151
 
152
        $authurl = new moodle_url($reqtoken['authorize_url'], ['perms' => 'write']);
153
 
154
        return $authurl->out(false);
155
    }
156
 
157
    public function post_control($stage, $params) {
158
        if ($stage != PORTFOLIO_STAGE_CONFIG) {
159
            return;
160
        }
161
 
162
        if (empty($params['oauth_token']) || empty($params['oauth_verifier'])) {
163
            throw new portfolio_plugin_exception('noauthtoken', 'portfolio_flickr');
164
        }
165
 
166
        $callbackurl = new moodle_url('/portfolio/add.php', ['postcontrol' => 1, 'type' => 'flickr']);
167
        $this->flickr = new flickr_client($this->get_config('apikey'), $this->get_config('sharedsecret'), $callbackurl);
168
 
169
        $secret = $this->flickr->get_request_token_secret(['caller' => 'portfolio_flickr']);
170
 
171
        // Exchange the request token for the access token.
172
        $accesstoken = $this->flickr->get_access_token($params['oauth_token'], $secret, $params['oauth_verifier']);
173
 
174
        // Store the access token and the access token secret as the user
175
        // config so that we can use it on behalf of the user in next exports.
176
        $this->set_user_config([
177
            'accesstoken' => $accesstoken['oauth_token'],
178
            'accesstokensecret' => $accesstoken['oauth_token_secret'],
179
        ]);
180
    }
181
 
182
    public function export_config_form(&$mform) {
183
        $mform->addElement('text', 'plugin_title', get_string('title', 'portfolio_flickr'));
184
        $mform->setType('plugin_title', PARAM_TEXT);
185
        $mform->addElement('textarea', 'plugin_description', get_string('description'));
186
        $mform->setType('plugin_description', PARAM_CLEANHTML);
187
        $mform->addElement('text', 'plugin_tags', get_string('tags'));
188
        $mform->setType('plugin_tags', PARAM_TAGLIST);
189
        $mform->addElement('checkbox', 'plugin_is_public', get_string('ispublic', 'portfolio_flickr'));
190
        $mform->addElement('checkbox', 'plugin_is_family', get_string('isfamily', 'portfolio_flickr'));
191
        $mform->addElement('checkbox', 'plugin_is_friend', get_string('isfriend', 'portfolio_flickr'));
192
 
193
        $mform->disabledIf('plugin_is_friend', 'plugin_is_public', 'checked');
194
        $mform->disabledIf('plugin_is_family', 'plugin_is_public', 'checked');
195
 
196
        $safety_levels = array(1 => $this->get_export_value_name('safety_level', 1),
197
                               2 => $this->get_export_value_name('safety_level', 2),
198
                               3 => $this->get_export_value_name('safety_level', 3));
199
 
200
        $content_types = array(1 => $this->get_export_value_name('content_type', 1),
201
                               2 => $this->get_export_value_name('content_type', 2),
202
                               3 => $this->get_export_value_name('content_type', 3));
203
 
204
        $hidden_values = array(1,2);
205
 
206
        $mform->addElement('select', 'plugin_safety_level', get_string('safetylevel', 'portfolio_flickr'), $safety_levels);
207
        $mform->addElement('select', 'plugin_content_type', get_string('contenttype', 'portfolio_flickr'), $content_types);
208
        $mform->addElement('advcheckbox', 'plugin_hidden', get_string('hidefrompublicsearches', 'portfolio_flickr'), get_string('yes'), null, $hidden_values);
209
 
210
        $mform->setDefaults(array('plugin_is_public' => true));
211
 
212
        $rawsets = $this->get_sets();
213
        if (!empty($rawsets)) {
214
            $sets = array('0' => '----');
215
            foreach ($rawsets as $key => $value) {
216
                $sets[$key] = $value;
217
            }
218
            $mform->addElement('select', 'plugin_set', get_string('set', 'portfolio_flickr'), $sets);
219
        }
220
    }
221
 
222
    /**
223
     * Fetches a list of current user's photosets (albums) on flickr.
224
     *
225
     * @return array (int)id => (string)title
226
     */
227
    private function get_sets() {
228
 
229
        if (empty($this->raw_sets)) {
230
            $this->raw_sets = $this->flickr->call('photosets.getList');
231
        }
232
 
233
        if ($this->raw_sets === false) {
234
            // Authentication failed, drop the locally stored token to force re-authentication.
235
            $this->set_user_config([
236
                'accesstoken' => null,
237
                'accesstokensecret' => null,
238
            ]);
239
            return array();
240
        }
241
 
242
        $sets = array();
243
        foreach ($this->raw_sets->photosets->photoset as $set) {
244
            $sets[$set->id] = $set->title->_content;
245
        }
246
        return $sets;
247
    }
248
 
249
    public function get_allowed_export_config() {
250
        return array('set', 'title', 'description', 'tags', 'is_public', 'is_family', 'is_friend', 'safety_level', 'content_type', 'hidden');
251
    }
252
 
253
    public function get_export_summary() {
254
        return array(get_string('set', 'portfolio_flickr') => $this->get_export_value_name('set', $this->get_export_config('set')),
255
                     get_string('title', 'portfolio_flickr') => $this->get_export_config('title'),
256
                     get_string('description') => $this->get_export_config('description'),
257
                     get_string('tags') => $this->get_export_config('tags'),
258
                     get_string('ispublic', 'portfolio_flickr') => $this->get_export_value_name('is_public', $this->get_export_config('is_public')),
259
                     get_string('isfamily', 'portfolio_flickr') => $this->get_export_value_name('is_family', $this->get_export_config('is_family')),
260
                     get_string('isfriend', 'portfolio_flickr') => $this->get_export_value_name('is_friend', $this->get_export_config('is_friend')),
261
                     get_string('safetylevel', 'portfolio_flickr') => $this->get_export_value_name('safety_level', $this->get_export_config('safety_level')),
262
                     get_string('contenttype', 'portfolio_flickr') => $this->get_export_value_name('content_type', $this->get_export_config('content_type')),
263
                     get_string('hidefrompublicsearches', 'portfolio_flickr') => $this->get_export_value_name('hidden', $this->get_export_config('hidden')));
264
    }
265
 
266
    private function get_export_value_name($param, $value) {
267
        $params = array('set' => $this->get_sets(),
268
                        'is_public' => array(0 => get_string('no'), 1 => get_string('yes')),
269
                        'is_family' => array(0 => get_string('no'), 1 => get_string('yes')),
270
                        'is_friend' => array(0 => get_string('no'), 1 => get_string('yes')),
271
                        'safety_level' => array(1 => get_string('safe', 'portfolio_flickr'),
272
                                                2 => get_string('moderate', 'portfolio_flickr'),
273
                                                3 => get_string('restricted', 'portfolio_flickr')),
274
                        'content_type' => array(1 => get_string('photo', 'portfolio_flickr'),
275
                                                2 => get_string('screenshot', 'portfolio_flickr'),
276
                                                3 => get_string('other', 'portfolio_flickr')),
277
                        'hidden' => array(1 => get_string('no'), 2 => get_string('yes')));
278
 
279
        if (isset($params[$param][$value])) {
280
            return $params[$param][$value];
281
        } else {
282
            return '-';
283
        }
284
    }
285
 
286
    /**
287
     * For now, flickr doesn't support this because we can't dynamically construct callbackurl
288
     */
289
    public static function allows_multiple_exports() {
290
        return false;
291
    }
292
 
293
    /**
294
     * Verifies the file is a valid optimised image - gif, png and jpeg only.
295
     * Currently, Flickr only supports these file types.
296
     *
297
     * @param stored_file $file
298
     * @return bool true if the file is ok
299
     */
300
    private function is_valid_image(stored_file $file): bool {
301
        $mimetype = $file->get_mimetype();
302
        if (!file_mimetype_in_typegroup($mimetype, 'optimised_image')) {
303
            return false;
304
        }
305
        if (!$info = $file->get_imageinfo()) {
306
            return false;
307
        }
308
        if ($info['mimetype'] !== $mimetype) {
309
            return false;
310
        }
311
        return true;
312
    }
313
}