2018-03-06 13:44:29 +01:00
|
|
|
|
class Procedure < ApplicationRecord
|
2017-04-13 14:48:18 +02:00
|
|
|
|
has_many :types_de_piece_justificative, -> { order "order_place ASC" }, dependent: :destroy
|
2018-02-13 15:38:26 +01:00
|
|
|
|
has_many :types_de_champ, -> { public_only }, dependent: :destroy
|
|
|
|
|
has_many :types_de_champ_private, -> { private_only }, class_name: 'TypeDeChamp', dependent: :destroy
|
2016-06-20 17:37:04 +02:00
|
|
|
|
has_many :dossiers
|
2018-05-30 11:36:48 +02:00
|
|
|
|
has_many :deleted_dossiers, dependent: :destroy
|
2017-03-07 10:25:28 +01:00
|
|
|
|
|
2016-01-18 16:20:51 +01:00
|
|
|
|
has_one :module_api_carto, dependent: :destroy
|
2017-06-08 14:16:48 +02:00
|
|
|
|
has_one :attestation_template, dependent: :destroy
|
2016-01-18 16:20:51 +01:00
|
|
|
|
|
|
|
|
|
belongs_to :administrateur
|
2018-04-24 15:23:07 +02:00
|
|
|
|
belongs_to :parent_procedure, class_name: 'Procedure'
|
2018-04-17 16:11:49 +02:00
|
|
|
|
belongs_to :service
|
2016-01-18 16:20:51 +01:00
|
|
|
|
|
2016-06-20 17:37:04 +02:00
|
|
|
|
has_many :assign_to, dependent: :destroy
|
2018-03-23 11:39:36 +01:00
|
|
|
|
has_many :administrateurs_procedures
|
|
|
|
|
has_many :administrateurs, through: :administrateurs_procedures
|
2016-05-20 15:39:17 +02:00
|
|
|
|
has_many :gestionnaires, through: :assign_to
|
|
|
|
|
|
2017-05-26 21:55:19 +02:00
|
|
|
|
has_one :initiated_mail, class_name: "Mails::InitiatedMail", dependent: :destroy
|
|
|
|
|
has_one :received_mail, class_name: "Mails::ReceivedMail", dependent: :destroy
|
|
|
|
|
has_one :closed_mail, class_name: "Mails::ClosedMail", dependent: :destroy
|
|
|
|
|
has_one :refused_mail, class_name: "Mails::RefusedMail", dependent: :destroy
|
|
|
|
|
has_one :without_continuation_mail, class_name: "Mails::WithoutContinuationMail", dependent: :destroy
|
|
|
|
|
|
2018-04-11 14:35:52 +02:00
|
|
|
|
has_one_attached :notice
|
2018-05-31 10:59:38 +02:00
|
|
|
|
has_one_attached :deliberation
|
2018-04-11 14:35:52 +02:00
|
|
|
|
|
2016-01-25 15:54:21 +01:00
|
|
|
|
delegate :use_api_carto, to: :module_api_carto
|
|
|
|
|
|
2016-08-31 16:07:11 +02:00
|
|
|
|
accepts_nested_attributes_for :types_de_champ, :reject_if => proc { |attributes| attributes['libelle'].blank? }, :allow_destroy => true
|
2015-11-20 13:54:08 +01:00
|
|
|
|
accepts_nested_attributes_for :types_de_piece_justificative, :reject_if => proc { |attributes| attributes['libelle'].blank? }, :allow_destroy => true
|
2015-12-08 10:11:58 +01:00
|
|
|
|
accepts_nested_attributes_for :module_api_carto
|
2016-08-03 18:19:56 +02:00
|
|
|
|
accepts_nested_attributes_for :types_de_champ_private
|
2015-11-10 10:23:15 +01:00
|
|
|
|
|
2015-12-10 17:13:39 +01:00
|
|
|
|
mount_uploader :logo, ProcedureLogoUploader
|
|
|
|
|
|
2017-06-27 14:22:43 +02:00
|
|
|
|
default_scope { where(hidden_at: nil) }
|
2018-05-16 17:21:12 +02:00
|
|
|
|
scope :brouillons, -> { where(aasm_state: :brouillon) }
|
|
|
|
|
scope :publiees, -> { where(aasm_state: :publiee) }
|
|
|
|
|
scope :archivees, -> { where(aasm_state: :archivee) }
|
|
|
|
|
scope :publiees_ou_archivees, -> { where(aasm_state: [:publiee, :archivee]) }
|
2017-07-17 15:06:36 +02:00
|
|
|
|
scope :by_libelle, -> { order(libelle: :asc) }
|
2018-04-12 18:36:09 +02:00
|
|
|
|
scope :created_during, -> (range) { where(created_at: range) }
|
|
|
|
|
scope :cloned_from_library, -> { where(cloned_from_library: true) }
|
2017-05-26 21:30:11 +02:00
|
|
|
|
|
2015-09-22 11:21:52 +02:00
|
|
|
|
validates :libelle, presence: true, allow_blank: false, allow_nil: false
|
|
|
|
|
validates :description, presence: true, allow_blank: false, allow_nil: false
|
2018-05-31 10:59:38 +02:00
|
|
|
|
validate :check_juridique
|
2015-11-17 15:30:03 +01:00
|
|
|
|
|
2018-05-17 15:38:49 +02:00
|
|
|
|
include AASM
|
|
|
|
|
|
|
|
|
|
aasm whiny_persistence: true do
|
|
|
|
|
state :brouillon, initial: true
|
|
|
|
|
state :publiee
|
|
|
|
|
state :archivee
|
|
|
|
|
state :hidden
|
|
|
|
|
|
|
|
|
|
event :publish, after: :after_publish, guard: :can_publish? do
|
|
|
|
|
transitions from: :brouillon, to: :publiee
|
|
|
|
|
transitions from: :archivee, to: :publiee
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
event :archive, after: :after_archive do
|
|
|
|
|
transitions from: :publiee, to: :archivee
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
event :hide, after: :after_hide do
|
|
|
|
|
transitions from: :brouillon, to: :hidden
|
|
|
|
|
transitions from: :publiee, to: :hidden
|
|
|
|
|
transitions from: :archivee, to: :hidden
|
|
|
|
|
end
|
|
|
|
|
end
|
2018-05-17 15:39:37 +02:00
|
|
|
|
|
|
|
|
|
def after_publish(path)
|
|
|
|
|
now = Time.now
|
|
|
|
|
update(
|
|
|
|
|
test_started_at: now,
|
|
|
|
|
archived_at: nil,
|
|
|
|
|
published_at: now
|
|
|
|
|
)
|
|
|
|
|
procedure_path = ProcedurePath.find_by(path: path)
|
|
|
|
|
|
|
|
|
|
if procedure_path.present?
|
2018-05-29 11:44:44 +02:00
|
|
|
|
procedure_path.publish!(self)
|
2018-05-17 15:39:37 +02:00
|
|
|
|
else
|
|
|
|
|
ProcedurePath.create(procedure: self, administrateur: administrateur, path: path)
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
def after_archive
|
|
|
|
|
update(archived_at: Time.now)
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
def after_hide
|
|
|
|
|
now = Time.now
|
|
|
|
|
update(hidden_at: now)
|
|
|
|
|
procedure_path&.hide!(self)
|
|
|
|
|
dossiers.update_all(hidden_at: now)
|
|
|
|
|
end
|
|
|
|
|
|
2018-05-17 15:41:44 +02:00
|
|
|
|
def locked?
|
|
|
|
|
publiee_ou_archivee?
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
def publiee_ou_archivee?
|
|
|
|
|
publiee? || archivee?
|
|
|
|
|
end
|
|
|
|
|
|
2018-05-17 15:39:37 +02:00
|
|
|
|
def can_publish?(path)
|
|
|
|
|
procedure_path = ProcedurePath.find_by(path: path)
|
|
|
|
|
if procedure_path.present?
|
|
|
|
|
administrateur.owns?(procedure_path)
|
|
|
|
|
else
|
|
|
|
|
true
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
2018-04-13 16:11:37 +02:00
|
|
|
|
# Warning: dossier after_save build_default_champs must be removed
|
|
|
|
|
# to save a dossier created from this method
|
|
|
|
|
def new_dossier
|
|
|
|
|
champs = types_de_champ
|
|
|
|
|
.ordered
|
|
|
|
|
.map { |tdc| tdc.champ.build }
|
|
|
|
|
|
|
|
|
|
champs_private = types_de_champ_private
|
|
|
|
|
.ordered
|
|
|
|
|
.map { |tdc| tdc.champ.build }
|
|
|
|
|
|
|
|
|
|
Dossier.new(procedure: self, champs: champs, champs_private: champs_private)
|
|
|
|
|
end
|
|
|
|
|
|
2018-05-17 15:32:36 +02:00
|
|
|
|
def procedure_path
|
|
|
|
|
ProcedurePath.find_with_procedure(self)
|
|
|
|
|
end
|
|
|
|
|
|
2016-06-24 16:41:44 +02:00
|
|
|
|
def path
|
2018-01-11 19:04:39 +01:00
|
|
|
|
procedure_path.path if procedure_path.present?
|
2016-06-24 16:41:44 +02:00
|
|
|
|
end
|
|
|
|
|
|
2016-06-30 10:24:01 +02:00
|
|
|
|
def default_path
|
2017-04-14 11:34:53 +02:00
|
|
|
|
libelle.parameterize.first(50)
|
2016-06-30 10:24:01 +02:00
|
|
|
|
end
|
|
|
|
|
|
2015-11-17 15:30:03 +01:00
|
|
|
|
def types_de_champ_ordered
|
|
|
|
|
types_de_champ.order(:order_place)
|
|
|
|
|
end
|
2015-11-19 11:37:01 +01:00
|
|
|
|
|
2016-08-03 18:19:56 +02:00
|
|
|
|
def types_de_champ_private_ordered
|
|
|
|
|
types_de_champ_private.order(:order_place)
|
|
|
|
|
end
|
|
|
|
|
|
2018-02-09 17:38:30 +01:00
|
|
|
|
def all_types_de_champ
|
|
|
|
|
types_de_champ + types_de_champ_private
|
|
|
|
|
end
|
|
|
|
|
|
2018-03-20 17:47:37 +01:00
|
|
|
|
def self.active(id)
|
2017-07-11 15:52:06 +02:00
|
|
|
|
publiees.find(id)
|
2016-06-09 17:49:38 +02:00
|
|
|
|
end
|
|
|
|
|
|
2018-03-20 17:47:37 +01:00
|
|
|
|
def switch_types_de_champ(index_of_first_element)
|
2016-06-08 16:45:18 +02:00
|
|
|
|
switch_list_order(types_de_champ_ordered, index_of_first_element)
|
|
|
|
|
end
|
|
|
|
|
|
2018-03-20 17:47:37 +01:00
|
|
|
|
def switch_types_de_champ_private(index_of_first_element)
|
2016-08-03 18:19:56 +02:00
|
|
|
|
switch_list_order(types_de_champ_private_ordered, index_of_first_element)
|
|
|
|
|
end
|
|
|
|
|
|
2018-03-20 17:47:37 +01:00
|
|
|
|
def switch_types_de_piece_justificative(index_of_first_element)
|
2017-04-13 14:48:18 +02:00
|
|
|
|
switch_list_order(types_de_piece_justificative, index_of_first_element)
|
2016-06-08 16:45:18 +02:00
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
def switch_list_order(list, index_of_first_element)
|
2017-05-26 21:43:44 +02:00
|
|
|
|
if index_of_first_element < 0 ||
|
|
|
|
|
index_of_first_element == list.count - 1 ||
|
|
|
|
|
list.count < 1
|
|
|
|
|
|
|
|
|
|
false
|
|
|
|
|
else
|
2018-03-02 16:27:03 +01:00
|
|
|
|
list[index_of_first_element].update(order_place: index_of_first_element + 1)
|
|
|
|
|
list[index_of_first_element + 1].update(order_place: index_of_first_element)
|
2017-05-26 21:43:44 +02:00
|
|
|
|
|
|
|
|
|
true
|
|
|
|
|
end
|
2015-11-19 11:37:01 +01:00
|
|
|
|
end
|
2015-12-21 14:40:28 +01:00
|
|
|
|
|
2018-04-12 18:35:13 +02:00
|
|
|
|
def clone(admin, from_library)
|
2017-03-07 18:19:48 +01:00
|
|
|
|
procedure = self.deep_clone(include:
|
2017-06-26 17:18:47 +02:00
|
|
|
|
{
|
|
|
|
|
types_de_piece_justificative: nil,
|
|
|
|
|
module_api_carto: nil,
|
|
|
|
|
attestation_template: nil,
|
|
|
|
|
types_de_champ: :drop_down_list,
|
|
|
|
|
types_de_champ_private: :drop_down_list
|
|
|
|
|
})
|
2018-05-28 14:58:40 +02:00
|
|
|
|
procedure.aasm_state = :brouillon
|
|
|
|
|
procedure.test_started_at = nil
|
2017-07-10 23:42:33 +02:00
|
|
|
|
procedure.archived_at = nil
|
2017-07-11 14:21:10 +02:00
|
|
|
|
procedure.published_at = nil
|
2016-09-02 17:10:55 +02:00
|
|
|
|
procedure.logo_secure_token = nil
|
|
|
|
|
procedure.remote_logo_url = self.logo_url
|
2017-03-07 18:19:48 +01:00
|
|
|
|
|
2018-05-31 11:00:22 +02:00
|
|
|
|
%i(notice deliberation).each { |attachment| clone_attachment(procedure, attachment) }
|
2018-04-26 14:36:27 +02:00
|
|
|
|
|
2018-01-08 14:44:15 +01:00
|
|
|
|
procedure.administrateur = admin
|
2018-05-30 18:45:46 +02:00
|
|
|
|
procedure.initiated_mail = initiated_mail&.dup
|
|
|
|
|
procedure.received_mail = received_mail&.dup
|
|
|
|
|
procedure.closed_mail = closed_mail&.dup
|
|
|
|
|
procedure.refused_mail = refused_mail&.dup
|
|
|
|
|
procedure.without_continuation_mail = without_continuation_mail&.dup
|
2017-03-07 18:19:48 +01:00
|
|
|
|
|
2018-04-12 18:35:13 +02:00
|
|
|
|
procedure.cloned_from_library = from_library
|
2018-04-24 15:23:07 +02:00
|
|
|
|
procedure.parent_procedure = self
|
2018-04-12 18:35:13 +02:00
|
|
|
|
|
2018-01-08 15:42:38 +01:00
|
|
|
|
procedure
|
2016-06-15 11:34:05 +02:00
|
|
|
|
end
|
|
|
|
|
|
2018-01-10 17:52:43 +01:00
|
|
|
|
def whitelisted?
|
|
|
|
|
whitelisted_at.present?
|
|
|
|
|
end
|
|
|
|
|
|
2016-07-22 15:06:30 +02:00
|
|
|
|
def total_dossier
|
2017-07-10 16:11:12 +02:00
|
|
|
|
self.dossiers.state_not_brouillon.size
|
2016-07-22 15:06:30 +02:00
|
|
|
|
end
|
2017-01-26 12:12:52 +01:00
|
|
|
|
|
2018-03-16 12:00:01 +01:00
|
|
|
|
def export_filename
|
|
|
|
|
procedure_identifier = procedure_path&.path || "procedure-#{id}"
|
|
|
|
|
"dossiers_#{procedure_identifier}_#{Time.now.strftime('%Y-%m-%d_%H-%M')}"
|
|
|
|
|
end
|
|
|
|
|
|
2017-04-05 11:34:16 +02:00
|
|
|
|
def generate_export
|
2017-12-01 12:24:41 +01:00
|
|
|
|
exportable_dossiers = dossiers.downloadable_sorted
|
2017-04-05 11:34:16 +02:00
|
|
|
|
|
2018-04-26 16:59:16 +02:00
|
|
|
|
headers = exportable_dossiers&.first&.export_headers || []
|
2018-04-26 18:25:59 +02:00
|
|
|
|
data = exportable_dossiers.any? ? exportable_dossiers.map(&:export_values) : [[]]
|
2017-04-05 11:34:16 +02:00
|
|
|
|
|
|
|
|
|
{
|
|
|
|
|
headers: headers,
|
|
|
|
|
data: data
|
|
|
|
|
}
|
|
|
|
|
end
|
|
|
|
|
|
2017-06-27 18:00:05 +02:00
|
|
|
|
def procedure_overview(start_date)
|
|
|
|
|
ProcedureOverview.new(self, start_date)
|
2017-05-12 15:47:05 +02:00
|
|
|
|
end
|
2017-05-26 21:55:19 +02:00
|
|
|
|
|
2017-05-27 01:08:01 +02:00
|
|
|
|
def initiated_mail_template
|
2017-12-22 21:37:08 +01:00
|
|
|
|
initiated_mail || Mails::InitiatedMail.default_for_procedure(self)
|
2017-05-26 21:55:19 +02:00
|
|
|
|
end
|
|
|
|
|
|
2017-05-27 01:08:01 +02:00
|
|
|
|
def received_mail_template
|
2017-12-22 21:37:08 +01:00
|
|
|
|
received_mail || Mails::ReceivedMail.default_for_procedure(self)
|
2017-05-26 21:55:19 +02:00
|
|
|
|
end
|
|
|
|
|
|
2017-05-27 01:08:01 +02:00
|
|
|
|
def closed_mail_template
|
2017-12-22 21:37:08 +01:00
|
|
|
|
closed_mail || Mails::ClosedMail.default_for_procedure(self)
|
2017-05-26 21:55:19 +02:00
|
|
|
|
end
|
|
|
|
|
|
2017-05-27 01:08:01 +02:00
|
|
|
|
def refused_mail_template
|
2017-12-22 21:37:08 +01:00
|
|
|
|
refused_mail || Mails::RefusedMail.default_for_procedure(self)
|
2017-05-26 21:55:19 +02:00
|
|
|
|
end
|
|
|
|
|
|
2017-05-27 01:08:01 +02:00
|
|
|
|
def without_continuation_mail_template
|
2017-12-22 21:37:08 +01:00
|
|
|
|
without_continuation_mail || Mails::WithoutContinuationMail.default_for_procedure(self)
|
2017-05-26 21:55:19 +02:00
|
|
|
|
end
|
2017-10-02 17:03:38 +02:00
|
|
|
|
|
|
|
|
|
def fields
|
|
|
|
|
fields = [
|
|
|
|
|
field_hash('Créé le', 'self', 'created_at'),
|
|
|
|
|
field_hash('Mis à jour le', 'self', 'updated_at'),
|
|
|
|
|
field_hash('Demandeur', 'user', 'email')
|
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
fields << [
|
|
|
|
|
field_hash('Civilité (FC)', 'france_connect_information', 'gender'),
|
|
|
|
|
field_hash('Prénom (FC)', 'france_connect_information', 'given_name'),
|
|
|
|
|
field_hash('Nom (FC)', 'france_connect_information', 'family_name')
|
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
if !for_individual || (for_individual && individual_with_siret)
|
|
|
|
|
fields << [
|
2018-04-23 11:57:38 +02:00
|
|
|
|
field_hash('SIREN', 'etablissement', 'entreprise_siren'),
|
|
|
|
|
field_hash('Forme juridique', 'etablissement', 'entreprise_forme_juridique'),
|
|
|
|
|
field_hash('Nom commercial', 'etablissement', 'entreprise_nom_commercial'),
|
|
|
|
|
field_hash('Raison sociale', 'etablissement', 'entreprise_raison_sociale'),
|
|
|
|
|
field_hash('SIRET siège social', 'etablissement', 'entreprise_siret_siege_social'),
|
|
|
|
|
field_hash('Date de création', 'etablissement', 'entreprise_date_creation')
|
2017-10-02 17:03:38 +02:00
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
fields << [
|
|
|
|
|
field_hash('SIRET', 'etablissement', 'siret'),
|
2017-12-06 17:34:03 +01:00
|
|
|
|
field_hash('Libellé NAF', 'etablissement', 'libelle_naf'),
|
2017-10-02 17:03:38 +02:00
|
|
|
|
field_hash('Code postal', 'etablissement', 'code_postal')
|
|
|
|
|
]
|
|
|
|
|
end
|
|
|
|
|
|
2017-10-03 15:08:51 +02:00
|
|
|
|
types_de_champ
|
2018-01-15 22:10:46 +01:00
|
|
|
|
.reject { |tdc| ['header_section', 'explication'].include?(tdc.type_champ) }
|
2017-10-03 15:08:51 +02:00
|
|
|
|
.each do |type_de_champ|
|
|
|
|
|
|
2017-10-02 17:03:38 +02:00
|
|
|
|
fields << field_hash(type_de_champ.libelle, 'type_de_champ', type_de_champ.id.to_s)
|
|
|
|
|
end
|
|
|
|
|
|
2017-10-03 15:08:51 +02:00
|
|
|
|
types_de_champ_private
|
2018-01-15 22:10:46 +01:00
|
|
|
|
.reject { |tdc| ['header_section', 'explication'].include?(tdc.type_champ) }
|
2017-10-03 15:08:51 +02:00
|
|
|
|
.each do |type_de_champ|
|
|
|
|
|
|
2017-10-02 17:03:38 +02:00
|
|
|
|
fields << field_hash(type_de_champ.libelle, 'type_de_champ_private', type_de_champ.id.to_s)
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
fields.flatten
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
def fields_for_select
|
|
|
|
|
fields.map do |field|
|
|
|
|
|
[field['label'], "#{field['table']}/#{field['column']}"]
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
2017-09-27 15:16:07 +02:00
|
|
|
|
def self.default_sort
|
|
|
|
|
{
|
|
|
|
|
'table' => 'self',
|
|
|
|
|
'column' => 'id',
|
|
|
|
|
'order' => 'desc'
|
|
|
|
|
}.to_json
|
|
|
|
|
end
|
|
|
|
|
|
2018-01-10 16:46:12 +01:00
|
|
|
|
def whitelist!
|
|
|
|
|
update_attribute('whitelisted_at', DateTime.now)
|
|
|
|
|
end
|
|
|
|
|
|
2018-04-03 11:33:11 +02:00
|
|
|
|
def closed_mail_template_attestation_inconsistency_state
|
|
|
|
|
# As an optimization, don’t check the predefined templates (they are presumed correct)
|
|
|
|
|
if closed_mail.present?
|
|
|
|
|
tag_present = closed_mail.body.include?("--lien attestation--")
|
|
|
|
|
if attestation_template&.activated? && !tag_present
|
|
|
|
|
:missing_tag
|
|
|
|
|
elsif !attestation_template&.activated? && tag_present
|
|
|
|
|
:extraneous_tag
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
2017-10-02 17:03:38 +02:00
|
|
|
|
private
|
|
|
|
|
|
2018-05-31 11:00:22 +02:00
|
|
|
|
def clone_attachment(cloned_procedure, attachment_symbol)
|
|
|
|
|
attachment = send(attachment_symbol)
|
|
|
|
|
if attachment.attached?
|
|
|
|
|
response = Typhoeus.get(attachment.service_url, timeout: 5)
|
|
|
|
|
if response.success?
|
|
|
|
|
cloned_procedure.send(attachment_symbol).attach(
|
|
|
|
|
io: StringIO.new(response.body),
|
|
|
|
|
filename: attachment.filename
|
|
|
|
|
)
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
2018-05-31 10:59:38 +02:00
|
|
|
|
def check_juridique
|
|
|
|
|
if cadre_juridique.blank? && !deliberation.attached?
|
|
|
|
|
errors.add(:cadre_juridique, " : veuillez remplir le texte de loi ou la délibération")
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
2017-10-02 17:03:38 +02:00
|
|
|
|
def field_hash(label, table, column)
|
|
|
|
|
{
|
|
|
|
|
'label' => label,
|
|
|
|
|
'table' => table,
|
|
|
|
|
'column' => column
|
|
|
|
|
}
|
|
|
|
|
end
|
2015-09-21 17:59:03 +02:00
|
|
|
|
end
|