2019-01-13 13:01:16 -07:00
|
|
|
using System;
|
2019-01-13 12:24:58 -07:00
|
|
|
using System.Collections.Generic;
|
|
|
|
using System.Linq;
|
|
|
|
using MediaBrowser.Controller.Dto;
|
2015-01-24 12:03:55 -07:00
|
|
|
using MediaBrowser.Controller.Entities;
|
2013-05-12 08:27:56 -07:00
|
|
|
using MediaBrowser.Controller.Entities.Audio;
|
2013-03-15 22:52:33 -07:00
|
|
|
using MediaBrowser.Controller.Library;
|
2013-12-07 08:52:38 -07:00
|
|
|
using MediaBrowser.Controller.Net;
|
2014-03-25 14:13:55 -07:00
|
|
|
using MediaBrowser.Controller.Session;
|
2015-01-24 12:03:55 -07:00
|
|
|
using MediaBrowser.Model.Entities;
|
2016-10-25 12:02:04 -07:00
|
|
|
using MediaBrowser.Model.Services;
|
2019-02-08 14:59:28 -07:00
|
|
|
using MediaBrowser.Model.Querying;
|
2019-01-13 12:24:58 -07:00
|
|
|
using Microsoft.Extensions.Logging;
|
2013-03-15 22:52:33 -07:00
|
|
|
|
|
|
|
namespace MediaBrowser.Api
|
|
|
|
{
|
|
|
|
/// <summary>
|
|
|
|
/// Class BaseApiService
|
|
|
|
/// </summary>
|
2016-11-10 07:41:24 -07:00
|
|
|
public class BaseApiService : IService, IRequiresRequest
|
2013-03-15 22:52:33 -07:00
|
|
|
{
|
2013-03-23 19:45:00 -07:00
|
|
|
/// <summary>
|
|
|
|
/// Gets or sets the logger.
|
|
|
|
/// </summary>
|
|
|
|
/// <value>The logger.</value>
|
2019-01-13 13:31:14 -07:00
|
|
|
public ILogger Logger => ApiEntryPoint.Instance.Logger;
|
2015-01-22 09:41:34 -07:00
|
|
|
|
2013-03-23 19:45:00 -07:00
|
|
|
/// <summary>
|
|
|
|
/// Gets or sets the HTTP result factory.
|
|
|
|
/// </summary>
|
|
|
|
/// <value>The HTTP result factory.</value>
|
2019-01-13 13:31:14 -07:00
|
|
|
public IHttpResultFactory ResultFactory => ApiEntryPoint.Instance.ResultFactory;
|
2013-03-23 19:45:00 -07:00
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Gets or sets the request context.
|
|
|
|
/// </summary>
|
|
|
|
/// <value>The request context.</value>
|
2013-12-07 08:52:38 -07:00
|
|
|
public IRequest Request { get; set; }
|
|
|
|
|
|
|
|
public string GetHeader(string name)
|
|
|
|
{
|
|
|
|
return Request.Headers[name];
|
|
|
|
}
|
2013-05-12 08:27:56 -07:00
|
|
|
|
2017-08-19 12:43:35 -07:00
|
|
|
public static string[] SplitValue(string value, char delim)
|
|
|
|
{
|
2019-02-02 15:55:47 -07:00
|
|
|
if (value == null)
|
2017-08-19 12:43:35 -07:00
|
|
|
{
|
2018-12-27 14:43:48 -07:00
|
|
|
return Array.Empty<string>();
|
2017-08-19 12:43:35 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
return value.Split(new[] { delim }, StringSplitOptions.RemoveEmptyEntries);
|
|
|
|
}
|
2018-09-12 10:26:21 -07:00
|
|
|
|
|
|
|
public static Guid[] GetGuids(string value)
|
|
|
|
{
|
2019-02-02 15:55:47 -07:00
|
|
|
if (value == null)
|
|
|
|
{
|
|
|
|
return Array.Empty<Guid>();
|
|
|
|
}
|
|
|
|
|
2019-02-02 16:15:28 -07:00
|
|
|
return value.Split(new[] { ',' }, StringSplitOptions.RemoveEmptyEntries)
|
2019-02-02 15:55:47 -07:00
|
|
|
.Select(i => new Guid(i))
|
|
|
|
.ToArray();
|
2018-09-12 10:26:21 -07:00
|
|
|
}
|
|
|
|
|
2013-03-23 19:45:00 -07:00
|
|
|
/// <summary>
|
|
|
|
/// To the optimized result.
|
|
|
|
/// </summary>
|
|
|
|
/// <typeparam name="T"></typeparam>
|
|
|
|
/// <param name="result">The result.</param>
|
|
|
|
/// <returns>System.Object.</returns>
|
|
|
|
protected object ToOptimizedResult<T>(T result)
|
|
|
|
where T : class
|
|
|
|
{
|
2018-09-12 10:26:21 -07:00
|
|
|
return ResultFactory.GetResult(Request, result);
|
2013-03-23 19:45:00 -07:00
|
|
|
}
|
|
|
|
|
2018-09-12 10:26:21 -07:00
|
|
|
protected void AssertCanUpdateUser(IAuthorizationContext authContext, IUserManager userManager, Guid userId, bool restrictUserPreferences)
|
2015-03-15 18:48:25 -07:00
|
|
|
{
|
2016-11-10 07:41:24 -07:00
|
|
|
var auth = authContext.GetAuthorizationInfo(Request);
|
2015-03-15 18:48:25 -07:00
|
|
|
|
2018-09-12 10:26:21 -07:00
|
|
|
var authenticatedUser = auth.User;
|
2015-03-15 18:48:25 -07:00
|
|
|
|
|
|
|
// If they're going to update the record of another user, they must be an administrator
|
2018-09-12 10:26:21 -07:00
|
|
|
if (!userId.Equals(auth.UserId))
|
2015-03-15 18:48:25 -07:00
|
|
|
{
|
|
|
|
if (!authenticatedUser.Policy.IsAdministrator)
|
|
|
|
{
|
|
|
|
throw new SecurityException("Unauthorized access.");
|
|
|
|
}
|
|
|
|
}
|
2017-04-21 13:03:07 -07:00
|
|
|
else if (restrictUserPreferences)
|
2015-03-15 18:48:25 -07:00
|
|
|
{
|
|
|
|
if (!authenticatedUser.Policy.EnableUserPreferenceAccess)
|
|
|
|
{
|
|
|
|
throw new SecurityException("Unauthorized access.");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2016-05-05 21:50:39 -07:00
|
|
|
|
2014-03-25 14:13:55 -07:00
|
|
|
/// <summary>
|
|
|
|
/// Gets the session.
|
|
|
|
/// </summary>
|
|
|
|
/// <returns>SessionInfo.</returns>
|
2018-09-12 10:26:21 -07:00
|
|
|
protected SessionInfo GetSession(ISessionContext sessionContext)
|
2014-03-25 14:13:55 -07:00
|
|
|
{
|
2018-09-12 10:26:21 -07:00
|
|
|
var session = sessionContext.GetSession(Request);
|
2014-06-25 08:12:39 -07:00
|
|
|
|
|
|
|
if (session == null)
|
|
|
|
{
|
|
|
|
throw new ArgumentException("Session not found.");
|
|
|
|
}
|
|
|
|
|
|
|
|
return session;
|
2014-03-25 14:13:55 -07:00
|
|
|
}
|
|
|
|
|
2016-11-10 07:41:24 -07:00
|
|
|
protected DtoOptions GetDtoOptions(IAuthorizationContext authContext, object request)
|
2015-01-24 12:03:55 -07:00
|
|
|
{
|
|
|
|
var options = new DtoOptions();
|
|
|
|
|
2019-02-08 14:59:28 -07:00
|
|
|
if (request is IHasItemFields hasFields)
|
2015-01-24 12:03:55 -07:00
|
|
|
{
|
2017-08-19 12:43:35 -07:00
|
|
|
options.Fields = hasFields.GetItemFields();
|
2015-01-24 12:03:55 -07:00
|
|
|
}
|
|
|
|
|
2019-02-02 15:55:47 -07:00
|
|
|
if (!options.ContainsField(Model.Querying.ItemFields.RecursiveItemCount)
|
|
|
|
|| !options.ContainsField(Model.Querying.ItemFields.ChildCount))
|
2016-12-13 00:36:30 -07:00
|
|
|
{
|
2018-09-12 10:26:21 -07:00
|
|
|
var client = authContext.GetAuthorizationInfo(Request).Client ?? string.Empty;
|
|
|
|
if (client.IndexOf("kodi", StringComparison.OrdinalIgnoreCase) != -1 ||
|
|
|
|
client.IndexOf("wmc", StringComparison.OrdinalIgnoreCase) != -1 ||
|
|
|
|
client.IndexOf("media center", StringComparison.OrdinalIgnoreCase) != -1 ||
|
|
|
|
client.IndexOf("classic", StringComparison.OrdinalIgnoreCase) != -1)
|
|
|
|
{
|
2019-02-08 14:59:28 -07:00
|
|
|
int oldLen = options.Fields.Length;
|
|
|
|
var arr = new ItemFields[oldLen + 1];
|
|
|
|
options.Fields.CopyTo(arr, 0);
|
|
|
|
arr[oldLen] = Model.Querying.ItemFields.RecursiveItemCount;
|
|
|
|
options.Fields = arr;
|
2018-09-12 10:26:21 -07:00
|
|
|
}
|
2016-12-13 00:36:30 -07:00
|
|
|
|
2018-09-12 10:26:21 -07:00
|
|
|
if (client.IndexOf("kodi", StringComparison.OrdinalIgnoreCase) != -1 ||
|
|
|
|
client.IndexOf("wmc", StringComparison.OrdinalIgnoreCase) != -1 ||
|
|
|
|
client.IndexOf("media center", StringComparison.OrdinalIgnoreCase) != -1 ||
|
|
|
|
client.IndexOf("classic", StringComparison.OrdinalIgnoreCase) != -1 ||
|
|
|
|
client.IndexOf("roku", StringComparison.OrdinalIgnoreCase) != -1 ||
|
|
|
|
client.IndexOf("samsung", StringComparison.OrdinalIgnoreCase) != -1 ||
|
|
|
|
client.IndexOf("androidtv", StringComparison.OrdinalIgnoreCase) != -1)
|
|
|
|
{
|
2019-02-08 14:59:28 -07:00
|
|
|
|
|
|
|
int oldLen = options.Fields.Length;
|
|
|
|
var arr = new ItemFields[oldLen + 1];
|
|
|
|
options.Fields.CopyTo(arr, 0);
|
|
|
|
arr[oldLen] = Model.Querying.ItemFields.ChildCount;
|
|
|
|
options.Fields = arr;
|
2018-09-12 10:26:21 -07:00
|
|
|
}
|
2016-12-13 00:36:30 -07:00
|
|
|
}
|
|
|
|
|
2019-02-02 15:55:47 -07:00
|
|
|
if (request is IHasDtoOptions hasDtoOptions)
|
2015-01-24 12:03:55 -07:00
|
|
|
{
|
|
|
|
options.EnableImages = hasDtoOptions.EnableImages ?? true;
|
|
|
|
|
|
|
|
if (hasDtoOptions.ImageTypeLimit.HasValue)
|
|
|
|
{
|
|
|
|
options.ImageTypeLimit = hasDtoOptions.ImageTypeLimit.Value;
|
|
|
|
}
|
2019-03-13 14:32:52 -07:00
|
|
|
|
2016-08-17 12:28:43 -07:00
|
|
|
if (hasDtoOptions.EnableUserData.HasValue)
|
|
|
|
{
|
|
|
|
options.EnableUserData = hasDtoOptions.EnableUserData.Value;
|
|
|
|
}
|
2015-01-24 12:03:55 -07:00
|
|
|
|
|
|
|
if (!string.IsNullOrWhiteSpace(hasDtoOptions.EnableImageTypes))
|
|
|
|
{
|
2019-02-26 12:47:23 -07:00
|
|
|
options.ImageTypes = hasDtoOptions.EnableImageTypes.Split(new[] { ',' }, StringSplitOptions.RemoveEmptyEntries)
|
|
|
|
.Select(v => (ImageType)Enum.Parse(typeof(ImageType), v, true))
|
|
|
|
.ToArray();
|
2015-01-24 12:03:55 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return options;
|
|
|
|
}
|
|
|
|
|
2017-05-21 21:54:02 -07:00
|
|
|
protected MusicArtist GetArtist(string name, ILibraryManager libraryManager, DtoOptions dtoOptions)
|
2013-05-12 08:27:56 -07:00
|
|
|
{
|
2016-05-05 21:50:39 -07:00
|
|
|
if (name.IndexOf(BaseItem.SlugChar) != -1)
|
|
|
|
{
|
2017-06-12 23:33:29 -07:00
|
|
|
var result = GetItemFromSlugName<MusicArtist>(libraryManager, name, dtoOptions);
|
2013-06-10 20:31:00 -07:00
|
|
|
|
2016-05-05 21:50:39 -07:00
|
|
|
if (result != null)
|
|
|
|
{
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
}
|
2013-09-16 19:08:18 -07:00
|
|
|
|
2017-05-21 21:54:02 -07:00
|
|
|
return libraryManager.GetArtist(name, dtoOptions);
|
2013-05-12 08:27:56 -07:00
|
|
|
}
|
|
|
|
|
2017-05-21 21:54:02 -07:00
|
|
|
protected Studio GetStudio(string name, ILibraryManager libraryManager, DtoOptions dtoOptions)
|
2013-05-12 08:27:56 -07:00
|
|
|
{
|
2016-05-05 21:50:39 -07:00
|
|
|
if (name.IndexOf(BaseItem.SlugChar) != -1)
|
2016-03-24 21:08:22 -07:00
|
|
|
{
|
2017-06-12 23:33:29 -07:00
|
|
|
var result = GetItemFromSlugName<Studio>(libraryManager, name, dtoOptions);
|
2016-05-05 21:50:39 -07:00
|
|
|
|
|
|
|
if (result != null)
|
|
|
|
{
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
}
|
2013-05-12 08:27:56 -07:00
|
|
|
|
2016-05-05 21:50:39 -07:00
|
|
|
return libraryManager.GetStudio(name);
|
2013-05-12 08:27:56 -07:00
|
|
|
}
|
|
|
|
|
2017-05-21 21:54:02 -07:00
|
|
|
protected Genre GetGenre(string name, ILibraryManager libraryManager, DtoOptions dtoOptions)
|
2013-05-12 08:27:56 -07:00
|
|
|
{
|
2016-05-05 21:50:39 -07:00
|
|
|
if (name.IndexOf(BaseItem.SlugChar) != -1)
|
2013-05-12 08:27:56 -07:00
|
|
|
{
|
2017-06-12 23:33:29 -07:00
|
|
|
var result = GetItemFromSlugName<Genre>(libraryManager, name, dtoOptions);
|
2016-05-05 21:50:39 -07:00
|
|
|
|
|
|
|
if (result != null)
|
|
|
|
{
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
}
|
2013-05-12 08:27:56 -07:00
|
|
|
|
2016-05-05 21:50:39 -07:00
|
|
|
return libraryManager.GetGenre(name);
|
2013-05-12 08:27:56 -07:00
|
|
|
}
|
|
|
|
|
2017-05-21 21:54:02 -07:00
|
|
|
protected MusicGenre GetMusicGenre(string name, ILibraryManager libraryManager, DtoOptions dtoOptions)
|
2013-07-01 11:17:55 -07:00
|
|
|
{
|
2016-05-05 21:50:39 -07:00
|
|
|
if (name.IndexOf(BaseItem.SlugChar) != -1)
|
2013-07-01 11:17:55 -07:00
|
|
|
{
|
2017-06-12 23:33:29 -07:00
|
|
|
var result = GetItemFromSlugName<MusicGenre>(libraryManager, name, dtoOptions);
|
2016-03-24 21:08:22 -07:00
|
|
|
|
2016-05-05 21:50:39 -07:00
|
|
|
if (result != null)
|
2013-07-01 11:17:55 -07:00
|
|
|
{
|
2016-05-05 21:50:39 -07:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
}
|
2013-07-01 11:17:55 -07:00
|
|
|
|
2016-05-05 21:50:39 -07:00
|
|
|
return libraryManager.GetMusicGenre(name);
|
2013-07-01 11:17:55 -07:00
|
|
|
}
|
2013-09-16 19:08:18 -07:00
|
|
|
|
2017-05-21 21:54:02 -07:00
|
|
|
protected Person GetPerson(string name, ILibraryManager libraryManager, DtoOptions dtoOptions)
|
2013-05-12 08:27:56 -07:00
|
|
|
{
|
2016-05-05 21:50:39 -07:00
|
|
|
if (name.IndexOf(BaseItem.SlugChar) != -1)
|
2013-05-12 08:27:56 -07:00
|
|
|
{
|
2017-06-12 23:33:29 -07:00
|
|
|
var result = GetItemFromSlugName<Person>(libraryManager, name, dtoOptions);
|
2016-05-05 21:50:39 -07:00
|
|
|
|
|
|
|
if (result != null)
|
|
|
|
{
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
}
|
2013-05-12 08:27:56 -07:00
|
|
|
|
2016-05-05 21:50:39 -07:00
|
|
|
return libraryManager.GetPerson(name);
|
2013-05-12 08:27:56 -07:00
|
|
|
}
|
2013-08-03 07:38:56 -07:00
|
|
|
|
2017-06-12 23:33:29 -07:00
|
|
|
private T GetItemFromSlugName<T>(ILibraryManager libraryManager, string name, DtoOptions dtoOptions)
|
|
|
|
where T : BaseItem, new()
|
|
|
|
{
|
|
|
|
var result = libraryManager.GetItemList(new InternalItemsQuery
|
|
|
|
{
|
|
|
|
Name = name.Replace(BaseItem.SlugChar, '&'),
|
|
|
|
IncludeItemTypes = new[] { typeof(T).Name },
|
|
|
|
DtoOptions = dtoOptions
|
|
|
|
|
2018-09-12 10:26:21 -07:00
|
|
|
}).OfType<T>().FirstOrDefault();
|
2017-06-12 23:33:29 -07:00
|
|
|
|
|
|
|
if (result == null)
|
|
|
|
{
|
|
|
|
result = libraryManager.GetItemList(new InternalItemsQuery
|
|
|
|
{
|
|
|
|
Name = name.Replace(BaseItem.SlugChar, '/'),
|
|
|
|
IncludeItemTypes = new[] { typeof(T).Name },
|
|
|
|
DtoOptions = dtoOptions
|
|
|
|
|
2018-09-12 10:26:21 -07:00
|
|
|
}).OfType<T>().FirstOrDefault();
|
2017-06-12 23:33:29 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
if (result == null)
|
|
|
|
{
|
|
|
|
result = libraryManager.GetItemList(new InternalItemsQuery
|
|
|
|
{
|
|
|
|
Name = name.Replace(BaseItem.SlugChar, '?'),
|
|
|
|
IncludeItemTypes = new[] { typeof(T).Name },
|
|
|
|
DtoOptions = dtoOptions
|
|
|
|
|
2018-09-12 10:26:21 -07:00
|
|
|
}).OfType<T>().FirstOrDefault();
|
2017-06-12 23:33:29 -07:00
|
|
|
}
|
|
|
|
|
2018-09-12 10:26:21 -07:00
|
|
|
return result;
|
2017-06-12 23:33:29 -07:00
|
|
|
}
|
|
|
|
|
2015-01-22 09:41:34 -07:00
|
|
|
protected string GetPathValue(int index)
|
|
|
|
{
|
2016-10-22 07:51:48 -07:00
|
|
|
var pathInfo = Parse(Request.PathInfo);
|
|
|
|
var first = pathInfo[0];
|
2015-01-22 09:41:34 -07:00
|
|
|
|
|
|
|
// backwards compatibility
|
2015-06-25 14:50:56 -07:00
|
|
|
if (string.Equals(first, "mediabrowser", StringComparison.OrdinalIgnoreCase) ||
|
|
|
|
string.Equals(first, "emby", StringComparison.OrdinalIgnoreCase))
|
2015-01-22 09:41:34 -07:00
|
|
|
{
|
|
|
|
index++;
|
|
|
|
}
|
|
|
|
|
2016-10-22 07:51:48 -07:00
|
|
|
return pathInfo[index];
|
|
|
|
}
|
|
|
|
|
2019-03-13 14:32:52 -07:00
|
|
|
private static string[] Parse(string pathUri)
|
2016-10-22 07:51:48 -07:00
|
|
|
{
|
|
|
|
var actionParts = pathUri.Split(new[] { "://" }, StringSplitOptions.None);
|
|
|
|
|
|
|
|
var pathInfo = actionParts[actionParts.Length - 1];
|
|
|
|
|
|
|
|
var optionsPos = pathInfo.LastIndexOf('?');
|
|
|
|
if (optionsPos != -1)
|
|
|
|
{
|
|
|
|
pathInfo = pathInfo.Substring(0, optionsPos);
|
|
|
|
}
|
|
|
|
|
|
|
|
var args = pathInfo.Split('/');
|
|
|
|
|
2019-02-02 15:55:47 -07:00
|
|
|
return args.Skip(1).ToArray();
|
2015-01-22 09:41:34 -07:00
|
|
|
}
|
|
|
|
|
2013-08-03 07:38:56 -07:00
|
|
|
/// <summary>
|
|
|
|
/// Gets the name of the item by.
|
|
|
|
/// </summary>
|
2017-05-21 21:54:02 -07:00
|
|
|
protected BaseItem GetItemByName(string name, string type, ILibraryManager libraryManager, DtoOptions dtoOptions)
|
2013-08-03 07:38:56 -07:00
|
|
|
{
|
2019-03-13 14:32:52 -07:00
|
|
|
if (type.Equals("Person", StringComparison.OrdinalIgnoreCase))
|
2013-08-03 07:38:56 -07:00
|
|
|
{
|
2019-03-13 14:32:52 -07:00
|
|
|
return GetPerson(name, libraryManager, dtoOptions);
|
2013-08-03 07:38:56 -07:00
|
|
|
}
|
2019-03-13 14:32:52 -07:00
|
|
|
else if (type.Equals("Artist", StringComparison.OrdinalIgnoreCase))
|
2013-08-03 07:38:56 -07:00
|
|
|
{
|
2019-03-13 14:32:52 -07:00
|
|
|
return GetArtist(name, libraryManager, dtoOptions);
|
2013-08-03 07:38:56 -07:00
|
|
|
}
|
2019-03-13 14:32:52 -07:00
|
|
|
else if (type.Equals("Genre", StringComparison.OrdinalIgnoreCase))
|
2013-08-03 07:38:56 -07:00
|
|
|
{
|
2019-03-13 14:32:52 -07:00
|
|
|
return GetGenre(name, libraryManager, dtoOptions);
|
2013-08-03 07:38:56 -07:00
|
|
|
}
|
2019-03-13 14:32:52 -07:00
|
|
|
else if (type.Equals("MusicGenre", StringComparison.OrdinalIgnoreCase))
|
2013-08-03 07:38:56 -07:00
|
|
|
{
|
2019-03-13 14:32:52 -07:00
|
|
|
return GetMusicGenre(name, libraryManager, dtoOptions);
|
2013-08-03 07:38:56 -07:00
|
|
|
}
|
2019-03-13 14:32:52 -07:00
|
|
|
else if (type.Equals("Studio", StringComparison.OrdinalIgnoreCase))
|
2013-08-03 07:38:56 -07:00
|
|
|
{
|
2019-03-13 14:32:52 -07:00
|
|
|
return GetStudio(name, libraryManager, dtoOptions);
|
2013-08-03 07:38:56 -07:00
|
|
|
}
|
2019-03-13 14:32:52 -07:00
|
|
|
else if (type.Equals("Year", StringComparison.OrdinalIgnoreCase))
|
2013-08-03 07:38:56 -07:00
|
|
|
{
|
2019-03-13 14:32:52 -07:00
|
|
|
return libraryManager.GetYear(int.Parse(name));
|
2013-08-03 07:38:56 -07:00
|
|
|
}
|
|
|
|
|
2019-03-13 14:32:52 -07:00
|
|
|
throw new ArgumentException("Invalid type", nameof(type));
|
2013-08-03 07:38:56 -07:00
|
|
|
}
|
2013-03-15 22:52:33 -07:00
|
|
|
}
|
|
|
|
}
|