Source: jDrupal/src/services/services.js

  1. /**
  2. * The Drupal services JSON object.
  3. */
  4. Drupal.services = {};
  5. /**
  6. * Drupal Services XMLHttpRequest Object.
  7. * @param {Object} options
  8. */
  9. Drupal.services.call = function(options) {
  10. try {
  11. options.debug = false;
  12. // Make sure the settings have been provided for Services.
  13. if (!services_ready()) {
  14. var error = 'Set the site_path and endpoint on Drupal.settings!';
  15. options.error(null, null, error);
  16. return;
  17. }
  18. module_invoke_all('services_preprocess', options);
  19. // Build the Request, URL and extract the HTTP method.
  20. var request = new XMLHttpRequest();
  21. var url = Drupal.settings.site_path +
  22. Drupal.settings.base_path + '?q=';
  23. // Use an endpoint, unless someone passed in an empty string.
  24. if (typeof options.endpoint === 'undefined') { url += Drupal.settings.endpoint + '/'; }
  25. else if (options.endpoint != '') { url += options.endpoint + '/'; }
  26. url += options.path;
  27. var method = options.method.toUpperCase();
  28. if (Drupal.settings.debug) { console.log(method + ': ' + url); }
  29. // Watch for net::ERR_CONNECTION_REFUSED and other oddities.
  30. request.onreadystatechange = function() {
  31. if (request.readyState == 4 && request.status == 0) {
  32. if (options.error) { options.error(request, 0, 'xhr network status problem'); }
  33. }
  34. };
  35. // Request Success Handler
  36. request.onload = function(e) {
  37. try {
  38. if (request.readyState == 4) {
  39. // Build a human readable response title.
  40. var title = request.status + ' - ' + request.statusText;
  41. // 200 OK
  42. if (request.status == 200) {
  43. if (Drupal.settings.debug) { console.log('200 - OK'); }
  44. // Extract the JSON result, or throw an error if the response wasn't
  45. // JSON.
  46. // Extract the JSON result if the server sent back JSON, otherwise
  47. // hand back the response as is.
  48. var result = null;
  49. var response_header = request.getResponseHeader('Content-Type');
  50. if (response_header.indexOf('application/json') != -1) {
  51. result = JSON.parse(request.responseText);
  52. }
  53. else { result = request.responseText; }
  54. // Give modules a chance to pre post process the results, send the
  55. // results to the success callback, then give modules a chance to
  56. // post process the results.
  57. module_invoke_all(
  58. 'services_request_pre_postprocess_alter',
  59. options,
  60. result
  61. );
  62. options.success(result);
  63. module_invoke_all(
  64. 'services_request_postprocess_alter',
  65. options,
  66. result
  67. );
  68. module_invoke_all('services_postprocess', options, result);
  69. }
  70. else {
  71. // Not OK...
  72. console.log(method + ': ' + url + ' - ' + title);
  73. if (Drupal.settings.debug) {
  74. if (!in_array(request.status, [403, 503])) { console.log(request.responseText); }
  75. console.log(request.getAllResponseHeaders());
  76. }
  77. if (typeof options.error !== 'undefined') {
  78. var message = request.responseText || '';
  79. if (!message || message == '') { message = title; }
  80. options.error(request, request.status, message);
  81. }
  82. module_invoke_all('services_postprocess', options, request);
  83. }
  84. }
  85. else {
  86. console.log('Drupal.services.call - request.readyState = ' + request.readyState);
  87. }
  88. }
  89. catch (error) {
  90. // Not OK...
  91. if (Drupal.settings.debug) {
  92. console.log(method + ': ' + url + ' - ' + request.statusText);
  93. console.log(request.responseText);
  94. console.log(request.getAllResponseHeaders());
  95. }
  96. console.log('Drupal.services.call - onload - ' + error);
  97. }
  98. };
  99. // Get the CSRF Token and Make the Request.
  100. services_get_csrf_token({
  101. debug: options.debug,
  102. success: function(token) {
  103. try {
  104. // Async, or sync? By default we'll use async if none is provided.
  105. var async = true;
  106. if (typeof options.async !== 'undefined' &&
  107. options.async === false) { async = false; }
  108. // Open the request.
  109. request.open(method, url, async);
  110. // Determine content type header, if necessary.
  111. var contentType = null;
  112. if (method == 'POST') {
  113. contentType = 'application/json';
  114. // The user login resource needs a url encoded data string.
  115. if (options.service == 'user' &&
  116. options.resource == 'login') {
  117. contentType = 'application/x-www-form-urlencoded';
  118. }
  119. }
  120. else if (method == 'PUT') { contentType = 'application/json'; }
  121. // Anyone overriding the content type?
  122. if (options.contentType) { contentType = options.contentType; }
  123. // Set the content type on the header, if we have one.
  124. if (contentType) {
  125. request.setRequestHeader('Content-type', contentType);
  126. }
  127. // Add the token to the header if we have one.
  128. if (token) {
  129. request.setRequestHeader('X-CSRF-Token', token);
  130. }
  131. // Any timeout handling?
  132. if (options.timeout) {
  133. request.timeout = options.timeout;
  134. if (options.ontimeout) { request.ontimeout = options.ontimeout; }
  135. }
  136. // Send the request with or without data.
  137. if (typeof options.data !== 'undefined') {
  138. // Print out debug information if debug is enabled. Don't print
  139. // out any sensitive debug data containing passwords.
  140. if (Drupal.settings.debug) {
  141. var show = true;
  142. if (
  143. (options.service == 'user' && in_array(options.resource, ['login', 'create', 'update'])) ||
  144. (options.service == 'file' && options.resource == 'create')
  145. ) { show = false; }
  146. if (show) {
  147. if (typeof options.data === 'object') {
  148. console.log(JSON.stringify(options.data));
  149. }
  150. else { console.log(options.data); }
  151. }
  152. }
  153. request.send(options.data);
  154. }
  155. else { request.send(null); }
  156. }
  157. catch (error) {
  158. console.log(
  159. 'Drupal.services.call - services_get_csrf_token - success - ' +
  160. error
  161. );
  162. }
  163. },
  164. error: function(xhr, status, message) {
  165. try {
  166. if (options.error) { options.error(xhr, status, message); }
  167. }
  168. catch (error) {
  169. console.log(
  170. 'Drupal.services.call - services_get_csrf_token - error - ' +
  171. error
  172. );
  173. }
  174. }
  175. });
  176. }
  177. catch (error) {
  178. console.log('Drupal.services.call - error - ' + error);
  179. }
  180. };
  181. /**
  182. * Gets the CSRF token from Services.
  183. * @param {Object} options
  184. */
  185. function services_get_csrf_token(options) {
  186. try {
  187. var token;
  188. // Are we resetting the token?
  189. if (options.reset) { Drupal.sessid = null; }
  190. // Do we already have a token? If we do, return it to the success callback.
  191. if (Drupal.sessid) { token = Drupal.sessid; }
  192. if (token) {
  193. if (options.success) { options.success(token); }
  194. return;
  195. }
  196. // We don't have a token, let's get it from Drupal...
  197. // Build the Request and URL.
  198. var token_request = new XMLHttpRequest();
  199. options.token_url = Drupal.settings.site_path + Drupal.settings.base_path + '?q=services/session/token';
  200. module_invoke_all('csrf_token_preprocess', options);
  201. // Watch for net::ERR_CONNECTION_REFUSED and other oddities.
  202. token_request.onreadystatechange = function() {
  203. if (token_request.readyState == 4 && token_request.status == 0) {
  204. if (options.error) { options.error(token_request, 0, 'xhr network status problem for csrf token'); }
  205. }
  206. };
  207. // Token Request Success Handler
  208. token_request.onload = function(e) {
  209. try {
  210. if (token_request.readyState == 4) {
  211. var title = token_request.status + ' - ' +
  212. http_status_code_title(token_request.status);
  213. if (token_request.status != 200) { // Not OK
  214. if (options.error) { options.error(token_request, token_request.status, token_request.responseText); }
  215. }
  216. else { // OK
  217. // Set Drupal.sessid with the token, then return the token to the success function.
  218. token = token_request.responseText.trim();
  219. Drupal.sessid = token;
  220. if (options.success) { options.success(token); }
  221. }
  222. }
  223. else {
  224. console.log('services_get_csrf_token - readyState - ' + token_request.readyState);
  225. }
  226. }
  227. catch (error) {
  228. console.log('services_get_csrf_token - token_request. onload - ' + error);
  229. }
  230. };
  231. // Open the token request.
  232. token_request.open('GET', options.token_url, true);
  233. // Send the token request.
  234. token_request.send(null);
  235. }
  236. catch (error) { console.log('services_get_csrf_token - ' + error); }
  237. }
  238. /**
  239. * Checks if we're ready to make a Services call.
  240. * @return {Boolean}
  241. */
  242. function services_ready() {
  243. try {
  244. var result = true;
  245. if (Drupal.settings.site_path == '') {
  246. result = false;
  247. console.log('jDrupal\'s Drupal.settings.site_path is not set!');
  248. }
  249. if (Drupal.settings.endpoint == '') {
  250. result = false;
  251. console.log('jDrupal\'s Drupal.settings.endpoint is not set!');
  252. }
  253. return result;
  254. }
  255. catch (error) { console.log('services_ready - ' + error); }
  256. }
  257. /**
  258. * Given the options for a service call, the service name and the resource name,
  259. * this will attach the names and their values as properties on the options.
  260. * @param {Object} options
  261. * @param {String} service
  262. * @param {String} resource
  263. */
  264. function services_resource_defaults(options, service, resource) {
  265. if (!options.service) { options.service = service; }
  266. if (!options.resource) { options.resource = resource; }
  267. }
  268. /**
  269. * Returns true if the entity_id is already queued for the service resource,
  270. * false otherwise.
  271. * @param {String} service
  272. * @param {String} resource
  273. * @param {Number} entity_id
  274. * @param {String} callback_type
  275. * @return {Boolean}
  276. */
  277. function _services_queue_already_queued(service, resource, entity_id, callback_type) {
  278. try {
  279. var queued = false;
  280. if (typeof Drupal.services_queue[service][resource][entity_id] !== 'undefined') {
  281. var queue = Drupal.services_queue[service][resource][entity_id];
  282. if (queue[callback_type].length != 0) { queued = true; }
  283. }
  284. return queued;
  285. }
  286. catch (error) { console.log('_services_queue_already_queued - ' + error); }
  287. }
  288. /**
  289. * Adds an entity id to the service resource queue.
  290. * @param {String} service
  291. * @param {String} resource
  292. * @param {Number} entity_id
  293. */
  294. function _services_queue_add_to_queue(service, resource, entity_id) {
  295. try {
  296. Drupal.services_queue[service][resource][entity_id] = {
  297. entity_id: entity_id,
  298. success: [],
  299. error: []
  300. };
  301. }
  302. catch (error) { console.log('_services_queue_add_to_queue - ' + error); }
  303. }
  304. /**
  305. * An internal function used to reset a services callback queue for a given entity CRUD op.
  306. * @param {String} entity_type
  307. * @param {String} resource - create, retrieve, update, delete, index, etc
  308. * @param {Number} entity_id
  309. * @param {String} callback_type - success or error
  310. * @private
  311. */
  312. function _services_queue_clear(entity_type, resource, entity_id, callback_type) {
  313. try {
  314. Drupal.services_queue[entity_type]['retrieve'][entity_id][callback_type] = [];
  315. }
  316. catch (error) { console.log('_services_queue_clear - ' + error); }
  317. }
  318. /**
  319. * Removes an entity id from the service resource queue.
  320. * @param {String} service
  321. * @param {String} resource
  322. * @param {Number} entity_id
  323. */
  324. function _services_queue_remove_from_queue(service, resource, entity_id) {
  325. console.log('WARNING: services_queue_remove_from_queue() not done yet!');
  326. }
  327. /**
  328. * Adds a callback function to the service resource queue.
  329. * @param {String} service
  330. * @param {String} resource
  331. * @param {Number} entity_id
  332. * @param {String} callback_type
  333. * @param {Function} callback
  334. */
  335. function _services_queue_callback_add(service, resource, entity_id, callback_type, callback) {
  336. try {
  337. Drupal.services_queue[service][resource][entity_id][callback_type].push(
  338. callback
  339. );
  340. }
  341. catch (error) { console.log('_services_queue_callback_add - ' + error); }
  342. }
  343. /**
  344. * Returns the number of callback functions for the service resource queue.
  345. * @param {String} service
  346. * @param {String} resource
  347. * @param {Number} entity_id
  348. * @param {String} callback_type
  349. * @return {Number}
  350. */
  351. function _services_queue_callback_count(service, resource, entity_id,
  352. callback_type) {
  353. try {
  354. var length =
  355. Drupal.services_queue[service][resource][entity_id][callback_type].length;
  356. return length;
  357. }
  358. catch (error) { console.log('_services_queue_callback_count - ' + error); }
  359. }