File: world.rb

package info (click to toggle)
mikutter 4.1.3%2Bdfsg1-1
  • links: PTS, VCS
  • area: main
  • in suites: bullseye
  • size: 9,260 kB
  • sloc: ruby: 20,126; sh: 183; makefile: 19
file content (121 lines) | stat: -rw-r--r-- 4,711 bytes parent folder | download
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
# -*- coding: utf-8 -*-

require_relative 'error'
require_relative 'keep'
require_relative 'model/lost_world'

require 'digest/sha1'

Plugin.create(:world) do
  # 登録されている全てのWorldを列挙する。
  # 次のような方法で、Enumeratorを取得できる。
  # ==== Example
  # collect(:worlds)
  defevent :worlds, prototype: [Pluggaloid::COLLECT]

  # 新たなアカウント _1_ を追加する
  defevent :world_create, prototype: [Diva::Model]

  # アカウント _1_ が変更された時に呼ばれる
  defevent :world_modify, prototype: [Diva::Model]

  # アカウント _1_ を削除する
  defevent :world_destroy, prototype: [Diva::Model]

  world_struct = Struct.new(:slug, :name, :proc)
  @world_slug_dict = {}         # world_slug(Symbol) => World URI(Diva::URI)

  defdsl :world_setting do |world_slug, world_name, &proc|
    filter_world_setting_list do |settings|
      [settings.merge(world_slug => world_struct.new(world_slug, world_name, proc))]
    end
  end

  def world_order
    at(:world_order) || []
  end

  def load_world
    Plugin::World::Keep.accounts.map { |id, serialized|
      provider = Diva::Model(serialized[:provider])
      if provider
        provider.new(serialized)
      else
        Miquire::Plugin.load(serialized[:provider])
        provider = Diva::Model(serialized[:provider])
        if provider
          provider.new(serialized)
        else
          activity :system, _('アカウント「%{world}」のためのプラグインが読み込めなかったため、このアカウントは現在利用できません。') % {world: id},
                   description: _('アカウント「%{world}」に必要な%{plugin}プラグインが見つからなかったため、このアカウントは一時的に利用できません。%{plugin}プラグインを意図的に消したのであれば、このアカウントの登録を解除してください。') % {plugin: serialized[:provider], world: id}
          Plugin::World::LostWorld.new(serialized)
        end
      end
    }.compact.freeze.tap(&method(:check_world_uri))
  end

  def worlds_sort(world_list)
    world_list.sort_by.with_index do |a, index|
      [world_order.find_index(world_order_hash(a)) || Float::INFINITY, index]
    end
  end

  def check_world_uri(new_worlds)
    new_worlds.each do |w|
      if @world_slug_dict.key?(w.slug)
        if @world_slug_dict[w.slug] != w.uri
          warn "The URI of World `#{w.slug}' is not defined. You must define a consistent URI for World Model. see: https://dev.mikutter.hachune.net/issues/1231"
        end
      else
        @world_slug_dict[w.slug] = w.uri
      end
    end
  end

  def world_order_hash(world)
    Digest::SHA1.hexdigest("#{world.slug}mikutter")
  end

  collection(:worlds) do |mutation|
    mutation.rewind { |_| worlds_sort(load_world) }

    on_world_create do |new|
      return if new.is_a?(Plugin::World::LostWorld)
      Plugin::World::Keep.account_register(new.slug, **new.to_hash, provider: new.class.slug)
      mutation.rewind { |_| worlds_sort(load_world) }
      Plugin.call(:world_after_created, new)
      Plugin.call(:service_registered, new) # 互換性のため
    rescue Plugin::World::AlreadyExistError
      description = {
        new_world: new.title,
        duplicated_world: collect(:worlds).find{|w| w.slug == new.slug }&.title,
        world_slug: new.slug }
      activity :system, _('既に登録されているアカウントと重複しているため、登録に失敗しました。'),
               description: _('登録しようとしたアカウント「%{new_world}」は、既に登録されている「%{duplicated_world}」と同じ識別子「%{world_slug}」を持っているため、登録に失敗しました。') % description
    end

    on_world_modify do |target|
      return if target.is_a?(Plugin::World::LostWorld)
      if Plugin::World::Keep.accounts.has_key?(target.slug.to_sym)
        Plugin::World::Keep.account_modify(target.slug, { **target.to_hash, provider: target.class.slug })
        mutation.rewind { |_| worlds_sort(load_world) }
      end
    end

    on_world_destroy do |target|
      Plugin::World::Keep.account_destroy(target.slug)
      mutation.rewind { |_| worlds_sort(load_world) }
      Plugin.call(:service_destroyed, target) # 互換性のため
    end

    # Worldのリストを、 _worlds_ の順番に並び替える。
    on_world_reorder do |new_order|
      store(:world_order, new_order.map(&method(:world_order_hash)))
      mutation.rewind do |worlds|
        worlds_sort(worlds).tap do |reordered|
          Plugin.call(:world_reordered, reordered)
        end
      end
    end
  end
end