test_playlistcommands.js
Go to the documentation of this file.
1 /*
2  *=BEGIN SONGBIRD GPL
3  *
4  * This file is part of the Songbird web player.
5  *
6  * Copyright(c) 2005-2011 POTI, Inc.
7  * http://www.songbirdnest.com
8  *
9  * This file may be licensed under the terms of of the
10  * GNU General Public License Version 2 (the ``GPL'').
11  *
12  * Software distributed under the License is distributed
13  * on an ``AS IS'' basis, WITHOUT WARRANTY OF ANY KIND, either
14  * express or implied. See the GPL for the specific language
15  * governing rights and limitations.
16  *
17  * You should have received a copy of the GPL along with this
18  * program. If not, go to http://www.gnu.org/licenses/gpl.html
19  * or write to the Free Software Foundation, Inc.,
20  * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
21  *
22  *=END SONGBIRD GPL
23  */
24 
29 // ============================================================================
30 // GLOBALS & CONSTS
31 // ============================================================================
32 
33 const NUMRECURSE = 2;
34 const TESTLENGTH = 500;
36  ["medialist", "guid", "servicepane"],
37  ["medialist", "guid", "menu"],
38  ["medialist", "type", "servicepane"],
39  ["medialist", "type", "menu"],
40  ["library", "guid", "servicepane"],
41  ["library", "guid", "menu"]
42 ]
43 
44 const PlaylistCommandsBuilder = new Components.
45  Constructor("@songbirdnest.com/Songbird/PlaylistCommandsBuilder;1",
46  "sbIPlaylistCommandsBuilder", "init");
47 
48 var gCmdMgr =
49  Components.classes["@songbirdnest.com/Songbird/PlaylistCommandsManager;1"]
50  .createInstance(Components.interfaces.sbIPlaylistCommandsManager);
51 
56 
57 /* An instruction construct that has all the information necessary to
58  * perform, validate, and reverse a playlist command registration
59  */
60 function RegistrationInstruction(aMediaListOrLibrary,
61  aTypeOrGUID,
62  aServicePaneOrMenu,
63  aRegisterString,
64  aCommand) {
65  // Register this command to a medialist or a whole library?
66  this.mediaListOrLibrary = aMediaListOrLibrary;
67 
68  // Register this command to a medialist by guid or type?
69  // For library commands we always set this to 'guid'.
70  this.typeOrGUID = aTypeOrGUID;
71 
72  // Register this command to appear in the 'menu', i.e. the mediaitem context
73  // menu or toolbar, or the 'servicepane'?
74  this.servicePaneOrMenu = aServicePaneOrMenu;
75 
76  // Depending on if guid or type was specified above this will contain the
77  // guid or type to register to respectively
78  this.registerString = aRegisterString;
79 
80  // The command to register
81  this.command = aCommand;
82 }
83 
84 RegistrationInstruction.prototype.toString = function() {
85  var str = "Command '" + this.command.id + "' to '" + this.mediaListOrLibrary;
86  str += "' by '" + this.typeOrGUID + "' in the '" + this.servicePaneOrMenu;
87  str += "' of '" + this.registerString + "'";
88  return str;
89 }
90 
91 // ============================================================================
92 // ENTRY POINT
93 // ============================================================================
94 
95 function runTest () {
96  if (typeof(SBProperties) == "undefined") {
97  Components.utils.import("resource://app/jsmodules/sbProperties.jsm");
98  if (!SBProperties)
99  throw new Error("Import of sbProperties module failed!");
100  }
101 
102  log("Testing Playlist Commands:");
103  gTestPrefix = "PlaylistCommandsBuilder";
104  testCommandsBuilder();
105 
106  gTestPrefix = "Playlist Commands Registration";
108  log("OK");
109 }
110 
111 // ============================================================================
112 // TESTS
113 // ============================================================================
114 
115 function testCommandsBuilder() {
116 
117  var builder = new PlaylistCommandsBuilder("test-command-01");
118 
119  testAppendInsertRemove(builder, null, TESTLENGTH, 0);
121  testSubmenus(builder, null, NUMRECURSE, 0);
122 
123  builder.shutdown();
124 }
125 
126 function testRegistration() {
127  _log("Testing Playlist Command Registration");
128 
129  // Create our gTestLibrary. We have to register it so the playlist commands
130  // manager can find it for registration.
131  var libraryManager = Cc["@songbirdnest.com/Songbird/library/Manager;1"]
132  .getService(Ci.sbILibraryManager);
133  var databaseGUID = "test_playlistcommands_registration";
134  gTestLibrary = createLibrary(databaseGUID, false);
135 
136  libraryManager.registerLibrary(gTestLibrary, false);
137 
138  // Create our test medialists. We will only add commands directly to
139  // gPrimarySimpleList, the others are to confirm that a command wasn't
140  // added somewhere it shouldn't have been.
141  gPrimarySimpleList = gTestLibrary.createMediaList("simple");
142  gSecondarySimpleList = gTestLibrary.createMediaList("simple");
143  gSmartList = gTestLibrary.createMediaList("smart");
144 
145  // And create a very simple command with one action
146  var newCommandId = "test-command-02";
147  var newCommand = PlaylistCommandsBuilder(newCommandId);
148 
149  var newActionId = newCommandId + "-action"
150  var triggerCallback = makeTriggerCallback(newActionId);
151  newCommand.appendAction(null,
152  newActionId,
153  "label goes here",
154  "tooltip goes here",
155  triggerCallback);
156 
157  // Construct a list of instructions that is an exhaustive list of
158  // the ways commands can be registered.
159  var instructions = constructRegistrationInstructions(gTestLibrary,
160  gPrimarySimpleList,
161  newCommand);
162 
163  // Perform, validate, reverse, and validate the reversal of each instruction
164  handleInstructions(instructions);
165 
166  // Cleanup
167  libraryManager.unregisterLibrary(gTestLibrary);
168  newCommand.shutdownCommands();
169  _log("Playlist Command Registration Test Complete");
170 }
171 
172 /* This method constructs an array of RegistrationInstructions based on
173  * REGISTRATION_PARAM_COMBOS which is an exhaustive list of the
174  * "medialist" or "library", "guid" or "type", and "servicepane" or "menu"
175  * combinations that represent all the potential methods of
176  * registering a playlist command
177  */
178 function constructRegistrationInstructions(aLibrary, aMediaList, aCommand) {
179  _log("Constructing Test Instructions");
180  var instructions = [];
181 
182  var len = REGISTRATION_PARAM_COMBOS.length
183  for (var i = 0; i < len; i++) {
184  var params = REGISTRATION_PARAM_COMBOS[i];
185 
186  // a guid or type that will target our instruction
187  var registerString = "";
188 
189  if (params[0] == "library") {
190  // if we are targetting a library we know we'll use the guid
191  registerString = aLibrary.guid;
192  }
193  else if (params[1] == "guid") {
194  // if we are targetting a medialist by its guid, save the guid
195  registerString = aMediaList.guid;
196  }
197  else {
198  // if we are targetting a medialist by its type, save the type
199  registerString = aMediaList.type;
200  }
201 
202  var newInstruction = new RegistrationInstruction(params[0], // medialist or library
203  params[1], // guid or type
204  params[2], // servicepane or menu
205  registerString,
206  aCommand);
207  instructions.push(newInstruction);
208  }
209  return instructions;
210 }
211 
212 /* Handle each of the instructions that we created individually by performing
213  * the instruction (registering the command), validating that it was
214  * performed correctly, reversing the instruction (unregistering the command),
215  * and validating that it was reversed correctly.
216  */
217 function handleInstructions(aInstructions) {
218  _log("Enacting Instructions");
219 
220  var len = aInstructions.length
221  for (var i = 0; i < len; i++) {
222  var currInstruction = aInstructions[i];
223  performInstruction(currInstruction);
224  assertInstruction(currInstruction);
225  reverseInstruction(currInstruction);
226  assertReversedInstruction(currInstruction);
227  }
228 }
229 
230 /* This method enacts a RegistrationInstruction, registering a playlist command
231  * to a library, medialist, or medialist type.
232  */
233 function performInstruction(aInstruction) {
234  // We use different registration methods if we are targetting the servicepane
235  var targetServicePane = (aInstruction.servicePaneOrMenu == "servicepane");
236 
237  if (aInstruction.mediaListOrLibrary == "library") {
238  gCmdMgr.registerPlaylistCommandsForLibrary(targetServicePane,
239  gTestLibrary,
240  aInstruction.command);
241 
242  }
243  else {
244  // Registering to a medialist not a library.
245  // We must use registerPlaylistCommandsMediaItem if our target is a menu.
246  // We must use registerPlaylistCommandsMediaList if it's the servicepane.
247  var registerFunction = (targetServicePane ?
248  gCmdMgr.registerPlaylistCommandsMediaList :
249  gCmdMgr.registerPlaylistCommandsMediaItem);
250 
251  // One of guid or type will be null while the other will take on our
252  // saved value depending on if the instructions wants us to
253  // target by guid or type.
254  var guid = (aInstruction.typeOrGUID == "guid" ? aInstruction.registerString :
255  "");
256 
257  var type = (aInstruction.typeOrGUID == "type" ? aInstruction.registerString :
258  "");
259 
260  // Perform the registration
261  registerFunction(guid, type, aInstruction.command);
262  }
263 }
264 
265 /* This method reverses a RegistrationInstruction, unregistering a playlist
266  * command from a library, medialist, or medialist type.
267  */
268 function reverseInstruction(aInstruction) {
269  var targetServicePane = (aInstruction.servicePaneOrMenu == "servicepane");
270 
271  if (aInstruction.mediaListOrLibrary == "library") {
272  gCmdMgr.unregisterPlaylistCommandsForLibrary(targetServicePane,
273  gTestLibrary,
274  aInstruction.command);
275 
276  }
277  else {
278  // registering to a medialist not a library
279  // use registerPlaylistCommandsMediaItem if target is a menu
280  // use registerPlaylistCommandsMediaList if target is the servicepane
281  var unregisterFunction = (targetServicePane ?
282  gCmdMgr.unregisterPlaylistCommandsMediaList :
283  gCmdMgr.unregisterPlaylistCommandsMediaItem);
284  var guid = (aInstruction.typeOrGUID == "guid" ? aInstruction.registerString :
285  "");
286 
287  var type = (aInstruction.typeOrGUID == "type" ? aInstruction.registerString :
288  "");
289 
290  unregisterFunction(guid, type, aInstruction.command);
291  }
292 }
293 
294 /* An entry point into our instruction checking.
295  * We expect to find our command as this instruction was just performed so
296  * send aExpectedToFind = true for either assert function.
297  */
298 function assertInstruction(aInstruction) {
299  if (aInstruction.mediaListOrLibrary == "library") {
300  assertLibraryInstruction(aInstruction, true);
301  }
302  else {
303  assertMediaListInstruction(aInstruction, true);
304  }
305 }
306 
307 /* An entry point into our instruction reversal checking.
308  * We don't expect to find our command as this instruction was just reversed so
309  * send aExpectedToFind = false for either assert function.
310  */
311 function assertReversedInstruction(aInstruction) {
312  if (aInstruction.mediaListOrLibrary == "library") {
313  assertLibraryInstruction(aInstruction, false);
314  }
315  else {
316  assertMediaListInstruction(aInstruction, false);
317  }
318 }
319 
320 /* This method verifies that an instruction that targetted the library was
321  * performed or reversed successfully.
322  *
323  * If the insruction was just performed, this method should be called with
324  * aExpectedToFind as true. However, if an instruction was just reversed
325  * aExpectedToFind should be false as it should have been removed.
326  *
327  * When this test is called with aExpectedToFind as true it confirms
328  * that the instruction's command was added to all of the existing
329  * medialists of the library, then that it gets added to any new medialist,
330  * and finally that it gets removed from medialists when they are removed.
331  *
332  * When aExpectedToFind is false, it performs the same analysis but throws if
333  * it does find the command anywhere.
334  */
335 function assertLibraryInstruction(aInstruction, aExpectedToFind) {
336  // We need to use a different get method if we are targetting the servicepane
337  var getFunction = (aInstruction.servicePaneOrMenu == "servicepane" ?
338  gCmdMgr.getPlaylistCommandsMediaList :
339  gCmdMgr.getPlaylistCommandsMediaItem);
340 
341  // First we check if our command was added to all existing
342  // medialists in the library, including the library's medialist itself
343  var mediaLists = gTestLibrary.getItemsByProperty(SBProperties.isList, "1");
344  mediaLists = mediaLists.QueryInterface(Ci.nsIMutableArray);
345  mediaLists.appendElement(gTestLibrary, false);
346 
347  var len = mediaLists.length
348  for (var i = 0; i < len; i++) {
349  var guid = mediaLists.queryElementAt(i, Ci.sbIMediaList).guid;
350  var rootCommand = getFunction(guid, "");
351 
352  var found = (rootCommand != null) &&
353  isSubCommandPresent(rootCommand, aInstruction.command);
354  assertTrue(aExpectedToFind == found,
355  "We expected this instruction on an existing list to " +
356  (aExpectedToFind ? "work and it didn't" : "be reversed and it wasn't") +
357  ":\n " + aInstruction.toString());
358  }
359 
360  // Next we create a new list and see if our command was added to it
361  var newList = gTestLibrary.createMediaList("simple");
362  var newListRootCommand = getFunction(newList.guid, "");
363 
364  found = (newListRootCommand != null) &&
365  isSubCommandPresent(newListRootCommand, aInstruction.command);
366  assertTrue(aExpectedToFind == found,
367  "We expected this instruction on an added list to " +
368  (aExpectedToFind ? "work and it didn't" : "be reversed and it wasn't") +
369  ":\n " + aInstruction.toString());
370 
371  // Lastly we remove the list and see if our command is present
372  gTestLibrary.remove(newList);
373  newListRootCommand = getFunction(newList.guid, "");
374  found = (newListRootCommand != null) &&
375  isSubCommandPresent(newListRootCommand, aInstruction.command);
376 
377  assertFalse(found,
378  "Instruction's command present on added and removed list: " +
379  aInstruction.toString());
380 }
381 
382 /* This method verifies that an instruction that targetted a medialist or
383  * medialist type was performed or reversed successfully.
384  *
385  * If the insruction was just performed, this method should be called with
386  * aExpectedToFind as true. However, if an instruction was just reversed
387  * aExpectedToFind should be false as it should have been removed.
388  *
389  * When this test is called with aExpectedToFind as true it confirms
390  * that the instruction's command was added to all of the existing
391  * medialists of the library, then that it gets added to any new medialist,
392  * and finally that it gets removed from medialists when they are removed.
393  *
394  * When aExpectedToFind is false, it performs the same analysis but throws if
395  * it does find the command anywhere.
396  */
397 function assertMediaListInstruction(aInstruction, aExpectedToFind) {
398  // We need to use a different get method if we are targetting the servicepane
399  var getFunction = (aInstruction.servicePaneOrMenu == "servicepane" ?
400  gCmdMgr.getPlaylistCommandsMediaList :
401  gCmdMgr.getPlaylistCommandsMediaItem);
402 
403  // Retrieve the root command that we are interested in
404  var guid = (aInstruction.typeOrGUID == "guid" ? aInstruction.registerString :
405  "");
406 
407  var type = (aInstruction.typeOrGUID == "type" ? aInstruction.registerString :
408  "");
409 
410  var primaryRootCommand = getFunction(guid, type);
411 
412  // Search the root command for our instruction's command
413  var found = (primaryRootCommand != null) &&
414  isSubCommandPresent(primaryRootCommand, aInstruction.command);
415  assertTrue(aExpectedToFind == found,
416  "We expected this instruction to " +
417  (aExpectedToFind ? "work and it didn't" : "be reversed and it wasn't") +
418  ":\n " + aInstruction.toString());
419 
420  // Next we look for our command somewhere that it should not be.
421  var secondaryRootCommand;
422  if (guid.length > 0) {
423  // if we registered the command to guid, ensure that it isn't present for
424  // a command with the same type, but a different guid
425  secondaryRootCommand = getFunction(gSecondarySimpleList.guid, "");
426  }
427  else {
428  // if we registered the command to type, ensure that it isn't present for
429  // a command with a different type
430  secondaryRootCommand = getFunction("", gSmartList.type);
431  }
432 
433  found = (secondaryRootCommand != null) &&
434  isSubCommandPresent(secondaryRootCommand, aInstruction.command);
435 
436  // We would never expect the command to be found here so aExpectedtoFind
437  // is irrelevant.
438  assertFalse(found,
439  "Instruction affected a type or guid it shouldn't have: " +
440  aInstruction.toString());
441 }
442 
443 /* A helper method to determine if any of the direct children of
444  * aRootCommand is aSubCommand.
445  */
446 function isSubCommandPresent(aRootCommand, aSubCommand) {
447  for (var i = 0; i < aRootCommand.getNumCommands("",""); i++) {
448  var currCommand = aRootCommand.getCommandSubObject("", i, "");
449  if (currCommand == aSubCommand) {
450  return true;
451  }
452  }
453  return false;
454 }
var gSecondarySimpleList
const PlaylistCommandsBuilder
const Cc
function log(s)
const REGISTRATION_PARAM_COMBOS
const NUMRECURSE
Playlist Commands Unit Test File.
function assertTrue(aTest, aMessage)
const TESTLENGTH
function makeTriggerCallback(aActionID)
var libraryManager
var gSmartList
var gCmdMgr
function testSubmenus(aCommand, aMenuID, aTestLength, aNumSubCommands)
function testAppendInsertRemove(aCommand, aMenuID, aTestLength, aNumSubCommands)
var gPrimarySimpleList
return null
Definition: FeedWriter.js:1143
function createLibrary(databaseGuid, databaseLocation)
Definition: test_load.js:151
var gTestPrefix
function assertFalse(aTest, aMessage)
function _log(aMsg, aMenuID)
function testRegistration()
function testCommandCallbacksAndShortcuts(aCommand, aMenuID, aTestLength, aNumSubCommands)
function RegistrationInstruction(aMediaListOrLibrary, aTypeOrGUID, aServicePaneOrMenu, aRegisterString, aCommand)
var gTestLibrary
const Ci
function runTest()
_getSelectedPageStyle s i