on-demand release 2.3dev
[moodle.git] / portfolio / boxnet / lib.php
1 <?php
2 require_once($CFG->libdir.'/portfolio/plugin.php');
3 require_once($CFG->libdir.'/filelib.php');
4 require_once($CFG->libdir.'/boxlib.php');
6 class portfolio_plugin_boxnet extends portfolio_plugin_push_base {
8     public $boxclient;
9     private $ticket;
10     private $authtoken;
11     private $folders;
12     private $accounttree;
14     public static function get_name() {
15         return get_string('pluginname', 'portfolio_boxnet');
16     }
18     public function prepare_package() {
19         // don't do anything for this plugin, we want to send all files as they are.
20     }
22     public function send_package() {
23         // if we need to create the folder, do it now
24         if ($newfolder = $this->get_export_config('newfolder')) {
25             if (!$created = $this->boxclient->createFolder($newfolder, array('share' => (int)$this->get_export_config('sharefolder')))) {
26                 throw new portfolio_plugin_exception('foldercreatefailed', 'portfolio_boxnet');
27             }
28             $this->folders[$created['folder_id']] = $created['folder_name'];
29             $this->set_export_config(array('folder' => $created['folder_id']));
30         }
31         foreach ($this->exporter->get_tempfiles() as $file) {
32             $return = $this->boxclient->uploadFile(
33                 array(
34                     'file'      => $file,
35                     'folder_id' => $this->get_export_config('folder'),
36                     'share'     => $this->get_export_config('sharefile'),
37                 )
38             );
39             if (array_key_exists('status', $return) && $return['status'] == 'upload_ok'
40                 && array_key_exists('id', $return) && count($return['id']) == 1) {
41                 $this->rename_file($return['id'][array_pop(array_keys($return['id']))], $file->get_filename());
42                 // if this fails, the file was sent but not renamed - this triggers a warning but is not fatal.
43             }
44         }
45         if ($this->boxclient->isError()) {
46             throw new portfolio_plugin_exception('sendfailed', 'portfolio_boxnet', $this->boxclient->getErrorMsg());
47         }
48     }
50     public function get_export_summary() {
51         $allfolders = $this->get_folder_list();
52         if ($newfolder = $this->get_export_config('newfolder')) {
53             $foldername = $newfolder . ' (' . get_string('tobecreated', 'portfolio_boxnet') . ')';
54         } elseif ($this->get_export_config('folder')) {
55             $foldername = $allfolders[$this->get_export_config('folder')];
56         } else {
57             $foldername = '';
58         }
59         return array(
60             get_string('targetfolder', 'portfolio_boxnet') => $foldername
61         );
62     }
64     public function get_interactive_continue_url() {
65         return 'http://box.net/files#0:f:' . $this->get_export_config('folder');
66     }
68     public function expected_time($callertime) {
69         return $callertime;
70     }
72     public static function has_admin_config() {
73         return true;
74     }
76     public static function get_allowed_config() {
77         return array('apikey');
78     }
80     public function has_export_config() {
81         return true;
82     }
84     public function get_allowed_user_config() {
85         return array('authtoken', 'authtokenctime');
86     }
88     public function get_allowed_export_config() {
89         return array('folder', 'newfolder', 'sharefile', 'sharefolder');
90     }
92     public function export_config_form(&$mform) {
93         $folders = $this->get_folder_list();
94         $mform->addElement('checkbox', 'plugin_sharefile', get_string('sharefile', 'portfolio_boxnet'));
95         $mform->addElement('text', 'plugin_newfolder', get_string('newfolder', 'portfolio_boxnet'));
96         $mform->addElement('checkbox', 'plugin_sharefolder', get_string('sharefolder', 'portfolio_boxnet'));
97         $folders[0] = '----';
98         ksort($folders);
99         $mform->addElement('select', 'plugin_folder', get_string('existingfolder', 'portfolio_boxnet'), $folders);
100     }
102     public function export_config_validation(array $data) {
103         $allfolders = $this->get_folder_list();
104         if (in_array($data['plugin_newfolder'], $allfolders)) {
105             return array('plugin_newfolder' => get_string('folderclash', 'portfolio_boxnet'));
106         }
107     }
109     public function admin_config_form(&$mform) {
110         global $CFG;
112         $mform->addElement('text', 'apikey', get_string('apikey', 'portfolio_boxnet'));
113         $mform->addRule('apikey', get_string('required'), 'required', null, 'client');
114         $a = new stdClass();
115         $a->servicesurl = 'http://www.box.net/developers/services';
116         $a->callbackurl = $CFG->wwwroot . '/portfolio/add.php?postcontrol=1&type=boxnet';
117         $mform->addElement('static', 'setupinfo', get_string('setupinfo', 'portfolio_boxnet'),
118             get_string('setupinfodetails', 'portfolio_boxnet', $a));
119     }
121     public function steal_control($stage) {
122         if ($stage != PORTFOLIO_STAGE_CONFIG) {
123             return false;
124         }
125         if ($this->authtoken) {
126             return false;
127         }
128         if (!$this->ensure_ticket()) {
129             throw new portfolio_plugin_exception('noticket', 'portfolio_boxnet');
130         }
131         $token = $this->get_user_config('authtoken', $this->get('user')->id);
132         $ctime= $this->get_user_config('authtokenctime', $this->get('user')->id);
133         if (!empty($token) && (($ctime + 60*60*20) > time())) {
134             $this->authtoken = $token;
135             $this->boxclient->auth_token = $token;
136             return false;
137         }
138         return 'http://www.box.net/api/1.0/auth/'.$this->ticket;
139     }
141     public function post_control($stage, $params) {
142         if ($stage != PORTFOLIO_STAGE_CONFIG) {
143             return;
144         }
145         if (!array_key_exists('auth_token', $params) || empty($params['auth_token'])) {
146             throw new portfolio_plugin_exception('noauthtoken', 'portfolio_boxnet');
147         }
148         $this->authtoken = $params['auth_token'];
149         $this->boxclient->auth_token = $this->authtoken;
150         $this->set_user_config(array('authtoken' => $this->authtoken, 'authtokenctime' => time()), $this->get('user')->id);
151     }
153     private function ensure_ticket() {
154         if (!empty($this->boxclient)) {
155             return true;
156         }
157         $this->boxclient = new boxclient($this->get_config('apikey'), '');
158         $ticket_return = $this->boxclient->getTicket();
159         if ($this->boxclient->isError() || empty($ticket_return)) {
160             throw new portfolio_plugin_exception('noticket', 'portfolio_boxnet');
161         }
162         $this->ticket = $ticket_return['ticket'];
163         return $this->ticket;
164     }
166     private function ensure_account_tree() {
167         if (!empty($this->accounttree)) {
168             return;
169         }
170         if (empty($this->ticket)
171             || empty($this->authtoken)
172             || empty($this->boxclient)) {
173             // if we don't have these we're pretty much screwed
174             throw new portfolio_plugin_exception('folderlistfailed', 'portfolio_boxnet');
175             return false;
176         }
177         $this->accounttree = $this->boxclient->getAccountTree();
178         if ($this->boxclient->isError()) {
179             throw new portfolio_plugin_exception('folderlistfailed', 'portfolio_boxnet');
180         }
181         if (!is_array($this->accounttree)) {
182             return false;
183         }
184     }
186     private function get_folder_list() {
187         if (!empty($this->folders)) {
188             return $this->folders;
189         }
190         $this->ensure_account_tree();
191         $folders = array();
192         foreach ($this->accounttree['folder_id'] as $key => $id) {
193             if (empty($id)) {
194                 continue;
195             }
196             $name = $this->accounttree['folder_name'][$key];
197             if (!empty($this->accounttree['shared'][$key])) {
198                 $name .= ' (' . get_string('sharedfolder', 'portfolio_boxnet') . ')';
199             }
200             $folders[$id] = $name;
201         }
202         $this->folders = $folders;
203         return $folders;
204     }
206     private function rename_file($fileid, $newname) {
207         // look at moving this to the boxnet client class
208         $this->ensure_account_tree();
209         $count = 1;
210         $bits = explode('.', $newname);
211         $suffix = '';
212         if (count($bits) == 1) {
213             $prefix = $newname;
214         } else {
215             $suffix = '.' . array_pop($bits);
216             $prefix = implode('.', $bits);
217         }
218         while (true) {
219             if (!array_key_exists('file_name', $this->accounttree) || !in_array($newname, $this->accounttree['file_name'])) {
220                 for ($i = 0; $i < 2; $i++) {
221                     if ($this->boxclient->renameFile($fileid, $newname)) {
222                         return true;
223                     }
224                 }
225                 debugging("tried three times to rename file and failed");
226                 return false;
227             }
228             $newname = $prefix . '(' . $count . ')' . $suffix;
229             $count++;
230         }
231         return false;
232     }
234     public function instance_sanity_check() {
235         if (!$this->get_config('apikey')) {
236             return 'err_noapikey';
237         }
238     //@TODO see if we can verify the api key without actually getting an authentication token
239     }
241     public static function allows_multiple_instances() {
242         return false;
243     }
245     public function supported_formats() {
246         return array(PORTFOLIO_FORMAT_FILE, PORTFOLIO_FORMAT_RICHHTML);
247     }
249     /*
250      * for now , boxnet doesn't support this,
251      * because we can't dynamically construct return urls.
252      */
253     public static function allows_multiple_exports() {
254         return false;
255     }