commit
6959f6e13a
|
@ -16,7 +16,6 @@ using NzbDrone.Core.Tv.Events;
|
||||||
using NzbDrone.Core.Validation.Paths;
|
using NzbDrone.Core.Validation.Paths;
|
||||||
using NzbDrone.Core.DataAugmentation.Scene;
|
using NzbDrone.Core.DataAugmentation.Scene;
|
||||||
using NzbDrone.SignalR;
|
using NzbDrone.SignalR;
|
||||||
using Omu.ValueInjecter;
|
|
||||||
|
|
||||||
namespace NzbDrone.Api.Series
|
namespace NzbDrone.Api.Series
|
||||||
{
|
{
|
||||||
|
|
|
@ -1,15 +1,20 @@
|
||||||
using System;
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using NzbDrone.Api.Mapping;
|
using NzbDrone.Api.Mapping;
|
||||||
|
using NzbDrone.Core.Datastore.Events;
|
||||||
|
using NzbDrone.Core.Messaging.Events;
|
||||||
using NzbDrone.Core.Tags;
|
using NzbDrone.Core.Tags;
|
||||||
|
using NzbDrone.SignalR;
|
||||||
|
|
||||||
namespace NzbDrone.Api.Tags
|
namespace NzbDrone.Api.Tags
|
||||||
{
|
{
|
||||||
public class TagModule : NzbDroneRestModule<TagResource>
|
public class TagModule : NzbDroneRestModuleWithSignalR<TagResource, Tag>, IHandle<TagsUpdatedEvent>
|
||||||
{
|
{
|
||||||
private readonly ITagService _tagService;
|
private readonly ITagService _tagService;
|
||||||
|
|
||||||
public TagModule(ITagService tagService)
|
public TagModule(IBroadcastSignalRMessage signalRBroadcaster,
|
||||||
|
ITagService tagService)
|
||||||
|
: base(signalRBroadcaster)
|
||||||
{
|
{
|
||||||
_tagService = tagService;
|
_tagService = tagService;
|
||||||
|
|
||||||
|
@ -44,5 +49,10 @@ namespace NzbDrone.Api.Tags
|
||||||
{
|
{
|
||||||
_tagService.Delete(id);
|
_tagService.Delete(id);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void Handle(TagsUpdatedEvent message)
|
||||||
|
{
|
||||||
|
BroadcastResourceChange(ModelAction.Sync);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,148 @@
|
||||||
|
using System;
|
||||||
|
using System.Linq;
|
||||||
|
using FluentAssertions;
|
||||||
|
using NUnit.Framework;
|
||||||
|
using NzbDrone.Core.Jobs;
|
||||||
|
using NzbDrone.Core.Tags;
|
||||||
|
using NzbDrone.Core.Test.Framework;
|
||||||
|
using NzbDrone.Core.Tv;
|
||||||
|
|
||||||
|
namespace NzbDrone.Core.Test.Datastore.Migration
|
||||||
|
{
|
||||||
|
[TestFixture]
|
||||||
|
public class dedupe_tags : MigrationTest<Core.Datastore.Migration.dedupe_tags>
|
||||||
|
{
|
||||||
|
[Test]
|
||||||
|
public void should_not_fail_if_series_tags_are_null()
|
||||||
|
{
|
||||||
|
WithTestDb(c =>
|
||||||
|
{
|
||||||
|
c.Insert.IntoTable("Series").Row(new
|
||||||
|
{
|
||||||
|
Tvdbid = 1,
|
||||||
|
TvRageId = 1,
|
||||||
|
Title = "Title1",
|
||||||
|
CleanTitle = "CleanTitle1",
|
||||||
|
Status = 1,
|
||||||
|
Images = "",
|
||||||
|
Path = "c:\\test",
|
||||||
|
Monitored = 1,
|
||||||
|
SeasonFolder = 1,
|
||||||
|
Runtime = 0,
|
||||||
|
SeriesType = 0,
|
||||||
|
UseSceneNumbering = 0,
|
||||||
|
LastInfoSync = "2000-01-01 00:00:00"
|
||||||
|
});
|
||||||
|
|
||||||
|
c.Insert.IntoTable("Tags").Row(new
|
||||||
|
{
|
||||||
|
Label = "test"
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
Mocker.Resolve<TagRepository>().All().Should().HaveCount(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void should_not_fail_if_series_tags_are_empty()
|
||||||
|
{
|
||||||
|
WithTestDb(c =>
|
||||||
|
{
|
||||||
|
c.Insert.IntoTable("Series").Row(new
|
||||||
|
{
|
||||||
|
Tvdbid = 1,
|
||||||
|
TvRageId = 1,
|
||||||
|
Title = "Title1",
|
||||||
|
CleanTitle = "CleanTitle1",
|
||||||
|
Status = 1,
|
||||||
|
Images = "",
|
||||||
|
Path = "c:\\test",
|
||||||
|
Monitored = 1,
|
||||||
|
SeasonFolder = 1,
|
||||||
|
Runtime = 0,
|
||||||
|
SeriesType = 0,
|
||||||
|
UseSceneNumbering = 0,
|
||||||
|
LastInfoSync = "2000-01-01 00:00:00",
|
||||||
|
Tags = "[]"
|
||||||
|
});
|
||||||
|
|
||||||
|
c.Insert.IntoTable("Tags").Row(new
|
||||||
|
{
|
||||||
|
Label = "test"
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
Mocker.Resolve<TagRepository>().All().Should().HaveCount(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void should_remove_duplicate_labels_from_tags()
|
||||||
|
{
|
||||||
|
WithTestDb(c =>
|
||||||
|
{
|
||||||
|
c.Insert.IntoTable("Tags").Row(new
|
||||||
|
{
|
||||||
|
Label = "test"
|
||||||
|
});
|
||||||
|
|
||||||
|
c.Insert.IntoTable("Tags").Row(new
|
||||||
|
{
|
||||||
|
Label = "test"
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
Mocker.Resolve<TagRepository>().All().Should().HaveCount(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void should_not_allow_duplicate_tag_to_be_inserted()
|
||||||
|
{
|
||||||
|
WithTestDb(c =>
|
||||||
|
{
|
||||||
|
c.Insert.IntoTable("Tags").Row(new
|
||||||
|
{
|
||||||
|
Label = "test"
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
Assert.That(() => Mocker.Resolve<TagRepository>().Insert(new Tag { Label = "test" }), Throws.Exception);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void should_replace_duplicated_tag_with_proper_tag()
|
||||||
|
{
|
||||||
|
WithTestDb(c =>
|
||||||
|
{
|
||||||
|
c.Insert.IntoTable("Series").Row(new
|
||||||
|
{
|
||||||
|
Tvdbid = 1,
|
||||||
|
TvRageId = 1,
|
||||||
|
Title = "Title1",
|
||||||
|
CleanTitle = "CleanTitle1",
|
||||||
|
Status = 1,
|
||||||
|
Images = "",
|
||||||
|
Path = "c:\\test",
|
||||||
|
Monitored = 1,
|
||||||
|
SeasonFolder = 1,
|
||||||
|
Runtime = 0,
|
||||||
|
SeriesType = 0,
|
||||||
|
UseSceneNumbering = 0,
|
||||||
|
LastInfoSync = "2000-01-01 00:00:00",
|
||||||
|
Tags = "[2]"
|
||||||
|
});
|
||||||
|
|
||||||
|
c.Insert.IntoTable("Tags").Row(new
|
||||||
|
{
|
||||||
|
Label = "test"
|
||||||
|
});
|
||||||
|
|
||||||
|
c.Insert.IntoTable("Tags").Row(new
|
||||||
|
{
|
||||||
|
Label = "test"
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
Mocker.Resolve<SeriesRepository>().Get(1).Tags.First().Should().Be(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -121,6 +121,7 @@
|
||||||
<Compile Include="Datastore\Migration\074_disable_eztv.cs" />
|
<Compile Include="Datastore\Migration\074_disable_eztv.cs" />
|
||||||
<Compile Include="Datastore\Migration\072_history_grabIdFixture.cs" />
|
<Compile Include="Datastore\Migration\072_history_grabIdFixture.cs" />
|
||||||
<Compile Include="Datastore\Migration\070_delay_profileFixture.cs" />
|
<Compile Include="Datastore\Migration\070_delay_profileFixture.cs" />
|
||||||
|
<Compile Include="Datastore\Migration\079_dedupe_tagsFixture.cs" />
|
||||||
<Compile Include="Datastore\Migration\075_force_lib_updateFixture.cs" />
|
<Compile Include="Datastore\Migration\075_force_lib_updateFixture.cs" />
|
||||||
<Compile Include="Datastore\ObjectDatabaseFixture.cs" />
|
<Compile Include="Datastore\ObjectDatabaseFixture.cs" />
|
||||||
<Compile Include="Datastore\PagingSpecExtensionsTests\PagingOffsetFixture.cs" />
|
<Compile Include="Datastore\PagingSpecExtensionsTests\PagingOffsetFixture.cs" />
|
||||||
|
|
|
@ -16,6 +16,9 @@ namespace NzbDrone.Core.Datastore.Migration
|
||||||
|
|
||||||
Alter.Table("Notifications")
|
Alter.Table("Notifications")
|
||||||
.AddColumn("Tags").AsString().Nullable();
|
.AddColumn("Tags").AsString().Nullable();
|
||||||
|
|
||||||
|
Execute.Sql("UPDATE Series SET Tags = '[]'");
|
||||||
|
Execute.Sql("UPDATE Notifications SET Tags = '[]'");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,157 @@
|
||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Data;
|
||||||
|
using System.Linq;
|
||||||
|
using FluentMigrator;
|
||||||
|
using NzbDrone.Common.Extensions;
|
||||||
|
using NzbDrone.Common.Serializer;
|
||||||
|
using NzbDrone.Core.Datastore.Migration.Framework;
|
||||||
|
|
||||||
|
namespace NzbDrone.Core.Datastore.Migration
|
||||||
|
{
|
||||||
|
[Migration(79)]
|
||||||
|
public class dedupe_tags : NzbDroneMigrationBase
|
||||||
|
{
|
||||||
|
protected override void MainDbUpgrade()
|
||||||
|
{
|
||||||
|
Execute.WithConnection(CleanupTags);
|
||||||
|
|
||||||
|
Alter.Table("Tags").AlterColumn("Label").AsString().Unique();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void CleanupTags(IDbConnection conn, IDbTransaction tran)
|
||||||
|
{
|
||||||
|
var tags = GetTags(conn, tran);
|
||||||
|
var grouped = tags.GroupBy(t => t.Label.ToLowerInvariant());
|
||||||
|
var replacements = new List<TagReplacement079>();
|
||||||
|
|
||||||
|
foreach (var group in grouped.Where(g => g.Count() > 1))
|
||||||
|
{
|
||||||
|
var first = group.First().Id;
|
||||||
|
|
||||||
|
foreach (var other in group.Skip(1).Select(t => t.Id))
|
||||||
|
{
|
||||||
|
replacements.Add(new TagReplacement079 { OldId = other, NewId = first });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
UpdateTaggedModel(conn, tran, "Series", replacements);
|
||||||
|
UpdateTaggedModel(conn, tran, "Notifications", replacements);
|
||||||
|
UpdateTaggedModel(conn, tran, "DelayProfiles", replacements);
|
||||||
|
UpdateTaggedModel(conn, tran, "Restrictions", replacements);
|
||||||
|
|
||||||
|
DeleteTags(conn, tran, replacements);
|
||||||
|
}
|
||||||
|
|
||||||
|
private List<Tag079> GetTags(IDbConnection conn, IDbTransaction tran)
|
||||||
|
{
|
||||||
|
var tags = new List<Tag079>();
|
||||||
|
|
||||||
|
using (IDbCommand tagCmd = conn.CreateCommand())
|
||||||
|
{
|
||||||
|
tagCmd.Transaction = tran;
|
||||||
|
tagCmd.CommandText = @"SELECT Id, Label FROM Tags";
|
||||||
|
|
||||||
|
using (IDataReader tagReader = tagCmd.ExecuteReader())
|
||||||
|
{
|
||||||
|
while (tagReader.Read())
|
||||||
|
{
|
||||||
|
var id = tagReader.GetInt32(0);
|
||||||
|
var label = tagReader.GetString(1);
|
||||||
|
|
||||||
|
tags.Add(new Tag079 { Id = id, Label = label });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return tags;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void UpdateTaggedModel(IDbConnection conn, IDbTransaction tran, string table, List<TagReplacement079> replacements)
|
||||||
|
{
|
||||||
|
var tagged = new List<TaggedModel079>();
|
||||||
|
|
||||||
|
using (IDbCommand tagCmd = conn.CreateCommand())
|
||||||
|
{
|
||||||
|
tagCmd.Transaction = tran;
|
||||||
|
tagCmd.CommandText = String.Format("SELECT Id, Tags FROM {0}", table);
|
||||||
|
|
||||||
|
using (IDataReader tagReader = tagCmd.ExecuteReader())
|
||||||
|
{
|
||||||
|
while (tagReader.Read())
|
||||||
|
{
|
||||||
|
if (!tagReader.IsDBNull(1))
|
||||||
|
{
|
||||||
|
var id = tagReader.GetInt32(0);
|
||||||
|
var tags = tagReader.GetString(1);
|
||||||
|
|
||||||
|
tagged.Add(new TaggedModel079
|
||||||
|
{
|
||||||
|
Id = id,
|
||||||
|
Tags = Json.Deserialize<HashSet<int>>(tags)
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var toUpdate = new List<TaggedModel079>();
|
||||||
|
|
||||||
|
foreach (var model in tagged)
|
||||||
|
{
|
||||||
|
foreach (var replacement in replacements)
|
||||||
|
{
|
||||||
|
if (model.Tags.Contains(replacement.OldId))
|
||||||
|
{
|
||||||
|
model.Tags.Remove(replacement.OldId);
|
||||||
|
model.Tags.Add(replacement.NewId);
|
||||||
|
|
||||||
|
toUpdate.Add(model);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach (var model in toUpdate.DistinctBy(m => m.Id))
|
||||||
|
{
|
||||||
|
using (IDbCommand updateCmd = conn.CreateCommand())
|
||||||
|
{
|
||||||
|
updateCmd.Transaction = tran;
|
||||||
|
updateCmd.CommandText = String.Format(@"UPDATE {0} SET Tags = ?", table);
|
||||||
|
updateCmd.AddParameter(model.Tags.ToJson());
|
||||||
|
|
||||||
|
updateCmd.ExecuteNonQuery();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void DeleteTags(IDbConnection conn, IDbTransaction tran, List<TagReplacement079> replacements)
|
||||||
|
{
|
||||||
|
var idsToRemove = replacements.Select(r => r.OldId).Distinct();
|
||||||
|
|
||||||
|
using (IDbCommand removeCmd = conn.CreateCommand())
|
||||||
|
{
|
||||||
|
removeCmd.Transaction = tran;
|
||||||
|
removeCmd.CommandText = String.Format("DELETE FROM Tags WHERE Id IN ({0})", String.Join(",", idsToRemove));
|
||||||
|
removeCmd.ExecuteNonQuery();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private class Tag079
|
||||||
|
{
|
||||||
|
public int Id { get; set; }
|
||||||
|
public string Label { get; set; }
|
||||||
|
}
|
||||||
|
|
||||||
|
private class TagReplacement079
|
||||||
|
{
|
||||||
|
public int OldId { get; set; }
|
||||||
|
public int NewId { get; set; }
|
||||||
|
}
|
||||||
|
|
||||||
|
private class TaggedModel079
|
||||||
|
{
|
||||||
|
public int Id { get; set; }
|
||||||
|
public HashSet<int> Tags { get; set; }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -246,6 +246,7 @@
|
||||||
<Compile Include="Datastore\Migration\074_disable_eztv.cs" />
|
<Compile Include="Datastore\Migration\074_disable_eztv.cs" />
|
||||||
<Compile Include="Datastore\Migration\073_clear_ratings.cs" />
|
<Compile Include="Datastore\Migration\073_clear_ratings.cs" />
|
||||||
<Compile Include="Datastore\Migration\077_add_add_options_to_series.cs" />
|
<Compile Include="Datastore\Migration\077_add_add_options_to_series.cs" />
|
||||||
|
<Compile Include="Datastore\Migration\079_dedupe_tags.cs" />
|
||||||
<Compile Include="Datastore\Migration\070_delay_profile.cs" />
|
<Compile Include="Datastore\Migration\070_delay_profile.cs" />
|
||||||
<Compile Include="Datastore\Migration\Framework\MigrationContext.cs" />
|
<Compile Include="Datastore\Migration\Framework\MigrationContext.cs" />
|
||||||
<Compile Include="Datastore\Migration\Framework\MigrationController.cs" />
|
<Compile Include="Datastore\Migration\Framework\MigrationController.cs" />
|
||||||
|
@ -809,6 +810,7 @@
|
||||||
<Compile Include="Tags\Tag.cs" />
|
<Compile Include="Tags\Tag.cs" />
|
||||||
<Compile Include="Tags\TagRepository.cs" />
|
<Compile Include="Tags\TagRepository.cs" />
|
||||||
<Compile Include="Tags\TagService.cs" />
|
<Compile Include="Tags\TagService.cs" />
|
||||||
|
<Compile Include="Tags\TagsUpdatedEvent.cs" />
|
||||||
<Compile Include="ThingiProvider\ConfigContractNotFoundException.cs" />
|
<Compile Include="ThingiProvider\ConfigContractNotFoundException.cs" />
|
||||||
<Compile Include="ThingiProvider\Events\ProviderUpdatedEvent.cs" />
|
<Compile Include="ThingiProvider\Events\ProviderUpdatedEvent.cs" />
|
||||||
<Compile Include="ThingiProvider\IProvider.cs" />
|
<Compile Include="ThingiProvider\IProvider.cs" />
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
using System;
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
|
using NzbDrone.Core.Messaging.Events;
|
||||||
|
|
||||||
namespace NzbDrone.Core.Tags
|
namespace NzbDrone.Core.Tags
|
||||||
{
|
{
|
||||||
|
@ -15,36 +16,51 @@ namespace NzbDrone.Core.Tags
|
||||||
|
|
||||||
public class TagService : ITagService
|
public class TagService : ITagService
|
||||||
{
|
{
|
||||||
private readonly ITagRepository _tagRepository;
|
private readonly ITagRepository _repo;
|
||||||
|
private readonly IEventAggregator _eventAggregator;
|
||||||
|
|
||||||
public TagService(ITagRepository tagRepository)
|
public TagService(ITagRepository repo, IEventAggregator eventAggregator)
|
||||||
{
|
{
|
||||||
_tagRepository = tagRepository;
|
_repo = repo;
|
||||||
|
_eventAggregator = eventAggregator;
|
||||||
}
|
}
|
||||||
|
|
||||||
public Tag GetTag(Int32 tagId)
|
public Tag GetTag(Int32 tagId)
|
||||||
{
|
{
|
||||||
return _tagRepository.Get(tagId);
|
return _repo.Get(tagId);
|
||||||
}
|
}
|
||||||
|
|
||||||
public List<Tag> All()
|
public List<Tag> All()
|
||||||
{
|
{
|
||||||
return _tagRepository.All().ToList();
|
return _repo.All().OrderBy(t => t.Label).ToList();
|
||||||
}
|
}
|
||||||
|
|
||||||
public Tag Add(Tag tag)
|
public Tag Add(Tag tag)
|
||||||
{
|
{
|
||||||
return _tagRepository.Insert(tag);
|
//TODO: check for duplicate tag by label and return that tag instead?
|
||||||
|
|
||||||
|
tag.Label = tag.Label.ToLowerInvariant();
|
||||||
|
|
||||||
|
_repo.Insert(tag);
|
||||||
|
_eventAggregator.PublishEvent(new TagsUpdatedEvent());
|
||||||
|
|
||||||
|
return tag;
|
||||||
}
|
}
|
||||||
|
|
||||||
public Tag Update(Tag tag)
|
public Tag Update(Tag tag)
|
||||||
{
|
{
|
||||||
return _tagRepository.Update(tag);
|
tag.Label = tag.Label.ToLowerInvariant();
|
||||||
|
|
||||||
|
_repo.Update(tag);
|
||||||
|
_eventAggregator.PublishEvent(new TagsUpdatedEvent());
|
||||||
|
|
||||||
|
return tag;
|
||||||
}
|
}
|
||||||
|
|
||||||
public void Delete(Int32 tagId)
|
public void Delete(Int32 tagId)
|
||||||
{
|
{
|
||||||
_tagRepository.Delete(tagId);
|
_repo.Delete(tagId);
|
||||||
|
_eventAggregator.PublishEvent(new TagsUpdatedEvent());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,8 @@
|
||||||
|
using NzbDrone.Common.Messaging;
|
||||||
|
|
||||||
|
namespace NzbDrone.Core.Tags
|
||||||
|
{
|
||||||
|
public class TagsUpdatedEvent : IEvent
|
||||||
|
{
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,11 +1,16 @@
|
||||||
var Backbone = require('backbone');
|
var Backbone = require('backbone');
|
||||||
var TagModel = require('./TagModel');
|
var TagModel = require('./TagModel');
|
||||||
var ApiData = require('../Shared/ApiData');
|
var ApiData = require('../Shared/ApiData');
|
||||||
|
|
||||||
module.exports = (function(){
|
require('../Mixins/backbone.signalr.mixin');
|
||||||
var Collection = Backbone.Collection.extend({
|
|
||||||
|
var collection = Backbone.Collection.extend({
|
||||||
url : window.NzbDrone.ApiRoot + '/tag',
|
url : window.NzbDrone.ApiRoot + '/tag',
|
||||||
model : TagModel
|
model : TagModel,
|
||||||
});
|
|
||||||
return new Collection(ApiData.get('tag'));
|
comparator : function(model){
|
||||||
}).call(this);
|
return model.get('label');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
module.exports = new collection(ApiData.get('tag')).bindSignalR();
|
||||||
|
|
Loading…
Reference in New Issue