TrackXML.php 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468
  1. <?php
  2. /* GNU FM -- a free network service for sharing your music listening habits
  3. Copyright (C) 2009 Free Software Foundation, Inc
  4. This program is free software: you can redistribute it and/or modify
  5. it under the terms of the GNU Affero General Public License as published by
  6. the Free Software Foundation, either version 3 of the License, or
  7. (at your option) any later version.
  8. This program is distributed in the hope that it will be useful,
  9. but WITHOUT ANY WARRANTY; without even the implied warranty of
  10. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  11. GNU Affero General Public License for more details.
  12. You should have received a copy of the GNU Affero General Public License
  13. along with this program. If not, see <http://www.gnu.org/licenses/>.
  14. */
  15. require_once($install_path . '/database.php');
  16. require_once($install_path . '/data/Track.php');
  17. require_once($install_path . '/scrobble-utils.php');
  18. require_once('xml.php');
  19. /**
  20. * Class with functions that returns XML-formatted data for tracks.
  21. *
  22. * These functions are mainly used by web service methods.
  23. *
  24. * @package API
  25. */
  26. class TrackXML {
  27. public static function addTags($userid, $artist, $album, $trackName, $tags) {
  28. try {
  29. $track = new Track($trackName, $artist);
  30. $res = $track->addTags($tags, $userid);
  31. } catch (Exception $e) {
  32. return(XML::error('failed', '7', 'Invalid resource specified'));
  33. }
  34. if(!$res) {
  35. $xml = XML::error('failed', '7', 'Invalid resource specified');
  36. } else {
  37. $xml = new SimpleXMLElement('<lfm status="ok"></lfm>');
  38. }
  39. return $xml;
  40. }
  41. public static function removeTag($userid, $artist, $trackName, $tag) {
  42. try {
  43. $track = new Track($trackName, $artist);
  44. $res = $track->removeTag($tag, $userid);
  45. } catch (Exception $e) {
  46. return(XML::error('failed', '7', 'Invalid resource specified'));
  47. }
  48. if(!$res) {
  49. $xml = XML::error('failed', '7', 'Invalid resource specified');
  50. } else {
  51. $xml = new SimpleXMLElement('<lfm status="ok"></lfm>');
  52. }
  53. return $xml;
  54. }
  55. public static function getInfo($artist, $name, $username) {
  56. global $adodb;
  57. try {
  58. $track = new Track($name, $artist);
  59. } catch (Exception $e) {
  60. return(XML::error('failed', '7', 'Invalid resource specified'));
  61. }
  62. $xml = new SimpleXMLElement('<lfm status="ok"></lfm>');
  63. $root = $xml->addChild('track', null);
  64. $root->addChild('name', $track->name);
  65. $root->addChild('mbid', $track->mbid);
  66. $root->addChild('url', $track->getURL());
  67. $root->addChild('duration', $track->duration * 1000);
  68. $streamable = $root->addChild('streamable', $track->streamable);
  69. $streamable->addAttribute('fulltrack', $track->streamable);
  70. $root->addChild('listeners', $track->getListenerCount());
  71. $root->addChild('playcount', $track->getPlayCount());
  72. if($username) {
  73. $userid = $adodb->GetOne('SELECT uniqueid FROM Users WHERE '
  74. . 'username = ' . $adodb->qstr($username));
  75. $root->addChild('userloved', $track->isLoved($userid) ? 1 : 0);
  76. }
  77. return $xml;
  78. }
  79. public static function getTopTags($artist, $name, $limit, $cache) {
  80. try {
  81. $track = new Track($name, $artist);
  82. $res = $track->getTopTags($limit, 0, $cache);
  83. } catch (Exception $e) {
  84. return(XML::error('failed', '7', 'Invalid resource specified'));
  85. }
  86. if(!$res) {
  87. return(XML::error('failed', '6', 'No tags for this track'));
  88. }
  89. $xml = new SimpleXMLElement('<lfm status="ok"></lfm>');
  90. $root = $xml->addChild('toptags', null);
  91. $root->addAttribute('artist', $artist);
  92. $root->addAttribute('track', $name);
  93. foreach ($res as &$row) {
  94. $tag_node = $root->addChild('tag', null);
  95. $tag_node->addChild('name', repamp($row['tag']));
  96. $tag_node->addChild('count', $row['freq']);
  97. $tag_node->addChild('url', Server::getTagURL($row['tag']));
  98. }
  99. return $xml;
  100. }
  101. public static function getTopFans($name, $artistname, $limit, $cache) {
  102. global $adodb;
  103. try {
  104. $track = new Track($name, $artistname);
  105. $res = $track->getTopListeners($limit, 0, False, null, null, $cache);
  106. } catch (Exception $e) {
  107. return XML::error('error', '7', 'Invalid resource specified');
  108. }
  109. $xml = new SimpleXMLElement('<lfm status="ok"></lfm>');
  110. $root = $xml->addChild('topfans', null);
  111. $root->addAttribute('artist', $track->artist_name);
  112. $root->addAttribute('track', $track->name);
  113. foreach($res as &$row) {
  114. try {
  115. $user = new User($row['username']);
  116. $user_node = $root->addChild('user', null);
  117. $user_node->addChild('name', $user->name);
  118. $user_node->addChild('realname', $user->fullname);
  119. $user_node->addChild('url', repamp($user->getURL()));
  120. $image_small = $user_node->addChild('image', null);
  121. $image_small->addAttribute('size', 'small');
  122. $image_medium = $user_node->addChild('image', null);
  123. $image_medium->addAttribute('size', 'medium');
  124. $image_large = $user_node->addChild('image', null);
  125. $image_large->addAttribute('size', 'large');
  126. $user_node->addChild('weight', $row['freq']);
  127. } catch (Exception $e) {}
  128. }
  129. return $xml;
  130. }
  131. public static function getTags($artist, $name, $userid, $limit, $cache) {
  132. try {
  133. $track = new Track($name, $artist);
  134. $res = $track->getTags($userid, $limit, 0, $cache);
  135. } catch (Exception $e) {
  136. return(XML::error('failed', '7', 'Invalid resource specified'));
  137. }
  138. if(!$res) {
  139. return(XML::error('failed', '6', 'No tags for this track'));
  140. }
  141. $xml = new SimpleXMLElement('<lfm status="ok"></lfm>');
  142. $root = $xml->addChild('tags', null);
  143. $root->addAttribute('artist', $artist);
  144. $root->addAttribute('track', $name);
  145. foreach ($res as &$row) {
  146. $tag_node = $root->addChild('tag', null);
  147. $tag_node->addChild('name', repamp($row['tag']));
  148. $tag_node->addChild('url', Server::getTagURL($row['tag']));
  149. }
  150. return $xml;
  151. }
  152. public static function ban($artist, $name, $userid) {
  153. try {
  154. $track = new Track($name, $artist);
  155. $res = $track->ban($userid);
  156. } catch (Exception $e) {
  157. return XML::error('failed', '7', 'Invalid resource specified');
  158. }
  159. if(!$res) {
  160. $xml = XML::error('failed', '7', 'Invalid resource specified');
  161. } else {
  162. $xml = new SimpleXMLElement('<lfm status="ok"></lfm>');
  163. }
  164. return $xml;
  165. }
  166. public static function love($artist, $name, $userid) {
  167. try {
  168. $track = new Track($name, $artist);
  169. $res = $track->love($userid);
  170. } catch (Exception $e) {
  171. return XML::error('failed', '7', 'Invalid resource specified');
  172. }
  173. if(!$res) {
  174. $xml = XML::error('failed', '7', 'Invalid resource specified');
  175. } else {
  176. $xml = new SimpleXMLElement('<lfm status="ok"></lfm>');
  177. }
  178. return $xml;
  179. }
  180. public static function unban($artist, $name, $userid) {
  181. try {
  182. $track = new Track($name, $artist);
  183. $res = $track->unban($userid);
  184. } catch (Exception $e) {
  185. return XML::error('failed', '7', 'Invalid resource specified');
  186. }
  187. if(!$res) {
  188. $xml = XML::error('failed', '7', 'Invalid resource specified');
  189. } else {
  190. $xml = new SimpleXMLElement('<lfm status="ok"></lfm>');
  191. }
  192. return $xml;
  193. }
  194. public static function unlove($artist, $name, $userid) {
  195. try {
  196. $track = new Track($name, $artist);
  197. $res = $track->unlove($userid);
  198. } catch (Exception $e) {
  199. return XML::error('failed', '7', 'Invalid resource specified');
  200. }
  201. if(!$res) {
  202. $xml = XML::error('failed', '7', 'Invalid resource specified');
  203. } else {
  204. $xml = new SimpleXMLElement('<lfm status="ok"></lfm>');
  205. }
  206. return $xml;
  207. }
  208. public static function updateNowPlaying($userid, $artist, $track, $album, $tracknumber, $context, $mbid, $duration, $albumartist, $api_key) {
  209. global $adodb;
  210. $sessionid = Server::getScrobbleSession($userid, $api_key);
  211. $t = array(
  212. 'artist' => $artist,
  213. 'track' => $track,
  214. 'album' => $album,
  215. 'tracknumber' => $tracknumber,
  216. 'mbid' => $mbid,
  217. 'duration' => $duration,
  218. 'albumartist' => $albumartist
  219. );
  220. $t = prepareTrack($userid, $t, 'nowplaying');
  221. // Delete last played track
  222. $query = 'DELETE FROM Now_Playing WHERE sessionid = ?';
  223. $params = array($sessionid);
  224. try {
  225. $adodb->Execute($query, $params);
  226. } catch (Exception $e) {}
  227. // Calculate expiry time
  228. if (!$t['duration'] || ($t['duration'] > 5400)) {
  229. // Default expiry time of 300 seconds if duration is false or above 5400 seconds
  230. $expires = time() + 300;
  231. } else {
  232. $expires = time() + $t['duration'];
  233. }
  234. if ($t['ignored_code'] === 0) {
  235. // Clean up expired tracks in now_playing table
  236. $params = array(time());
  237. $query = 'DELETE FROM Now_Playing WHERE expires < ?';
  238. $adodb->Execute($query, $params);
  239. $adodb->StartTrans();
  240. try {
  241. // getTrackID will create the track in Track table if it doesnt exist
  242. getTrackID($t['artist'], $t['album'], $t['track'], $t['mbid'], $t['duration']);
  243. $params = array($sessionid, $t['track'], $t['artist'], $t['album'], $t['mbid'], $expires);
  244. $query = 'INSERT INTO Now_Playing(sessionid, track, artist, album, mbid, expires) VALUES (?,?,?,?,?,?)';
  245. $adodb->Execute($query, $params);
  246. } catch (Exception $e) {
  247. $adodb->FailTrans();
  248. $adodb->CompleteTrans();
  249. reportError($e->getMessage(), $e->getTraceAsString());
  250. return XML::error('failed', '16', 'The service is temporarily unavailable, please try again.');
  251. }
  252. $adodb->CompleteTrans();
  253. }
  254. $xml = new SimpleXMLElement('<lfm status="ok"></lfm>');
  255. $root = $xml->addChild('nowplaying', null);
  256. $track_node = $root->addChild('track', repamp($t['track']));
  257. $track_node->addAttribute('corrected', $t['track_corrected']);
  258. $artist_node = $root->addChild('artist', repamp($t['artist']));
  259. $artist_node->addAttribute('corrected', $t['artist_corrected']);
  260. $album_node = $root->addChild('album', repamp($t['album']));
  261. $album_node->addAttribute('corrected', $t['album_corrected']);
  262. $albumartist_node = $root->addChild('albumArtist', repamp($t['albumartist']));
  263. $albumartist_node->addAttribute('corrected', $t['albumartist_corrected']);
  264. $ignoredmessage_node = $root->addChild('ignoredMessage', $t['ignored_message']);
  265. $ignoredmessage_node->addAttribute('code', $t['ignored_code']);
  266. return $xml;
  267. }
  268. public static function scrobble($userid, $artist, $track, $timestamp, $album, $context, $streamid, $chosenbyuser, $tracknumber, $mbid, $albumartist, $duration, $api_key) {
  269. global $adodb;
  270. $sessionid = Server::getScrobbleSession($userid, $api_key);
  271. $accepted_count = 0;
  272. $ignored_count = 0;
  273. $tracks_array = array();
  274. // Convert input to track arrays and add them to tracks_array
  275. if (is_array($artist)) {
  276. for ($i = 0; $i < count($artist); $i++) {
  277. $tracks_array[$i] = array(
  278. 'artist' => $artist[$i],
  279. 'track' => $track[$i],
  280. 'timestamp' => $timestamp[$i],
  281. 'album' => $album[$i],
  282. 'tracknumber' => $tracknumber[$i],
  283. 'mbid' => $mbid[$i],
  284. 'albumartist' => $albumartist[$i],
  285. 'duration' => $duration[$i],
  286. );
  287. }
  288. } else {
  289. $tracks_array[0] = array(
  290. 'artist' => $artist,
  291. 'track' => $track,
  292. 'timestamp' => $timestamp,
  293. 'album' => $album,
  294. 'tracknumber' => $tracknumber,
  295. 'mbid' => $mbid,
  296. 'albumartist' => $albumartist,
  297. 'duration' => $duration,
  298. );
  299. }
  300. // Correct and inspect scrobbles to see if some should be ignored
  301. for ($i = 0; $i < count($tracks_array); $i++) {
  302. $tracks_array[$i] = prepareTrack($userid, $tracks_array[$i], 'scrobble');
  303. }
  304. $adodb->StartTrans();
  305. for ($i = 0; $i < count($tracks_array); $i++) {
  306. $t = $tracks_array[$i];
  307. if ($t['ignored_code'] === 0) {
  308. try {
  309. // Create artist, album and track if not already in db
  310. $t['track_id'] = getTrackID($t['artist'], $t['album'], $t['track'], $t['mbid'], $t['duration']);
  311. $t['scrobbletrack_id'] = getScrobbleTrackID($t['artist'], $t['album'], $t['track'], $t['mbid'], $t['duration'], $t['track_id']);
  312. } catch (Exception $e) {
  313. // Roll back database entries, log error and respond with error message
  314. $adodb->FailTrans();
  315. $adodb->CompleteTrans();
  316. reportError($e->getMessage(), $e->getTraceAsString());
  317. return XML::error('failed', '16', 'The service is temporarily unavailable, please try again.');
  318. }
  319. try {
  320. // Scrobble
  321. // TODO last.fm spec says we shouldnt scrobble corrected values,
  322. // so maybe we should only use corrected values for validation and in xml
  323. $query = 'INSERT INTO Scrobbles (userid, artist, album, track, time, mbid, source, rating, length, stid) VALUES (?,?,?,?,?,?,?,?,?,?)';
  324. $params = array(
  325. $userid,
  326. $t['artist'],
  327. $t['album'],
  328. $t['track'],
  329. $t['timestamp'],
  330. $t['mbid'],
  331. null,
  332. null,
  333. $t['duration'],
  334. $t['scrobbletrack_id']
  335. );
  336. $adodb->Execute($query, $params);
  337. } catch (Exception $e) {
  338. // Roll back database entries, log error and respond with error message
  339. $adodb->FailTrans();
  340. $adodb->CompleteTrans();
  341. reportError($e->getMessage(), $e->getTraceAsString());
  342. return XML::error('failed', '16', 'The service is temporarily unavailable, please try again.');
  343. }
  344. }
  345. $tracks_array[$i] = $t;
  346. }
  347. $adodb->CompleteTrans();
  348. // Check if forwarding is enabled before looping through array
  349. $params = array($userid);
  350. $query = 'SELECT userid FROM Service_Connections WHERE userid = ? AND forward = 1';
  351. $forward_enabled = $adodb->CacheGetOne(600, $query, $params);
  352. if ($forward_enabled) {
  353. for ($i = 0; $i < count($tracks_array); $i++) {
  354. $t = $tracks_array[$i];
  355. if ($t['ignored_code'] === 0) {
  356. /* Forward scrobbles, we are forwarding unmodified input submitted by user,
  357. * but only the scrobbles that passed our ignore filters, see prepareTrack(). */
  358. forwardScrobble($userid,
  359. $t['artist_old'],
  360. $t['album_old'],
  361. $t['track_old'],
  362. $t['timestamp_old'],
  363. $t['mbid_old'],
  364. '',
  365. '',
  366. $t['duration_old']);
  367. }
  368. }
  369. }
  370. // Build xml
  371. $xml = new SimpleXMLElement('<lfm status="ok"></lfm>');
  372. $root = $xml->addChild('scrobbles', null);
  373. for ($i = 0; $i < count($tracks_array); $i++) {
  374. $t = $tracks_array[$i];
  375. $scrobble = $root->addChild('scrobble', null);
  376. $track_node = $scrobble->addChild('track', repamp($t['track']));
  377. $track_node->addAttribute('corrected', $t['track_corrected']);
  378. $artist_node = $scrobble->addChild('artist', repamp($t['artist']));
  379. $artist_node->addAttribute('corrected', $t['artist_corrected']);
  380. $album_node = $scrobble->addChild('album', repamp($t['album']));
  381. $album_node->addAttribute('corrected', $t['album_corrected']);
  382. $albumartist_node = $scrobble->addChild('albumArtist', repamp($t['albumartist']));
  383. $albumartist_node->addAttribute('corrected', $t['albumartist_corrected']);
  384. $scrobble->addChild('timestamp', $t['timestamp']);
  385. $ignoredmessage_node = $scrobble->addChild('ignoredMessage', $t['ignored_message']);
  386. $ignoredmessage_node->addAttribute('code', $t['ignored_code']);
  387. if ($t['ignored_code'] === 0) {
  388. $accepted_count += 1;
  389. } else {
  390. $ignored_count += 1;
  391. }
  392. }
  393. $root->addAttribute('accepted', $accepted_count);
  394. $root->addAttribute('ignored', $ignored_count);
  395. return $xml;
  396. }
  397. }