diff --git a/app/assets/images/extranet/avatar.png b/app/assets/images/extranet/avatar.png
new file mode 100644
index 0000000000000000000000000000000000000000..34b4e0cdea8b2ce18e1fb644989edb762cc0dd63
Binary files /dev/null and b/app/assets/images/extranet/avatar.png differ
diff --git a/app/assets/stylesheets/application.sass b/app/assets/stylesheets/application.sass
index 819e1e0d4ec1e07d84d8299ae5dd4435054b3481..1d71576b129df81fc868efe53a4d0f83414091a8 100644
--- a/app/assets/stylesheets/application.sass
+++ b/app/assets/stylesheets/application.sass
@@ -1,5 +1,5 @@
+@import 'application/variables'
 @import 'bootstrap'
-@import 'appstack/light'
 @import 'simple_form_password_with_hints'
 @import 'simple_form_bs5_file_input'
 @import 'cropperjs/dist/cropper'
diff --git a/app/assets/stylesheets/application/layout.sass b/app/assets/stylesheets/application/layout.sass
index 5e912ccbf0ec75611bcc73c6d0785175dfca77e5..16af462e1104683757ccb592320adfcaa2c7799d 100644
--- a/app/assets/stylesheets/application/layout.sass
+++ b/app/assets/stylesheets/application/layout.sass
@@ -1,7 +1,49 @@
-footer
-    margin-top: 100px
 
 .alert
     padding: .95rem
     &-danger
         color: #82322F
+
+.extranet
+    .navbar
+        margin-bottom: 100px
+        .navbar-brand
+            img
+                max-width: 100px
+    header
+        border-bottom: 1px solid
+        border-top: 1px solid
+        min-height: 160px
+        h1, p
+            padding-top: 3rem
+    footer
+        margin-top: 100px
+        .logo
+            width: 100px
+
+.breadcrumb
+    font-size: 14px
+    padding-bottom: 50px
+.card
+    background: black
+    border: none
+    &, a
+        color: white
+        text-decoration-color: rgba(white, 0.5)
+        &:hover
+            text-decoration-color: white
+
+a
+    color: $primary
+    text-decoration-color: adjust-color($primary, $alpha: -0.8)
+    text-decoration-thickness: 1px
+    text-underline-offset: 3px
+    transition: text-decoration 0.5s
+    &:hover
+        text-decoration-color: $primary
+    &[target="_blank"]
+        &::after
+            background: url(data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIHZpZXdCb3g9IjAgMCAxNiAxNiIgd2lkdGg9IjE2IiBoZWlnaHQ9IjE2Ij48cGF0aCBkPSJNOSAyTDkgMyAxMi4zIDMgNiA5LjMgNi43IDEwIDEzIDMuNyAxMyA3IDE0IDcgMTQgMlpNNCA0QzIuOSA0IDIgNC45IDIgNkwyIDEyQzIgMTMuMSAyLjkgMTQgNCAxNEwxMCAxNEMxMS4xIDE0IDEyIDEzLjEgMTIgMTJMMTIgNyAxMSA4IDExIDEyQzExIDEyLjYgMTAuNiAxMyAxMCAxM0w0IDEzQzMuNCAxMyAzIDEyLjYgMyAxMkwzIDZDMyA1LjQgMy40IDUgNCA1TDggNSA5IDRaIi8+PC9zdmc+) no-repeat
+            background-size: contain
+            content: ""
+            display: inline-block
diff --git a/app/assets/stylesheets/application/variables.sass b/app/assets/stylesheets/application/variables.sass
new file mode 100644
index 0000000000000000000000000000000000000000..2cff045b6011ee11152218ff33e0aeb3fecbce54
--- /dev/null
+++ b/app/assets/stylesheets/application/variables.sass
@@ -0,0 +1,2 @@
+$primary: black
+$border-radius: 0
diff --git a/app/controllers/admin/application_controller.rb b/app/controllers/admin/application_controller.rb
index 0f004a7487b42dd693e61e3ee6c28f88bf0394a4..921a46d7ae157ee5413edfc7767c95ae8065efa0 100644
--- a/app/controllers/admin/application_controller.rb
+++ b/app/controllers/admin/application_controller.rb
@@ -1,7 +1,6 @@
 class Admin::ApplicationController < ApplicationController
   layout 'admin/layouts/application'
 
-  before_action :authenticate_user!
   around_action :switch_locale
 
   protected
diff --git a/app/controllers/admin/education/cohorts_controller.rb b/app/controllers/admin/education/cohorts_controller.rb
new file mode 100644
index 0000000000000000000000000000000000000000..c32d9edf9cb122816d2d85191765d2ca214ba597
--- /dev/null
+++ b/app/controllers/admin/education/cohorts_controller.rb
@@ -0,0 +1,59 @@
+class Admin::Education::CohortsController < Admin::Education::ApplicationController
+  load_and_authorize_resource class: Education::Cohort,
+                              through: :current_university,
+                              through_association: :education_cohorts
+
+  def index
+    breadcrumb
+  end
+
+  def show
+    breadcrumb
+  end
+
+  def new
+    breadcrumb
+  end
+
+  def edit
+    breadcrumb
+  end
+
+  def create
+    @cohort.university = current_university
+    if @cohort.save
+      redirect_to [:admin, @cohort],
+                  notice: t('admin.successfully_created_html', model: @cohort.to_s)
+    else
+      breadcrumb
+      render :new, status: :unprocessable_entity
+    end
+  end
+
+  def update
+    if @cohort.update(cohort_params)
+      redirect_to [:admin, @cohort],
+                  notice: t('admin.successfully_updated_html', model: @cohort.to_s)
+    else
+      render :edit, status: :unprocessable_entity
+    end
+  end
+
+  def destroy
+    @cohort.destroy
+    redirect_to education_cohorts_url,
+                notice: t('admin.successfully_destroyed_html', model: @cohort.to_s)
+  end
+
+  protected
+
+  def breadcrumb
+    super
+    add_breadcrumb Education::cohort.model_name.human(count: 2), admin_education_cohorts_path
+    breadcrumb_for @cohort
+  end
+
+  def cohort_params
+    params.require(:education_cohort).permit(:university_id, :program_id, :academic_year_id, :name)
+  end
+end
diff --git a/app/controllers/admin/university/people_controller.rb b/app/controllers/admin/university/people_controller.rb
index 8babe56ea328c2158113c84cb111edb2bd879053..febdd982957cd889fa19d5cec9898e019490dcb8 100644
--- a/app/controllers/admin/university/people_controller.rb
+++ b/app/controllers/admin/university/people_controller.rb
@@ -64,7 +64,8 @@ class Admin::University::PeopleController < Admin::University::ApplicationContro
       :slug, :first_name, :last_name, :email, :phone, :description,
       :biography,  :picture, :picture_delete, :picture_infos,
       :habilitation, :tenure, :url, :linkedin, :twitter,
-      :is_researcher, :is_teacher, :is_administration, :user_id
+      :is_researcher, :is_teacher, :is_administration, :is_alumnus,
+      :user_id
     ).merge(university_id: current_university.id)
   end
 end
diff --git a/app/controllers/admin/university/person/alumni_controller.rb b/app/controllers/admin/university/person/alumni_controller.rb
new file mode 100644
index 0000000000000000000000000000000000000000..f86a16f465887e83060d153fabc10a6d9c35c0f5
--- /dev/null
+++ b/app/controllers/admin/university/person/alumni_controller.rb
@@ -0,0 +1,46 @@
+class Admin::University::Person::AlumniController < Admin::University::ApplicationController
+  load_and_authorize_resource class: University::Person::Alumnus,
+                              through: :current_university,
+                              through_association: :people
+  def index
+    @alumni = @alumni.alumni
+                     .accessible_by(current_ability)
+                     .ordered
+                     .page(params[:page])
+    breadcrumb
+  end
+
+  def show
+    breadcrumb
+  end
+
+  def edit
+    breadcrumb
+    add_breadcrumb t('edit')
+  end
+
+  def update
+    if @alumnus.update(alumnus_params)
+      redirect_to [:admin, @alumnus],
+                  notice: t('admin.successfully_updated_html', model: @alumnus.to_s)
+    else
+      render :edit
+      breadcrumb
+      add_breadcrumb t('edit')
+    end
+  end
+
+  protected
+
+  def breadcrumb
+    super
+    add_breadcrumb  University::Person::Alumnus.model_name.human(count: 2),
+                    admin_university_person_alumni_path
+    breadcrumb_for  @alumnus
+  end
+
+  def alumnus_params
+    params.require(:university_person_alumnus)
+          .permit()
+  end
+end
diff --git a/app/controllers/admin/university/person/alumnus/imports_controller.rb b/app/controllers/admin/university/person/alumnus/imports_controller.rb
new file mode 100644
index 0000000000000000000000000000000000000000..b8f7b52a12ce18b78126eb3b1bb85297b2e4b143
--- /dev/null
+++ b/app/controllers/admin/university/person/alumnus/imports_controller.rb
@@ -0,0 +1,43 @@
+class Admin::University::Person::Alumnus::ImportsController < Admin::University::ApplicationController
+  load_and_authorize_resource class: University::Person::Alumnus::Import,
+                              through: :current_university,
+                              through_association: :person_alumnus_imports
+
+  def index
+    breadcrumb
+  end
+
+  def show
+    breadcrumb
+  end
+
+  def new
+    breadcrumb
+  end
+
+  def create
+    @import.university = current_university
+    @import.user = current_user
+    if @import.save
+      redirect_to [:admin, @import], notice: "Import was successfully created."
+    else
+      render :new, status: :unprocessable_entity
+    end
+  end
+
+  protected
+
+  def breadcrumb
+    super
+    add_breadcrumb  University::Person::Alumnus.model_name.human(count: 2),
+                    admin_university_person_alumni_path
+    add_breadcrumb  University::Person::Alumnus::Import.model_name.human(count: 2),
+                    admin_university_person_alumnus_imports_path
+    breadcrumb_for  @import
+  end
+
+  def import_params
+    params.require(:university_person_alumnus_import)
+          .permit(:file)
+  end
+end
diff --git a/app/controllers/api/application_controller.rb b/app/controllers/api/application_controller.rb
new file mode 100644
index 0000000000000000000000000000000000000000..9445c98708ab0c6491c34bd92ca29f90826fb032
--- /dev/null
+++ b/app/controllers/api/application_controller.rb
@@ -0,0 +1,4 @@
+class Api::ApplicationController < ApplicationController
+  layout false
+  skip_before_action :authenticate_user!
+end
diff --git a/app/controllers/api/dashboard_controller.rb b/app/controllers/api/dashboard_controller.rb
new file mode 100644
index 0000000000000000000000000000000000000000..d9e2476a272600da70272b91f8b92ab8052e6d98
--- /dev/null
+++ b/app/controllers/api/dashboard_controller.rb
@@ -0,0 +1,6 @@
+class Api::DashboardController < Api::ApplicationController
+  layout 'api/layouts/application'
+
+  def index
+  end
+end
diff --git a/app/controllers/api/lheo_controller.rb b/app/controllers/api/lheo_controller.rb
new file mode 100644
index 0000000000000000000000000000000000000000..5e788e8f75c1492838f662ead3b40bfcdd7966c2
--- /dev/null
+++ b/app/controllers/api/lheo_controller.rb
@@ -0,0 +1,5 @@
+class Api::LheoController < Api::ApplicationController
+  def index
+    @programs = current_university.education_programs
+  end
+end
diff --git a/app/controllers/application_controller.rb b/app/controllers/application_controller.rb
index b296bd1603094237c05fa47fa3d948a30040fde1..d3b4ac76472b87a3e4ca59fca8f6150bf7b85273 100644
--- a/app/controllers/application_controller.rb
+++ b/app/controllers/application_controller.rb
@@ -3,6 +3,8 @@ class ApplicationController < ActionController::Base
   include WithLocale
   include WithDomain
 
+  before_action :authenticate_user!
+
   def breadcrumb
     add_breadcrumb t('home'), root_path
   end
diff --git a/app/controllers/extranet/academic_years_controller.rb b/app/controllers/extranet/academic_years_controller.rb
new file mode 100644
index 0000000000000000000000000000000000000000..a8439fd6ed87c6cfaaa3900373e67f628c97e7f3
--- /dev/null
+++ b/app/controllers/extranet/academic_years_controller.rb
@@ -0,0 +1,22 @@
+class Extranet::AcademicYearsController < Extranet::ApplicationController
+  load_and_authorize_resource class: Education::AcademicYear,
+                              through: :current_university,
+                              through_association: :academic_years
+
+  def index
+    @academic_years = @academic_years.ordered.page(params[:page])
+    breadcrumb
+  end
+
+  def show
+    breadcrumb
+  end
+
+  protected
+
+  def breadcrumb
+    super
+    add_breadcrumb Education::AcademicYear.model_name.human(count: 2), education_academic_years_path
+    add_breadcrumb @academic_year if @academic_year
+  end
+end
diff --git a/app/controllers/extranet/application_controller.rb b/app/controllers/extranet/application_controller.rb
new file mode 100644
index 0000000000000000000000000000000000000000..ba0116397edf6125094bc8b0790ee968157c22a9
--- /dev/null
+++ b/app/controllers/extranet/application_controller.rb
@@ -0,0 +1,7 @@
+class Extranet::ApplicationController < ApplicationController
+  layout 'extranet/layouts/application'
+
+  def breadcrumb
+    add_breadcrumb t('home'), root_path
+  end
+end
diff --git a/app/controllers/extranet/cohorts_controller.rb b/app/controllers/extranet/cohorts_controller.rb
new file mode 100644
index 0000000000000000000000000000000000000000..d6935d45b80e571b00805a48bf459a6edc06ffc4
--- /dev/null
+++ b/app/controllers/extranet/cohorts_controller.rb
@@ -0,0 +1,22 @@
+class Extranet::CohortsController < Extranet::ApplicationController
+  load_and_authorize_resource class: Education::Cohort,
+                              through: :current_university,
+                              through_association: :education_cohorts
+
+  def index
+    @cohorts = @cohorts.ordered.page(params[:page])
+    breadcrumb
+  end
+
+  def show
+    breadcrumb
+  end
+
+  protected
+
+  def breadcrumb
+    super
+    add_breadcrumb Education::Cohort.model_name.human(count: 2), education_cohorts_path
+    add_breadcrumb @cohort if @cohort
+  end
+end
diff --git a/app/controllers/extranet/home_controller.rb b/app/controllers/extranet/home_controller.rb
new file mode 100644
index 0000000000000000000000000000000000000000..6c097138065df7a0d5a018826bf23d7d86f05918
--- /dev/null
+++ b/app/controllers/extranet/home_controller.rb
@@ -0,0 +1,6 @@
+class Extranet::HomeController < Extranet::ApplicationController
+  def index
+    redirect_to admin_root_path unless current_extranet
+    @cohorts = current_university.education_cohorts.ordered.limit(2)
+  end
+end
diff --git a/app/controllers/extranet/organizations_controller.rb b/app/controllers/extranet/organizations_controller.rb
new file mode 100644
index 0000000000000000000000000000000000000000..d91329fb8279c04f91c6a3b8cbc040e21114cb7b
--- /dev/null
+++ b/app/controllers/extranet/organizations_controller.rb
@@ -0,0 +1,22 @@
+class Extranet::OrganizationsController < Extranet::ApplicationController
+  load_and_authorize_resource class: University::Organization,
+                              through: :current_university,
+                              through_association: :organizations
+
+  def index
+    @organizations = @organizations.ordered.page(params[:page])
+    breadcrumb
+  end
+
+  def show
+    breadcrumb
+  end
+
+  protected
+
+  def breadcrumb
+    super
+    add_breadcrumb University::Organization.model_name.human(count: 2), university_organizations_path
+    add_breadcrumb @organization if @organization
+  end
+end
diff --git a/app/controllers/extranet/persons_controller.rb b/app/controllers/extranet/persons_controller.rb
new file mode 100644
index 0000000000000000000000000000000000000000..245f8eb5c9def0dd0129abf876108c90e4b87471
--- /dev/null
+++ b/app/controllers/extranet/persons_controller.rb
@@ -0,0 +1,22 @@
+class Extranet::PersonsController < Extranet::ApplicationController
+  load_and_authorize_resource class: University::Person::Alumnus,
+                              through: :current_university,
+                              through_association: :people
+
+  def index
+    @people = @people.alumni
+    breadcrumb
+  end
+
+  def show
+    breadcrumb
+  end
+
+  protected
+
+  def breadcrumb
+    super
+    add_breadcrumb University::Person::Alumnus.model_name.human(count: 2), university_persons_path
+    add_breadcrumb @person if @person
+  end
+end
diff --git a/app/controllers/home_controller.rb b/app/controllers/home_controller.rb
deleted file mode 100644
index 26e35e8c8978d80f3ef93d0797e1583d0853502e..0000000000000000000000000000000000000000
--- a/app/controllers/home_controller.rb
+++ /dev/null
@@ -1,5 +0,0 @@
-class HomeController < ApplicationController
-  def index
-    redirect_to admin_root_path unless current_extranet
-  end
-end
diff --git a/app/controllers/university/organizations_controller.rb b/app/controllers/university/organizations_controller.rb
deleted file mode 100644
index f3340418978c0486416128b57e3e6faf0dcdd96e..0000000000000000000000000000000000000000
--- a/app/controllers/university/organizations_controller.rb
+++ /dev/null
@@ -1,12 +0,0 @@
-class University::OrganizationsController < ApplicationController
-  load_and_authorize_resource class: University::Organization,
-                              through: :current_university,
-                              through_association: :organizations
-
-  def index
-    @organizations = @organizations.ordered.page(params[:page])
-  end
-
-  def show
-  end
-end
diff --git a/app/helpers/application_helper.rb b/app/helpers/application_helper.rb
index eb980caeec12a69438811ee5abecc3688d80e17e..caddfa3c68c1e0746ccab3d395c8dd7e56205918 100644
--- a/app/helpers/application_helper.rb
+++ b/app/helpers/application_helper.rb
@@ -11,4 +11,35 @@ module ApplicationHelper
     classes
   end
 
+  def social_website_to_url(string)
+    string = "https://#{string}" unless string.start_with?('http')
+    string.gsub('http://', 'https://')
+  end
+
+  def social_website_to_s(string)
+    string.gsub('http://', '')
+          .gsub('https://', '')
+  end
+
+  def social_linkedin_to_url(string)
+    string.gsub('http://', 'https://')
+  end
+
+  def social_linkedin_to_s(string)
+    string.gsub('http://', 'https://')
+          .gsub('https://www.linkedin.com/in/', '')
+  end
+
+  def social_twitter_to_url(string)
+    string = "https://twitter.com/#{string}" unless 'twitter.com'.in? string
+    string = "https://#{string}" unless string.start_with?('http')
+    string.gsub('http://', 'https://')
+  end
+
+  def social_twitter_to_s(string)
+    string.gsub('http://', 'https://')
+          .gsub('twitter.com', 'https://twitter.com')
+          .gsub('https://www.twitter.com/', 'https://twitter.com/')
+          .gsub('https://twitter.com/', '')
+  end
 end
diff --git a/app/models/concerns/importable.rb b/app/models/concerns/importable.rb
new file mode 100644
index 0000000000000000000000000000000000000000..6bc569eef9894ed0082e55ff57a6ceb95c9a7774
--- /dev/null
+++ b/app/models/concerns/importable.rb
@@ -0,0 +1,36 @@
+module Importable
+  extend ActiveSupport::Concern
+
+  included do
+    belongs_to :user
+
+    has_one_attached :file
+
+    after_save :parse_async
+  end
+
+  def lines
+    csv.count
+  rescue
+    'NA'
+  end
+
+  def to_s
+    "#{user}, #{I18n.l created_at}"
+  end
+
+  protected
+
+  def parse_async
+    parse
+  end
+  handle_asynchronously :parse_async, queue: 'default'
+
+  def parse
+    raise NotImplementedError
+  end
+
+  def csv
+    @csv ||= CSV.parse file.blob.download, headers: true
+  end
+end
diff --git a/app/models/education/academic_year.rb b/app/models/education/academic_year.rb
new file mode 100644
index 0000000000000000000000000000000000000000..ff203144a1fc362b05ad0c8e41ddb71611541381
--- /dev/null
+++ b/app/models/education/academic_year.rb
@@ -0,0 +1,33 @@
+# == Schema Information
+#
+# Table name: education_academic_years
+#
+#  id            :uuid             not null, primary key
+#  year          :integer
+#  created_at    :datetime         not null
+#  updated_at    :datetime         not null
+#  university_id :uuid             not null, indexed
+#
+# Indexes
+#
+#  index_education_academic_years_on_university_id  (university_id)
+#
+# Foreign Keys
+#
+#  fk_rails_7d376afe35  (university_id => universities.id)
+#
+class Education::AcademicYear < ApplicationRecord
+  include WithUniversity
+
+  has_many :cohorts, class_name: 'Education::Cohort'
+
+  has_many :people,
+           class_name: 'University::Person',
+           through: :cohorts
+
+  scope :ordered, -> { order(year: :desc) }
+
+  def to_s
+    "#{year}"
+  end
+end
diff --git a/app/models/education/cohort.rb b/app/models/education/cohort.rb
new file mode 100644
index 0000000000000000000000000000000000000000..862b7578816503c4866990acc3571278096418e9
--- /dev/null
+++ b/app/models/education/cohort.rb
@@ -0,0 +1,39 @@
+# == Schema Information
+#
+# Table name: education_cohorts
+#
+#  id               :uuid             not null, primary key
+#  name             :string
+#  created_at       :datetime         not null
+#  updated_at       :datetime         not null
+#  academic_year_id :uuid             not null, indexed
+#  program_id       :uuid             not null, indexed
+#  university_id    :uuid             not null, indexed
+#
+# Indexes
+#
+#  index_education_cohorts_on_academic_year_id  (academic_year_id)
+#  index_education_cohorts_on_program_id        (program_id)
+#  index_education_cohorts_on_university_id     (university_id)
+#
+# Foreign Keys
+#
+#  fk_rails_0f4a4f43d9  (university_id => universities.id)
+#  fk_rails_72528c3d76  (program_id => education_programs.id)
+#  fk_rails_c2d725cabd  (academic_year_id => education_academic_years.id)
+#
+class Education::Cohort < ApplicationRecord
+  include WithUniversity
+  belongs_to :program, class_name: 'Education::Program'
+  belongs_to :academic_year, class_name: 'Education::AcademicYear'
+  has_and_belongs_to_many :people,
+                          class_name: 'University::Person',
+                          foreign_key: 'education_cohort_id',
+                          association_foreign_key: 'university_person_id'
+
+  scope :ordered, -> { includes(:academic_year).order('education_academic_years.year DESC') }
+
+  def to_s
+    "#{program} #{academic_year}"
+  end
+end
diff --git a/app/models/university/person.rb b/app/models/university/person.rb
index 45667c4a5d5ff3fbc49fd007f5e877573970fd18..73144b3d28b5e2fdf7dda8da80ad85a4c2e1157b 100644
--- a/app/models/university/person.rb
+++ b/app/models/university/person.rb
@@ -86,6 +86,11 @@ class University::Person < ApplicationRecord
                           through: :education_programs,
                           source: :websites
 
+  has_and_belongs_to_many :cohorts,
+                          class_name: 'Education::Cohort',
+                          foreign_key: 'university_person_id',
+                          association_foreign_key: 'education_cohort_id'
+
   accepts_nested_attributes_for :involvements
 
   validates_presence_of   :first_name, :last_name
@@ -101,9 +106,10 @@ class University::Person < ApplicationRecord
   before_validation :sanitize_email
 
   scope :ordered,         -> { order(:last_name, :first_name) }
-  scope :administration, -> { where(is_administration: true) }
+  scope :administration,  -> { where(is_administration: true) }
   scope :teachers,        -> { where(is_teacher: true) }
   scope :researchers,     -> { where(is_researcher: true) }
+  scope :alumni,          -> { where(is_alumnus: true) }
 
   def to_s
     "#{first_name} #{last_name}"
@@ -118,7 +124,7 @@ class University::Person < ApplicationRecord
   end
 
   def git_dependencies(website)
-    dependencies = website.menus
+    dependencies = website.menus.to_a
     if for_website?(website)
       dependencies << self
       dependencies.concat active_storage_blobs
diff --git a/app/models/university/person/alumnus.rb b/app/models/university/person/alumnus.rb
new file mode 100644
index 0000000000000000000000000000000000000000..477f0ef3006f781d0b03afeca3c420ae499844e1
--- /dev/null
+++ b/app/models/university/person/alumnus.rb
@@ -0,0 +1,49 @@
+# == Schema Information
+#
+# Table name: university_people
+#
+#  id                :uuid             not null, primary key
+#  biography         :text
+#  description       :text
+#  email             :string
+#  first_name        :string
+#  habilitation      :boolean          default(FALSE)
+#  is_administration :boolean
+#  is_alumnus        :boolean          default(FALSE)
+#  is_researcher     :boolean
+#  is_teacher        :boolean
+#  last_name         :string
+#  linkedin          :string
+#  phone             :string
+#  slug              :string
+#  tenure            :boolean          default(FALSE)
+#  twitter           :string
+#  url               :string
+#  created_at        :datetime         not null
+#  updated_at        :datetime         not null
+#  university_id     :uuid             not null, indexed
+#  user_id           :uuid             indexed
+#
+# Indexes
+#
+#  index_university_people_on_university_id  (university_id)
+#  index_university_people_on_user_id        (user_id)
+#
+# Foreign Keys
+#
+#  fk_rails_b47a769440  (user_id => users.id)
+#  fk_rails_da35e70d61  (university_id => universities.id)
+#
+class University::Person::Alumnus < University::Person
+  def self.polymorphic_name
+    'University::Person::Alumnus'
+  end
+
+  def git_path(website)
+    # TODO
+  end
+
+  def for_website?(website)
+    # TODO
+  end
+end
diff --git a/app/models/university/person/alumnus/import.rb b/app/models/university/person/alumnus/import.rb
new file mode 100644
index 0000000000000000000000000000000000000000..c35dcadee5096b3e5a8f7c3d93d55a8c11a1a6e4
--- /dev/null
+++ b/app/models/university/person/alumnus/import.rb
@@ -0,0 +1,102 @@
+# == Schema Information
+#
+# Table name: university_person_alumnus_imports
+#
+#  id            :uuid             not null, primary key
+#  created_at    :datetime         not null
+#  updated_at    :datetime         not null
+#  university_id :uuid             not null, indexed
+#  user_id       :uuid             not null, indexed
+#
+# Indexes
+#
+#  index_university_person_alumnus_imports_on_university_id  (university_id)
+#  index_university_person_alumnus_imports_on_user_id        (user_id)
+#
+# Foreign Keys
+#
+#  fk_rails_3ff74ac195  (user_id => users.id)
+#  fk_rails_d14eb003f9  (university_id => universities.id)
+#
+class University::Person::Alumnus::Import < ApplicationRecord
+  include WithUniversity
+  include Importable
+
+  def self.table_name
+    'university_person_alumnus_imports'
+  end
+
+  protected
+
+  def parse
+    csv.each do |row|
+      row['program'] = '23279cab-8bc1-4c75-bcd8-1fccaa03ad55' #TMP local fix
+      program = university.education_programs
+                          .find_by(id: row['program'])
+      next if program.nil?
+      academic_year = university.academic_years
+                                .where(year: row['year'])
+                                .first_or_create
+
+      cohort = university.education_cohorts
+                         .where(program: program, academic_year: academic_year)
+                         .first_or_create
+      first_name = clean_encoding row['first_name']
+      last_name = clean_encoding row['last_name']
+      email = clean_encoding(row['mail']).to_s.downcase
+      url = clean_encoding row['url']
+      if email.present?
+        person = university.people
+                           .where(email: email)
+                           .first_or_create
+        person.first_name = first_name
+        person.last_name = last_name
+      else
+        person = university.people
+                           .where(first_name: first_name, last_name: last_name)
+                           .first_or_create
+      end
+      # TODO all fields
+      # gender
+      # birth
+      # phonepro
+      # phoneperso
+      # address
+      # zipcode
+      # city
+      # country
+      # status
+      person.is_alumnus = true
+      person.url = url
+      person.slug = person.to_s.parameterize.dasherize
+      person.twitter ||= row['socialtwitter']
+      person.linkedin ||= row['sociallinkedin']
+      person.biography ||= row['status']
+      person.phone ||= row['mobile']
+      byebug unless person.valid?
+      person.save
+      cohort.people << person unless person.in?(cohort.people)
+      add_picture person, row['photo']
+    end
+  end
+
+  def add_picture(person, photo)
+    return if photo.nil?
+    return if person.picture.attached?
+    return unless photo.end_with?('.jpg') || photo.end_with?('.png')
+    begin
+      file = URI.open photo
+      filename = File.basename photo
+      person.picture.attach(io: file, filename: filename)
+    rescue
+    end
+  end
+
+  def clean_encoding(value)
+    return if value.nil?
+    if value.encoding != 'UTF-8'
+      value = value.force_encoding 'UTF-8'
+    end
+    value.strip
+  end
+end
diff --git a/app/models/university/person/with_education.rb b/app/models/university/person/with_education.rb
index 1074dea4c1897fc9f03117426452d242cd7ded06..a72fbde6bb02579ffd06675d9597ee2b9b9fa596 100644
--- a/app/models/university/person/with_education.rb
+++ b/app/models/university/person/with_education.rb
@@ -11,6 +11,9 @@ module University::Person::WithEducation
               through: :involvements_as_teacher,
               source: :target,
               source_type: "Education::Program"
+
+    has_and_belongs_to_many :cohorts,
+                            class_name: 'Education::Cohort'
   end
 
   def education_programs_as_administrator
diff --git a/app/models/university/with_education.rb b/app/models/university/with_education.rb
index 5acddc90f4055bb204487797d97d6994c82a189e..eab20006b33a912ae68713ef3a0d3892eb1fa1c8 100644
--- a/app/models/university/with_education.rb
+++ b/app/models/university/with_education.rb
@@ -2,7 +2,9 @@ module University::WithEducation
   extend ActiveSupport::Concern
 
   included do
+    has_many :education_cohorts, class_name: 'Education::Cohort', dependent: :destroy
     has_many :education_programs, class_name: 'Education::Program', dependent: :destroy
     has_many :education_schools, class_name: 'Education::School', dependent: :destroy
+    has_many :academic_years, class_name: 'Education::AcademicYear', dependent: :destroy
   end
 end
diff --git a/app/models/university/with_people_and_organizations.rb b/app/models/university/with_people_and_organizations.rb
index d943867e07b1dd6384584dccba13588c4267886f..ad270f63a8638aaef1549a029877ca523464715c 100644
--- a/app/models/university/with_people_and_organizations.rb
+++ b/app/models/university/with_people_and_organizations.rb
@@ -11,5 +11,8 @@ module University::WithPeopleAndOrganizations
     has_many  :organization_imports,
               class_name: 'University::Organization::Import',
               dependent: :destroy
+    has_many :person_alumnus_imports,
+              class_name: 'University::Person::Alumnus::Import',
+              dependent: :destroy
   end
 end
diff --git a/app/services/simple_navigation/bootstrap_renderer.rb b/app/services/simple_navigation/bootstrap_renderer.rb
new file mode 100644
index 0000000000000000000000000000000000000000..21646e9f9a18cc926ae4e1e890d68a7aea9b3ada
--- /dev/null
+++ b/app/services/simple_navigation/bootstrap_renderer.rb
@@ -0,0 +1,44 @@
+class SimpleNavigation::BootstrapRenderer < SimpleNavigation::Renderer::Base
+  def render(item_container)
+    content = '<ul class="nav">'
+    item_container.items.each do |item|
+      content << make(item)
+    end
+    content << '</ul>'
+    content.html_safe
+  end
+
+  protected
+
+  def make(item)
+    li = "<li class=\"sidebar-item #{ item.html_options[:class] } #{ ' disabled' unless item.url }\">"
+    li += make_a(item)
+    li += make_subnavigation(item) if consider_sub_navigation?(item)
+    li += '</li>'
+    li
+  end
+
+  def make_item(item)
+    li = "<li class=\"nav-item #{ item.html_options[:class] } #{ ' disabled' unless item.url }\">"
+    li += make_a(item)
+    li += make_subnavigation(item) if consider_sub_navigation?(item)
+    li += '</li>'
+    li
+  end
+
+  def make_a(item)
+    icon = item.send(:options)[:icon]
+    a = "<a href=\"#{ item.url }\" class=\"nav-link#{ item.selected? ? '' : ' collapsed' }\""
+    a += " data-bs-target=\"##{ item.key }\" data-bs-toggle=\"collapse\"" if consider_sub_navigation?(item)
+    a += ">"
+    a += "<i class=\"fas fa-#{ icon }\"></i>" if icon
+    a += "<span class=\"align-middle\">#{ item.name }</span></a>"
+    a
+  end
+
+  def make_subnavigation(item)
+    "<ul id=\"#{ item.key }\" class=\"sidebar-dropdown list-unstyled #{ item.selected? ? 'show' : 'collapse' }\">
+      #{ render_sub_navigation_for item }
+    </ul>"
+  end
+end
diff --git a/app/views/admin/application/_nav.html.erb b/app/views/admin/application/_nav.html.erb
index 53f822a2f23fed05a748e58d76329bc5ed51d172..8843006cbca3dea2d01a640db1fa75913e5ff3d5 100644
--- a/app/views/admin/application/_nav.html.erb
+++ b/app/views/admin/application/_nav.html.erb
@@ -7,6 +7,7 @@
 
     <footer class="small my-5">
       <hr>
+      <%= link_to 'API', api_root_path, class: 'sidebar-link' %>
       <%
       [
         :terms_of_service,
@@ -19,7 +20,9 @@
                     rel: 'noreferrer',
                     class: 'sidebar-link' %>
       <% end %>
-      <%= link_to t('cookies_consent_choice'), '', class: 'sidebar-link js-gdpr__cookie_consent__display_again' %>
+      <%= link_to t('cookies_consent_choice'),
+                  '',
+                  class: 'sidebar-link js-gdpr__cookie_consent__display_again' %>
     </footer>
   </div>
 </nav>
diff --git a/app/views/admin/education/cohorts/_form.html.erb b/app/views/admin/education/cohorts/_form.html.erb
new file mode 100644
index 0000000000000000000000000000000000000000..c2ed9f568c4c8921e94b008ef4dedf0c803d7fb7
--- /dev/null
+++ b/app/views/admin/education/cohorts/_form.html.erb
@@ -0,0 +1,16 @@
+
+<%= simple_form_for(@education_cohort) do |f| %>
+  <%= f.error_notification %>
+  <%= f.error_notification message: f.object.errors[:base].to_sentence if f.object.errors[:base].present? %>
+
+  <div class="form-inputs">
+    <%= f.association :university %>
+    <%= f.association :program %>
+    <%= f.association :academic_year %>
+    <%= f.input :name %>
+  </div>
+
+  <div class="form-actions">
+    <%= f.button :submit %>
+  </div>
+<% end %>
diff --git a/app/views/admin/education/cohorts/edit.html.erb b/app/views/admin/education/cohorts/edit.html.erb
new file mode 100644
index 0000000000000000000000000000000000000000..8bbbfe709d3ef787e5406d1db79ea8d70b31cd91
--- /dev/null
+++ b/app/views/admin/education/cohorts/edit.html.erb
@@ -0,0 +1,6 @@
+<h1>Editing Education Cohort</h1>
+
+<%= render 'form', education_cohort: @education_cohort %>
+
+<%= link_to 'Show', @education_cohort %> |
+<%= link_to 'Back', education_cohorts_path %>
diff --git a/app/views/admin/education/cohorts/index.html.erb b/app/views/admin/education/cohorts/index.html.erb
new file mode 100644
index 0000000000000000000000000000000000000000..58267d3e37dd85cf37bd7658da0dcd0442435af2
--- /dev/null
+++ b/app/views/admin/education/cohorts/index.html.erb
@@ -0,0 +1,33 @@
+<p id="notice"><%= notice %></p>
+
+<h1>Education Cohorts</h1>
+
+<table>
+  <thead>
+    <tr>
+      <th>University</th>
+      <th>Program</th>
+      <th>Academic year</th>
+      <th>Name</th>
+      <th colspan="3"></th>
+    </tr>
+  </thead>
+
+  <tbody>
+    <% @education_cohorts.each do |education_cohort| %>
+      <tr>
+        <td><%= education_cohort.university_id %></td>
+        <td><%= education_cohort.program_id %></td>
+        <td><%= education_cohort.academic_year_id %></td>
+        <td><%= education_cohort.name %></td>
+        <td><%= link_to 'Show', education_cohort %></td>
+        <td><%= link_to 'Edit', edit_education_cohort_path(education_cohort) %></td>
+        <td><%= link_to 'Destroy', education_cohort, method: :delete, data: { confirm: 'Are you sure?' } %></td>
+      </tr>
+    <% end %>
+  </tbody>
+</table>
+
+<br>
+
+<%= link_to 'New Education Cohort', new_education_cohort_path %>
diff --git a/app/views/admin/education/cohorts/new.html.erb b/app/views/admin/education/cohorts/new.html.erb
new file mode 100644
index 0000000000000000000000000000000000000000..7ac730f56df5d2ff85434b24e392f444c24d1918
--- /dev/null
+++ b/app/views/admin/education/cohorts/new.html.erb
@@ -0,0 +1,5 @@
+<h1>New Education Cohort</h1>
+
+<%= render 'form', education_cohort: @education_cohort %>
+
+<%= link_to 'Back', education_cohorts_path %>
diff --git a/app/views/admin/education/cohorts/show.html.erb b/app/views/admin/education/cohorts/show.html.erb
new file mode 100644
index 0000000000000000000000000000000000000000..e28142b11e21ae5eecba90ff87c32320035a3ad6
--- /dev/null
+++ b/app/views/admin/education/cohorts/show.html.erb
@@ -0,0 +1,24 @@
+<p id="notice"><%= notice %></p>
+
+<p>
+  <strong>University:</strong>
+  <%= @education_cohort.university_id %>
+</p>
+
+<p>
+  <strong>Program:</strong>
+  <%= @education_cohort.program_id %>
+</p>
+
+<p>
+  <strong>Academic year:</strong>
+  <%= @education_cohort.academic_year_id %>
+</p>
+
+<p>
+  <strong>Name:</strong>
+  <%= @education_cohort.name %>
+</p>
+
+<%= link_to 'Edit', edit_education_cohort_path(@education_cohort) %> |
+<%= link_to 'Back', education_cohorts_path %>
diff --git a/app/views/admin/university/organization/imports/index.html.erb b/app/views/admin/university/organization/imports/index.html.erb
index 7a1332c6f3bd64a31f9c676bc5fcf1dd878484b3..f06991e44d360ce2c18927d7cb4c75eda6ded0a5 100644
--- a/app/views/admin/university/organization/imports/index.html.erb
+++ b/app/views/admin/university/organization/imports/index.html.erb
@@ -16,6 +16,7 @@
     <% end %>
   </tbody>
 </table>
+
 <% content_for :action_bar_right do %>
   <%= create_link University::Organization::Import %>
 <% end %>
diff --git a/app/views/admin/university/organization/imports/new.html.erb b/app/views/admin/university/organization/imports/new.html.erb
index 78275482c062794c14cc211f87d3dc1662d96798..714bdfc7a59596dc1bfe3f15714f3a625506341b 100644
--- a/app/views/admin/university/organization/imports/new.html.erb
+++ b/app/views/admin/university/organization/imports/new.html.erb
@@ -5,8 +5,8 @@
     <p>
       Les données doivent être au format csv, comme l'exemple suivant.<br>
       La première ligne doit être dédiée aux entêtes.<br>
+      Les noms des entêtes sont obligatoires et doivent être respectés strictement.<br>
       Le champ name est obligatoire.<br>
-      Les noms des entêtes sont indicatifs, l'import est basé sur la position des champs.<br>
       Les caractères doivent être encodés en UTF-8.<br>
       Les valeurs possibles pour kind sont : company, non_profit, government.
     </p>
diff --git a/app/views/admin/university/people/_form.html.erb b/app/views/admin/university/people/_form.html.erb
index 21eb1486ae4133c7ec0d927773206b4406e57c3a..f2a1bd72ac8c6c23abc4d2bcfabb0215ebbfe217 100644
--- a/app/views/admin/university/people/_form.html.erb
+++ b/app/views/admin/university/people/_form.html.erb
@@ -51,9 +51,10 @@
             <div class="col-md-6">
               <%= f.input :is_teacher %>
               <%= f.input :is_administration %>
-              <%= f.input :tenure %>
+              <%= f.input :is_alumnus %>
             </div>
             <div class="col-md-6">
+              <%= f.input :tenure %>
               <%= f.input :is_researcher %>
               <%= f.input :habilitation %>
             </div>
diff --git a/app/views/admin/university/person/alumni/_list.html.erb b/app/views/admin/university/person/alumni/_list.html.erb
new file mode 100644
index 0000000000000000000000000000000000000000..0d3945bd5fcedc5a27458100b0105be9170e19c4
--- /dev/null
+++ b/app/views/admin/university/person/alumni/_list.html.erb
@@ -0,0 +1,30 @@
+<table class="<%= table_classes %>">
+  <thead>
+    <tr>
+      <th><%= University::Person.human_attribute_name('last_name') %></th>
+      <th><%= University::Person.human_attribute_name('first_name') %></th>
+      <th></th>
+    </tr>
+  </thead>
+  <tbody>
+    <% alumni.each do |alumnus| %>
+      <% path = admin_university_person_alumnus_path(alumnus) %>
+      <tr>
+        <td><%= link_to alumnus.last_name, path %></td>
+        <td><%= link_to alumnus.first_name, path %></td>
+        <td class="text-end">
+          <div class="btn-group" role="group">
+            <%= link_to t('edit'),
+                      edit_admin_university_person_alumnus_path(alumnus),
+                      class: button_classes if can?(:update, alumnus) %>
+            <%= link_to t('delete'),
+                      path,
+                      method: :delete,
+                      data: { confirm: t('please_confirm') },
+                      class: button_classes_danger if can?(:destroy, alumnus) %>
+          </div>
+        </td>
+      </tr>
+    <% end %>
+  </tbody>
+</table>
diff --git a/app/views/admin/university/person/alumni/index.html.erb b/app/views/admin/university/person/alumni/index.html.erb
new file mode 100644
index 0000000000000000000000000000000000000000..00d3f196195d3103f33c70e6cfc5760fc41a8e91
--- /dev/null
+++ b/app/views/admin/university/person/alumni/index.html.erb
@@ -0,0 +1,14 @@
+<% content_for :title, "#{University::Person::Alumnus.model_name.human(count: 2)} (#{@alumni.total_count})" %>
+
+<%= render 'admin/university/person/alumni/list', alumni: @alumni %>
+
+<%= paginate @alumni, theme: 'bootstrap-5' %>
+
+<% content_for :action_bar_left do %>
+  <%= link_to t('import'),
+              admin_university_person_alumnus_imports_path,
+              class: button_classes if can? :manage, University::Person::Alumnus::Import %>
+<% end %>
+
+<% content_for :action_bar_right do %>
+<% end %>
diff --git a/app/views/admin/university/person/alumni/show.html.erb b/app/views/admin/university/person/alumni/show.html.erb
new file mode 100644
index 0000000000000000000000000000000000000000..a82ea88ffa2827c84ac4aa00f8540ab2809c0c45
--- /dev/null
+++ b/app/views/admin/university/person/alumni/show.html.erb
@@ -0,0 +1 @@
+<% content_for :title, @alumnus %>
diff --git a/app/views/admin/university/person/alumnus/imports/index.html.erb b/app/views/admin/university/person/alumnus/imports/index.html.erb
new file mode 100644
index 0000000000000000000000000000000000000000..d5374b64929038ce56c32fa4c9d5c002d7e2bb20
--- /dev/null
+++ b/app/views/admin/university/person/alumnus/imports/index.html.erb
@@ -0,0 +1,22 @@
+<% content_for :title, University::Person::Alumnus::Import.model_name.human(count: 2) %>
+
+<table class="<%= table_classes %>">
+  <thead>
+    <tr>
+      <th><%= University::Person::Alumnus::Import.human_attribute_name('name') %></th>
+      <th><%= University::Person::Alumnus::Import.human_attribute_name('lines') %></th>
+    </tr>
+  </thead>
+  <tbody>
+    <% @imports.each do |import| %>
+      <tr>
+        <td><%= link_to import, [:admin, import] %></td>
+        <td><%= import.lines %></td>
+      </tr>
+    <% end %>
+  </tbody>
+</table>
+
+<% content_for :action_bar_right do %>
+  <%= create_link University::Person::Alumnus::Import %>
+<% end %>
diff --git a/app/views/admin/university/person/alumnus/imports/new.html.erb b/app/views/admin/university/person/alumnus/imports/new.html.erb
new file mode 100644
index 0000000000000000000000000000000000000000..e63db9d069141d622bfd79e194462e11313f2600
--- /dev/null
+++ b/app/views/admin/university/person/alumnus/imports/new.html.erb
@@ -0,0 +1,103 @@
+<% content_for :title, University::Organization::Import.model_name.human %>
+
+<div class="row">
+  <div class="col-md-6">
+    <p>
+      Les données doivent être au format csv, comme l'exemple suivant.<br>
+      La première ligne doit être dédiée aux entêtes.<br>
+      Les noms des entêtes sont obligatoires et doivent être respectés strictement.<br>
+      Les champs marqués d'une astérisque sont obligatoires.<br>
+      Les caractères doivent être encodés en UTF-8.
+      Les valeurs pour gender peuvent être m (masculin), m (féminin) et n (non binaire).
+    </p>
+    <%= simple_form_for [:admin, @import] do |f| %>
+      <%= f.input :file %>
+      <% content_for :action_bar_right do %>
+        <%= submit f %>
+      <% end %>
+    <% end %>
+
+  </div>
+  <div class="col-md-6">
+    <table class="table table-small">
+      <tbody>
+        <tr>
+          <th>program*</th>
+          <td>c6b78fac-0a5f-4c44-ad22-4ee68ed382bb</td>
+        </tr>
+        <tr>
+          <th>year*</th>
+          <td>2022</td>
+        </tr>
+        <tr>
+          <th>first_name*</th>
+          <td>Stéphane</td>
+        </tr>
+        <tr>
+          <th>last_name*</th>
+          <td>Dupond</td>
+        </tr>
+        <tr>
+          <th>gender</th>
+          <td>m</td>
+        </tr>
+        <tr>
+          <th>birth</th>
+          <td>2006-05-26</td>
+        </tr>
+        <tr>
+          <th>mail</th>
+          <td>jean.dupont@gmail.com</td>
+        </tr>
+        <tr>
+          <th>photo</th>
+          <td>https://www.example.com/photo.jpg</td>
+        </tr>
+        <tr>
+          <th>url</th>
+          <td>https://www.stephanedupond.fr</td>
+        </tr>
+        <tr>
+          <th>phonepro</th>
+          <td>+33 1 01 01 01 01</td>
+        </tr>
+        <tr>
+          <th>phoneperso</th>
+          <td>+33 1 01 01 01 01</td>
+        </tr>
+        <tr>
+          <th>mobile</th>
+          <td>+33 6 01 01 01 01</td>
+        </tr>
+        <tr>
+          <th>address</th>
+          <td>1 rue Jacques Ellul</td>
+        </tr>
+        <tr>
+          <th>zipcode</th>
+          <td>33000</td>
+        </tr>
+        <tr>
+          <th>city</th>
+          <td>Bordeaux</td>
+        </tr>
+        <tr>
+          <th>country</th>
+          <td>FR</td>
+        </tr>
+        <tr>
+          <th>status</th>
+          <td>Product Designer</td>
+        </tr>
+        <tr>
+          <th>socialtwitter</th>
+          <td>stephanedupond</td>
+        </tr>
+        <tr>
+          <th>sociallinkedin</th>
+          <td>https://www.linkedin.com/in/stephanedupond</td>
+        </tr>
+      </tbody>
+    </table>
+  </div>
+</div>
diff --git a/app/views/admin/university/person/alumnus/imports/show.html.erb b/app/views/admin/university/person/alumnus/imports/show.html.erb
new file mode 100644
index 0000000000000000000000000000000000000000..d3fec028c9c4e6dd9403ec227b056fc50b0f49ab
--- /dev/null
+++ b/app/views/admin/university/person/alumnus/imports/show.html.erb
@@ -0,0 +1,7 @@
+<% content_for :title, @import %>
+
+<p><%= @import.lines %> lines</p>
+
+<%= link_to "#{@import.file.filename} (#{number_to_human_size @import.file.byte_size})",
+            url_for(@import.file),
+            class: button_classes %>
diff --git a/app/views/api/dashboard/index.html.erb b/app/views/api/dashboard/index.html.erb
new file mode 100644
index 0000000000000000000000000000000000000000..5a1d57b884ae12719e7d6a0fc919cabece14e5f6
--- /dev/null
+++ b/app/views/api/dashboard/index.html.erb
@@ -0,0 +1,5 @@
+<% content_for :title, 'API' %>
+
+<h2>Lhéo</h2>
+<p>Langage Harmonisé d'Échange d'informations sur l'Offre de formation</p>
+<%= link_to 'API Lhéo', api_lheo_path %>
diff --git a/app/views/api/layouts/application.html.erb b/app/views/api/layouts/application.html.erb
new file mode 100644
index 0000000000000000000000000000000000000000..e7e4b164644aed76f496b20f404fbbb2eb4c4058
--- /dev/null
+++ b/app/views/api/layouts/application.html.erb
@@ -0,0 +1,29 @@
+<!DOCTYPE html>
+<html>
+  <head>
+    <meta charset="utf-8">
+    <meta name="viewport" content="width=device-width, initial-scale=1, shrink-to-fit=no">
+    <title><%= yield :title %></title>
+    <%= csrf_meta_tags %>
+    <%= csp_meta_tag %>
+    <%= stylesheet_link_tag 'application', media: 'all' %>
+    <%= javascript_include_tag 'application' %>
+    <%= favicon_link_tag 'favicon.png' %>
+  </head>
+  <body class="<%= body_classes %>">
+    <nav class="navbar navbar-light">
+      <div class="container">
+        <a class="navbar-brand" href="<%= api_root_path %>">
+          <%= render 'logo' %>
+        </a>
+      </div>
+    </nav>
+    <main class="container mt-5">
+      <h1><%= yield :title %></h1>
+      <%= yield %>
+    </main>
+    <%= render 'footer' %>
+    <%= render 'gdpr/cookie_consent' %>
+    <%= render 'bugsnag' %>
+  </body>
+</html>
diff --git a/app/views/api/lheo/index.xml.erb b/app/views/api/lheo/index.xml.erb
new file mode 100644
index 0000000000000000000000000000000000000000..7afb6d587da31f8d71bbf74288237a6685c7d16c
--- /dev/null
+++ b/app/views/api/lheo/index.xml.erb
@@ -0,0 +1,23 @@
+<?xml version="1.0" encoding="utf-8"?>
+<lheo xmlns="http://lheo.gouv.fr/2.3">
+  <offres>
+    <% @programs.each do |program| %>
+      <formation>
+        <domaine-formation>
+          <code-FORMACODE>...</code-FORMACODE>
+          <code-NSF>...</code-NSF>
+          <code-ROME>...</code-ROME>
+          <extras>N</extras>
+        </domaine-formation>
+        <intitule-formation><![CDATA[<%= program.name %>]]></intitule-formation>
+        <% if program.schools.any? %>
+        <nom-organisme><![CDATA[<%= program.schools.first.name %>]]></nom-organisme>
+        <% end %>
+        <objectif-formation><![CDATA[<%= program.best_objectives %>]]></objectif-formation>
+        <resultats-attendus><![CDATA[<%= program.best_results %>]]></resultats-attendus>
+        <contenu-formation><![CDATA[<%= program.best_content %>]]></contenu-formation>
+        <certifiante>1</certifiante>
+      </formation>
+    <% end %>
+  </offres>
+</lheo>
diff --git a/app/views/application/_footer.html.erb b/app/views/application/_footer.html.erb
index 8f8dfbf0dacda42e964329b7f5af7b5634b12825..530f5a966f9b8b3f19be1e545ee2823448ba94cd 100644
--- a/app/views/application/_footer.html.erb
+++ b/app/views/application/_footer.html.erb
@@ -1,11 +1,27 @@
-<footer class="text-center small">
-  <div class="mb-5">
-    <%= image_tag 'osuny-black.svg', width: 80 %>
+<footer class="pt-5">
+  <div class="container text-center">
+    <div class="mb-5">
+      <%= image_tag 'osuny-black.svg', width: 80 %>
+    </div>
+    <nav class="nav justify-content-center">
+      <%= link_to t('terms_of_service'),
+                  t('terms_of_service_url'),
+                  class: 'nav-link',
+                  target: '_blank',
+                  rel: 'noreferrer' %>
+      <%= link_to t('privacy_policy'),
+                  t('privacy_policy_url'),
+                  class: 'nav-link',
+                  target: '_blank',
+                  rel: 'noreferrer' %>
+      <%= link_to t('cookies_policy'),
+                  t('cookies_policy_url'),
+                  class: 'nav-link',
+                  target: '_blank',
+                  rel: 'noreferrer' %>
+      <%= link_to t('cookies_consent_choice'),
+                  '',
+                  class: 'nav-link js-gdpr__cookie_consent__display_again' %>
+    </nav>
   </div>
-  <%= current_university %> |
-  <%= link_to t('terms_of_service'), t('terms_of_service_url'), target: '_blank', rel: 'noreferrer' %> |
-  <%= link_to t('privacy_policy'), t('privacy_policy_url'), target: '_blank', rel: 'noreferrer' %> |
-  <%= link_to t('cookies_policy'), t('cookies_policy_url'), target: '_blank', rel: 'noreferrer' %> |
-  <%= link_to t('cookies_consent_choice'), '', class: 'js-gdpr__cookie_consent__display_again' %>
-
 </footer>
diff --git a/app/views/application/_nav.html.erb b/app/views/application/_nav.html.erb
index 8d4c112e49109312be2c3e8c87ee137a0e8ad5b4..2263f5391bed540bafaa3eade05bbf71e31dc911 100644
--- a/app/views/application/_nav.html.erb
+++ b/app/views/application/_nav.html.erb
@@ -3,5 +3,6 @@
     <a class="navbar-brand" href="/">
       <%= render 'logo' %>
     </a>
+    <%= render_navigation %>
   </div>
 </nav>
diff --git a/app/views/extranet/academic_years/index.html.erb b/app/views/extranet/academic_years/index.html.erb
new file mode 100644
index 0000000000000000000000000000000000000000..6c079cc94e1f15e56a8898ffd139820ffb5ee012
--- /dev/null
+++ b/app/views/extranet/academic_years/index.html.erb
@@ -0,0 +1,26 @@
+<% content_for :title, Education::AcademicYear.model_name.human(count: 2) %>
+
+<header class="mb-5">
+  <h1><%= Education::AcademicYear.model_name.human(count: 2) %></h1>
+</header>
+
+<div class="row">
+  <% @academic_years.each do |year| %>
+    <div class="col-md-3">
+      <article class="card mb-4">
+        <div class="card-body">
+          <h2 class="mb-5">
+            <%= link_to year, year, class: 'stretched-link' %>
+          </h2>
+          <p class="text-end mb-0">
+            <%= year.cohorts.count %>
+            <%= Education::Cohort.model_name.human(count: year.cohorts.count).downcase %>
+            <br>
+            <%= year.people.count %>
+            <%= University::Person::Alumnus.model_name.human(count: year.people.count).downcase %>
+          </p>
+        </div>
+      </article>
+    </div>
+  <% end %>
+</div>
diff --git a/app/views/extranet/academic_years/show.html.erb b/app/views/extranet/academic_years/show.html.erb
new file mode 100644
index 0000000000000000000000000000000000000000..b4db00980201f886b562422faa0a6b85f71bcb44
--- /dev/null
+++ b/app/views/extranet/academic_years/show.html.erb
@@ -0,0 +1,17 @@
+<% content_for :title, @academic_year %>
+
+<header class="mb-5">
+  <h1><%= @academic_year %></h1>
+</header>
+
+<h2 class="mb-4">
+  <%= @academic_year.cohorts.count %>
+  <%= Education::Cohort.model_name.human(count: @academic_year.cohorts.count).downcase %>
+</h2>
+<%= render 'extranet/cohorts/list', cohorts: @academic_year.cohorts %>
+
+<h2 class="mt-5 mb-4">
+  <%= @academic_year.people.count %>
+  <%= University::Person::Alumnus.model_name.human(count: @academic_year.people.count).downcase %>
+</h2>
+<%= render 'extranet/persons/list', people: @academic_year.people %>
diff --git a/app/views/extranet/application/_footer.html.erb b/app/views/extranet/application/_footer.html.erb
new file mode 100644
index 0000000000000000000000000000000000000000..309bb5e8b5f38c341ef30f3c17caec72efa0082b
--- /dev/null
+++ b/app/views/extranet/application/_footer.html.erb
@@ -0,0 +1,27 @@
+<footer class="pt-5">
+  <div class="container text-center">
+    <div class="mb-5">
+      <%= render 'logo' %>
+    </div>
+    <nav class="nav justify-content-center">
+      <%= link_to t('terms_of_service'),
+                  t('terms_of_service_url'),
+                  class: 'nav-link',
+                  target: '_blank',
+                  rel: 'noreferrer' %>
+      <%= link_to t('privacy_policy'),
+                  t('privacy_policy_url'),
+                  class: 'nav-link',
+                  target: '_blank',
+                  rel: 'noreferrer' %>
+      <%= link_to t('cookies_policy'),
+                  t('cookies_policy_url'),
+                  class: 'nav-link',
+                  target: '_blank',
+                  rel: 'noreferrer' %>
+      <%= link_to t('cookies_consent_choice'),
+                  '',
+                  class: 'nav-link js-gdpr__cookie_consent__display_again' %>
+    </nav>
+  </div>
+</footer>
diff --git a/app/views/extranet/application/_logo.html.erb b/app/views/extranet/application/_logo.html.erb
new file mode 100644
index 0000000000000000000000000000000000000000..210b239329d5eae639d60c4a97cf3824e7549f65
--- /dev/null
+++ b/app/views/extranet/application/_logo.html.erb
@@ -0,0 +1,5 @@
+<% if current_context.logo.attached? %>
+  <%= image_tag current_context.logo, width: 200, alt: current_context.to_s, class: 'logo' %>
+<% else %>
+  <%= current_context %>
+<% end %>
diff --git a/app/views/extranet/application/_nav.html.erb b/app/views/extranet/application/_nav.html.erb
new file mode 100644
index 0000000000000000000000000000000000000000..2263f5391bed540bafaa3eade05bbf71e31dc911
--- /dev/null
+++ b/app/views/extranet/application/_nav.html.erb
@@ -0,0 +1,8 @@
+<nav class="navbar navbar-light">
+  <div class="container">
+    <a class="navbar-brand" href="/">
+      <%= render 'logo' %>
+    </a>
+    <%= render_navigation %>
+  </div>
+</nav>
diff --git a/app/views/extranet/cohorts/_list.html.erb b/app/views/extranet/cohorts/_list.html.erb
new file mode 100644
index 0000000000000000000000000000000000000000..dee24aecaa565215d54598631b5a4011319d6bc7
--- /dev/null
+++ b/app/views/extranet/cohorts/_list.html.erb
@@ -0,0 +1,17 @@
+<div class="row">
+  <% cohorts.each do |cohort| %>
+    <div class="col-md-6">
+      <article class="card mb-4">
+        <div class="card-body">
+          <h2 class="mb-5">
+            <%= link_to cohort, cohort, class: 'stretched-link' %>
+          </h2>
+          <p class="text-end mb-0">
+            <%= cohort.people.count %>
+            <%= University::Person::Alumnus.model_name.human(count: cohort.people.count).downcase %>
+          </p>
+        </div>
+      </article>
+    </div>
+  <% end %>
+</div>
diff --git a/app/views/extranet/cohorts/index.html.erb b/app/views/extranet/cohorts/index.html.erb
new file mode 100644
index 0000000000000000000000000000000000000000..bf32bf86b3ee261df48e4ae5ec4655aba0d197b0
--- /dev/null
+++ b/app/views/extranet/cohorts/index.html.erb
@@ -0,0 +1,7 @@
+<% content_for :title, Education::Cohort.model_name.human(count: 2) %>
+
+<header class="mb-5">
+  <h1><%= Education::Cohort.model_name.human(count: 2) %></h1>
+</header>
+
+<%= render 'extranet/cohorts/list', cohorts: @cohorts %>
diff --git a/app/views/extranet/cohorts/show.html.erb b/app/views/extranet/cohorts/show.html.erb
new file mode 100644
index 0000000000000000000000000000000000000000..69da25296c024df6bac2a231ff2aa154941c349f
--- /dev/null
+++ b/app/views/extranet/cohorts/show.html.erb
@@ -0,0 +1,17 @@
+<% content_for :title, @cohort %>
+
+<header class="mb-5">
+  <div class="row">
+    <div class="col-md-8">
+      <h1><%= @cohort %></h1>
+    </div>
+    <div class="col-md-4 text-end">
+      <p>
+        <%= @cohort.people.count %>
+        <%= University::Person::Alumnus.model_name.human(count: @cohort.people.count).downcase %>
+      </p>
+    </div>
+  </div>
+</header>
+
+<%= render 'extranet/persons/list', people: @cohort.people %>
diff --git a/app/views/extranet/home/index.html.erb b/app/views/extranet/home/index.html.erb
new file mode 100644
index 0000000000000000000000000000000000000000..d2c5b60451a61c62cf6f8a2e14feaaa97584e9cb
--- /dev/null
+++ b/app/views/extranet/home/index.html.erb
@@ -0,0 +1,5 @@
+<% content_for :title, current_context %>
+
+<h2><%= Education::Cohort.model_name.human(count: 2) %></h2>
+<p><%= link_to 'Voir toutes les promotions', education_cohorts_path %></p>
+<%= render 'extranet/cohorts/list', cohorts: @cohorts %>
diff --git a/app/views/extranet/layouts/application.html.erb b/app/views/extranet/layouts/application.html.erb
new file mode 100644
index 0000000000000000000000000000000000000000..74bf7b45c19e463fb5e223fc27223bc4536c7a61
--- /dev/null
+++ b/app/views/extranet/layouts/application.html.erb
@@ -0,0 +1,23 @@
+<!DOCTYPE html>
+<html>
+  <head>
+    <meta charset="utf-8">
+    <meta name="viewport" content="width=device-width, initial-scale=1, shrink-to-fit=no">
+    <title><%= yield :title %></title>
+    <%= csrf_meta_tags %>
+    <%= csp_meta_tag %>
+    <%= stylesheet_link_tag 'application', media: 'all' %>
+    <%= javascript_include_tag 'application' %>
+    <%= favicon_link_tag 'favicon.png' %>
+  </head>
+  <body class="extranet <%= body_classes %>">
+    <%= render 'extranet/application/nav' %>
+    <main class="container">
+      <%= render_breadcrumbs builder: Appstack::BreadcrumbsOnRailsBuilder if breadcrumbs.any? %>
+      <%= yield %>
+    </main>
+    <%= render 'extranet/application/footer' %>
+    <%= render 'gdpr/cookie_consent' %>
+    <%= render 'bugsnag' %>
+  </body>
+</html>
diff --git a/app/views/university/organizations/index.html.erb b/app/views/extranet/organizations/index.html.erb
similarity index 77%
rename from app/views/university/organizations/index.html.erb
rename to app/views/extranet/organizations/index.html.erb
index e8810d4f6a92b7c3bdb775d7b904d2644736a2b3..5796718de6abe4e3aee343cff839fdb3ac81f74a 100644
--- a/app/views/university/organizations/index.html.erb
+++ b/app/views/extranet/organizations/index.html.erb
@@ -1,5 +1,9 @@
 <% content_for :title, University::Organization.model_name.human(count: 2) %>
 
+<header class="mb-5">
+  <h1><%= University::Organization.model_name.human(count: 2) %></h1>
+</header>
+
 <table class="<%= table_classes %>">
   <thead>
     <tr>
@@ -16,3 +20,4 @@
     <% end %>
   </tbody>
 </table>
+<%= paginate @organizations, theme: 'bootstrap-5' %>
diff --git a/app/views/extranet/organizations/show.html.erb b/app/views/extranet/organizations/show.html.erb
new file mode 100644
index 0000000000000000000000000000000000000000..0794ba8fd7b373246a382b9e2001f33240b492fe
--- /dev/null
+++ b/app/views/extranet/organizations/show.html.erb
@@ -0,0 +1,5 @@
+<% content_for :title, @organization %>
+
+<header class="mb-5">
+  <h1><%= @organization %></h1>
+</header>
diff --git a/app/views/extranet/persons/_list.html.erb b/app/views/extranet/persons/_list.html.erb
new file mode 100644
index 0000000000000000000000000000000000000000..577ec80f7ff386520578e4c2ae242985d3094af9
--- /dev/null
+++ b/app/views/extranet/persons/_list.html.erb
@@ -0,0 +1,23 @@
+<%
+people_paged = people.ordered.page(params[:page]).per(60)
+%>
+<div class="row">
+  <% people_paged.each do |person| %>
+    <div class="col-xxl-2 col-md-3">
+      <article class="mb-4 position-relative">
+        <% if person.picture.attached? %>
+          <%= kamifusen_tag person.picture, width: 400, class: 'img-fluid' %>
+        <% else %>
+          <%= image_tag 'extranet/avatar.png', width: 400, class: 'img-fluid' %>
+        <% end %>
+        <%= link_to person, class: 'stretched-link' do %>
+          <%= person.first_name %>
+          <b>
+            <%= person.last_name %>
+          </b>
+        <% end %>
+      </article>
+    </div>
+  <% end %>
+</div>
+<%= paginate people_paged, theme: 'bootstrap-5' %>
diff --git a/app/views/extranet/persons/index.html.erb b/app/views/extranet/persons/index.html.erb
new file mode 100644
index 0000000000000000000000000000000000000000..f12ba91324d414923362034d30c58f500749eb0c
--- /dev/null
+++ b/app/views/extranet/persons/index.html.erb
@@ -0,0 +1,17 @@
+<% content_for :title, University::Person::Alumnus.model_name.human(count: 2) %>
+
+<header class="mb-5">
+  <div class="row">
+    <div class="col-md-9">
+      <h1><%= University::Person::Alumnus.model_name.human(count: 2) %></h1>
+    </div>
+    <div class="col-md-3 text-end">
+      <p>
+        <%= @people.count %>
+        <%= University::Person::Alumnus.model_name.human(count: @people.count).downcase %>
+      </p>
+    </div>
+  </div>
+</header>
+
+<%= render 'extranet/persons/list', people: @people %>
diff --git a/app/views/extranet/persons/show.html.erb b/app/views/extranet/persons/show.html.erb
new file mode 100644
index 0000000000000000000000000000000000000000..9cedca1d9e3674e7d8de6f21188620fabfdf2f15
--- /dev/null
+++ b/app/views/extranet/persons/show.html.erb
@@ -0,0 +1,62 @@
+<% content_for :title, @person %>
+
+<div class="row mb-5">
+  <div class="col-md-9">
+    <header style="height: 100%">
+      <h1><%= @person %></h1>
+    </header>
+  </div>
+  <div class="col-md-3">
+    <% if @person.picture.attached? %>
+      <%= kamifusen_tag @person.picture, width: 400, class: 'img-fluid' %>
+    <% else %>
+      <%= image_tag 'extranet/avatar.png', width: 400, class: 'img-fluid' %>
+    <% end %>
+  </div>
+</div>
+
+<div class="row">
+  <div class="col-md-6">
+    <p><%= @person.biography %></p>
+  </div>
+  <div class="offset-md-3 col-md-3">
+    <dl>
+      <% @person.cohorts.each do |cohort| %>
+        <dt><%= cohort.program %></dt>
+        <dd><%= link_to cohort.academic_year, cohort %></dd>
+      <% end %>
+      <% if @person.phone %>
+        <dt><%= University::Person.human_attribute_name(:phone) %></dt>
+        <dd><a href="tel:<%= @person.phone %>" target="_blank" rel="noreferrer"><%= @person.phone %></a></dd>
+      <% end %>
+      <% if @person.email %>
+        <dt><%= University::Person.human_attribute_name(:email) %></dt>
+        <dd><a href="mailto:<%= @person.email %>" target="_blank" rel="noreferrer"><%= @person.email %></a></dd>
+      <% end %>
+      <% if @person.url %>
+        <dt><%= University::Person.human_attribute_name(:url) %></dt>
+        <dd>
+          <a href="<%= social_website_to_url @person.url %>" target="_blank" rel="noreferrer">
+            <%= social_website_to_s @person.url %>
+          </a>
+        </dd>
+      <% end %>
+      <% if @person.linkedin %>
+        <dt><%= University::Person.human_attribute_name(:linkedin) %></dt>
+        <dd>
+          <a href="<%= social_linkedin_to_url @person.linkedin %>" target="_blank" rel="noreferrer">
+            <%= social_linkedin_to_s @person.linkedin %>
+          </a>
+        </dd>
+      <% end %>
+      <% if @person.twitter %>
+        <dt><%= University::Person.human_attribute_name(:twitter) %></dt>
+        <dd>
+          <a href="<%= social_twitter_to_url @person.twitter %>" target="_blank" rel="noreferrer">
+            <%= social_twitter_to_s @person.twitter %>
+          </a>
+        </dd>
+      <% end %>
+    </dl>
+  </div>
+</div>
diff --git a/app/views/home/index.html.erb b/app/views/home/index.html.erb
deleted file mode 100644
index 18a3f18dc2a5db7e90a76ac343be35d17a698676..0000000000000000000000000000000000000000
--- a/app/views/home/index.html.erb
+++ /dev/null
@@ -1 +0,0 @@
-<% content_for :title, current_context %>
diff --git a/app/views/layouts/devise.html.erb b/app/views/layouts/devise.html.erb
index e8bd461767905d1ea26d7739a4c50d5189aca512..1d4c61febad03503e020267c100bfca43418091a 100644
--- a/app/views/layouts/devise.html.erb
+++ b/app/views/layouts/devise.html.erb
@@ -6,8 +6,8 @@
     <title><%= yield :title %></title>
     <%= csrf_meta_tags %>
     <%= csp_meta_tag %>
-    <%= stylesheet_link_tag 'application', media: 'all' %>
-    <%= javascript_include_tag 'application' %>
+    <%= stylesheet_link_tag 'admin', media: 'all' %>
+    <%= javascript_include_tag 'admin' %>
     <%= favicon_link_tag 'favicon.png' %>
   </head>
   <body class="<%= body_classes %>">
diff --git a/app/views/university/organizations/show.html.erb b/app/views/university/organizations/show.html.erb
deleted file mode 100644
index 1a8858a8183c77fbb221e456277a8e24e78b282f..0000000000000000000000000000000000000000
--- a/app/views/university/organizations/show.html.erb
+++ /dev/null
@@ -1 +0,0 @@
-<% content_for :title, @organization %>
diff --git a/config/admin_navigation.rb b/config/admin_navigation.rb
index 4ad235c332b336ebd1ebf98761c62ef0d092d408..b2d084377f0c423b281c766a43c531a1d87449ed 100644
--- a/config/admin_navigation.rb
+++ b/config/admin_navigation.rb
@@ -10,6 +10,7 @@ SimpleNavigation::Configuration.run do |navigation|
       primary.item :university, University.model_name.human, nil, { kind: :header }
       primary.item :university, University::Person.model_name.human(count: 2), admin_university_people_path, { icon: 'users-cog' }
       primary.item :university, University::Organization.model_name.human(count: 2), admin_university_organizations_path, { icon: 'building' }
+      primary.item :communication_alumni, University::Person::Alumnus.model_name.human(count: 2), admin_university_person_alumni_path, { icon: 'users' }
     end
 
     if can?(:read, Education::Program)
@@ -35,7 +36,6 @@ SimpleNavigation::Configuration.run do |navigation|
       primary.item :communication_websites, Communication::Website.model_name.human(count: 2), admin_communication_websites_path, { icon: 'sitemap' } if can?(:read, Communication::Website)
       primary.item :communication_extranets, Communication::Extranet.model_name.human(count: 2), admin_communication_extranets_path, { icon: 'project-diagram' }
       primary.item :communication_newsletters, 'Lettres d\'information', nil, { icon: 'envelope' }
-      primary.item :communication_alumni, 'Alumni', nil, { icon: 'users' }
     end
 
     if can?(:read, Administration::Qualiopi::Criterion)
diff --git a/config/initializers/inflections.rb b/config/initializers/inflections.rb
index 2f804f76171aff60b900d7dcf71294194cfbea26..e01ebddbf68f9fccf2d5b80ded0cb7f5eb2ca0fc 100644
--- a/config/initializers/inflections.rb
+++ b/config/initializers/inflections.rb
@@ -8,6 +8,7 @@ ActiveSupport::Inflector.inflections(:en) do |inflect|
   # inflect.singular /^(ox)en/i, '\1'
   # inflect.uncountable %w( fish sheep )
   inflect.irregular 'axis', 'axes'
+  inflect.irregular 'alumnus', 'alumni'
 end
 
 # These inflection rules are supported but not enabled by default:
diff --git a/config/initializers/mime_types.rb b/config/initializers/mime_types.rb
index dc1899682b01c3a6d9673faf746e235fb64fc4d2..1ce3a81f7e1d1cae513859231bea011311b7add5 100644
--- a/config/initializers/mime_types.rb
+++ b/config/initializers/mime_types.rb
@@ -2,3 +2,4 @@
 
 # Add new mime types for use in respond_to blocks:
 # Mime::Type.register "text/richtext", :rtf
+Mime::Type.register "application/xml", :xml, %w( text/xml application/x-xml )
diff --git a/config/locales/education/en.yml b/config/locales/education/en.yml
index d7d7ba80b698fd2c9f938ee5521dfcefb05c4494..02c60018f4e2213a6ee1e869f01d30c83210f957 100644
--- a/config/locales/education/en.yml
+++ b/config/locales/education/en.yml
@@ -4,6 +4,12 @@ en:
       education: Education
   activerecord:
     models:
+      education/academic_year:
+        one: Year
+        other: Years
+      education/cohort:
+        one: Cohort
+        other: Cohorts
       education/program:
         one: Program
         other: Programs
diff --git a/config/locales/education/fr.yml b/config/locales/education/fr.yml
index 0082a690ced00219aeb364a0955b39c465d7a0de..f7ff0e862554d026ccbc5939fc33339be8b3c848 100644
--- a/config/locales/education/fr.yml
+++ b/config/locales/education/fr.yml
@@ -4,6 +4,12 @@ fr:
       education: Enseignement
   activerecord:
     models:
+      education/academic_year:
+        one: Année
+        other: Années
+      education/cohort:
+        one: Promotion
+        other: Promotions
       education/program:
         one: Formation
         other: Formations
diff --git a/config/locales/university/en.yml b/config/locales/university/en.yml
index 9204e1417459ddd7190ad073d768e78e082c7dcc..10e33442220e2e96674b50d4367dcfe8790d9177 100644
--- a/config/locales/university/en.yml
+++ b/config/locales/university/en.yml
@@ -64,6 +64,9 @@ en:
       university/person/involvement:
         one: Involvement
         other: Involvements
+      university/person/alumnus:
+        one: Alumnus
+        other: Alumni
       university/organization:
         one: Third party
         other: Third parties
diff --git a/config/locales/university/fr.yml b/config/locales/university/fr.yml
index 418aefe390cd75ee2c928a8b9ea93c7d24487e0f..43b7cb9063ca2accd62039da43e86fc799700918 100644
--- a/config/locales/university/fr.yml
+++ b/config/locales/university/fr.yml
@@ -29,6 +29,7 @@ fr:
         is_author: Auteur·rice
         is_researcher: Chercheur·se
         is_teacher: Enseignant·e
+        is_alumnus: Alumnus
         last_name: Nom de famille
         linkedin: LinkedIn (URL)
         name: Nom
@@ -64,6 +65,9 @@ fr:
       university/person/involvement:
         one: Implication
         other: Implications
+      university/person/alumnus:
+        one: Alumnus
+        other: Alumni
       university/organization:
         one: Tierce partie
         other: Tierces parties
@@ -83,8 +87,9 @@ fr:
         is_author: "Écrit des articles pour les sites."
         is_researcher: "Écrit des articles dans des revues scientifiques."
         is_teacher: "Enseigne dans des formations."
+        is_alumnus: "Étudie ou a étudié dans l'établissement."
         linkedin: "Exemple : https://www.linkedin.com/in/osuny"
-        tenure: "A différencier d'une personne vacataire."
+        tenure: "À différencier d'une personne vacataire."
         twitter: "Exemple : osuny"
   university:
     invoice_informations: Données de facturation
diff --git a/config/navigation.rb b/config/navigation.rb
index 2a11b180ab319341afc9b55469a116f63048e92c..cb9701bc517107013309b64883701c420ee87d3b 100644
--- a/config/navigation.rb
+++ b/config/navigation.rb
@@ -1,81 +1,20 @@
-# -*- coding: utf-8 -*-
-# Configures your navigation
 SimpleNavigation::Configuration.run do |navigation|
-  # Specify a custom renderer if needed.
-  # The default renderer is SimpleNavigation::Renderer::List which renders HTML lists.
-  # The renderer can also be specified as option in the render_navigation call.
-  #navigation.renderer = Your::Custom::Renderer
-
-  # Specify the class that will be applied to active navigation items. Defaults to 'selected' 
-  #navigation.selected_class = 'selected'
-
-  # Specify the class that will be applied to the current leaf of
-  # active navigation items. Defaults to 'simple-navigation-active-leaf'
-  #navigation.active_leaf_class = 'simple-navigation-active-leaf'
-
-  # Specify if item keys are added to navigation items as id. Defaults to true
-  #navigation.autogenerate_item_ids = true
-
-  # You can override the default logic that is used to autogenerate the item ids.
-  # To do this, define a Proc which takes the key of the current item as argument.
-  # The example below would add a prefix to each key.
-  #navigation.id_generator = Proc.new {|key| "my-prefix-#{key}"}
-
-  # If you need to add custom html around item names, you can define a proc that
-  # will be called with the name you pass in to the navigation.
-  # The example below shows how to wrap items spans.
-  #navigation.name_generator = Proc.new {|name, item| tag.span(name) }
-
-  # Specify if the auto highlight feature is turned on (globally, for the whole navigation). Defaults to true
-  #navigation.auto_highlight = true
-  
-  # Specifies whether auto highlight should ignore query params and/or anchors when 
-  # comparing the navigation items with the current URL. Defaults to true 
-  #navigation.ignore_query_params_on_auto_highlight = true
-  #navigation.ignore_anchors_on_auto_highlight = true
-  
-  # If this option is set to true, all item names will be considered as safe (passed through html_safe). Defaults to false.
-  #navigation.consider_item_names_as_safe = false
-
-  # Define the primary navigation
+  navigation.renderer = ::SimpleNavigation::BootstrapRenderer
+  navigation.auto_highlight = true
+  navigation.highlight_on_subpath = true
+  navigation.selected_class = 'active'
   navigation.items do |primary|
-    # Add an item to the primary navigation. The following params apply:
-    # key - a symbol which uniquely defines your navigation item in the scope of the primary_navigation
-    # name - will be displayed in the rendered navigation. This can also be a call to your I18n-framework.
-    # url - the address that the generated item links to. You can also use url_helpers (named routes, restful routes helper, url_for etc.)
-    # options - can be used to specify attributes that will be included in the rendered navigation item (e.g. id, class etc.)
-    #           some special options that can be set:
-    #           :html - Specifies html attributes that will be included in the rendered navigation item
-    #           :if - Specifies a proc to call to determine if the item should
-    #                 be rendered (e.g. <tt>if: -> { current_user.admin? }</tt>). The
-    #                 proc should evaluate to a true or false value and is evaluated in the context of the view.
-    #           :unless - Specifies a proc to call to determine if the item should not
-    #                     be rendered (e.g. <tt>unless: -> { current_user.admin? }</tt>). The
-    #                     proc should evaluate to a true or false value and is evaluated in the context of the view.
-    #           :method - Specifies the http-method for the generated link - default is :get.
-    #           :highlights_on - if autohighlighting is turned off and/or you want to explicitly specify
-    #                            when the item should be highlighted, you can set a regexp which is matched
-    #                            against the current URI.  You may also use a proc, or the symbol <tt>:subpath</tt>.
-    #
-    primary.item :key_1, 'name', url, options
-
-    # Add an item which has a sub navigation (same params, but with block)
-    primary.item :key_2, 'name', url, options do |sub_nav|
-      # Add an item to the sub navigation (same params again)
-      sub_nav.item :key_2_1, 'name', url, options
-    end
-
-    # You can also specify a condition-proc that needs to be fullfilled to display an item.
-    # Conditions are part of the options. They are evaluated in the context of the views,
-    # thus you can use all the methods and vars you have available in the views.
-    primary.item :key_3, 'Admin', url, html: { class: 'special' }, if: -> { current_user.admin? }
-    primary.item :key_4, 'Account', url, unless: -> { logged_in? }
-
-    # you can also specify html attributes to attach to this particular level
-    # works for all levels of the menu
-    #primary.dom_attributes = {id: 'menu-id', class: 'menu-class'}
-
-    # You can turn off auto highlighting for a specific level
-    #primary.auto_highlight = false
+    primary.item  :person,
+                  University::Person::Alumnus.model_name.human(count: 2),
+                  university_persons_path
+    primary.item  :years,
+                  Education::AcademicYear.model_name.human(count: 2),
+                  education_academic_years_path
+    primary.item  :cohorts,
+                  Education::Cohort.model_name.human(count: 2),
+                  education_cohorts_path
+    primary.item  :organizations,
+                  University::Organization.model_name.human(count: 2),
+                  university_organizations_path
   end
 end
diff --git a/config/routes.rb b/config/routes.rb
index 7bf2ee0b0c0d5c65699cb0a2059c705454874c5b..5f10a141b009ccfa85992cf44ce2902944570d54 100644
--- a/config/routes.rb
+++ b/config/routes.rb
@@ -29,9 +29,9 @@ Rails.application.routes.draw do
     root to: 'dashboard#index'
   end
 
-  draw 'extranet'
-
   get '/media/:signed_id/:filename_with_transformations' => 'media#show', as: :medium
 
-  root to: 'home#index'
+  draw 'api'
+  draw 'extranet'
+  # Root is in extranet
 end
diff --git a/config/routes/admin/communication.rb b/config/routes/admin/communication.rb
index 7b07d35b56ea04d6dd3a83c853adea8dca3a1a1b..8b65d9c852c44bf5d5576cea3d8d52e28d18df98 100644
--- a/config/routes/admin/communication.rb
+++ b/config/routes/admin/communication.rb
@@ -57,4 +57,9 @@ namespace :communication do
     end
   end
   resources :extranets, controller: 'extranets'
+  resources :alumni do
+    collection do
+      resources :imports, only: [:index, :show, :new, :create]
+    end
+  end
 end
diff --git a/config/routes/admin/education.rb b/config/routes/admin/education.rb
index 9ef1f7832bbd33b22743d3c862af8178d96d0794..f4ae939d306a4de9b3b1eb6a8d3577fc10e9ec41 100644
--- a/config/routes/admin/education.rb
+++ b/config/routes/admin/education.rb
@@ -31,4 +31,6 @@ namespace :education do
       get :children
     end
   end
+  resources :academic_years
+  resources :cohorts
 end
diff --git a/config/routes/admin/university.rb b/config/routes/admin/university.rb
index cb94a97adfd364b3f68d1b59329de85b32ba0106..dedeb7272fe48278cfa7c9b4faae1b2eca7576af 100644
--- a/config/routes/admin/university.rb
+++ b/config/routes/admin/university.rb
@@ -3,4 +3,10 @@ namespace :university do
   namespace :organization do
     resources :imports, only: [:index, :show, :new, :create]
   end
+  namespace :person do
+    resources :alumni
+    namespace :alumnus do
+      resources :imports, only: [:index, :show, :new, :create]
+    end
+  end
 end
diff --git a/config/routes/api.rb b/config/routes/api.rb
new file mode 100644
index 0000000000000000000000000000000000000000..d711265d1559894b2bd9cff91f21fd78ab154abe
--- /dev/null
+++ b/config/routes/api.rb
@@ -0,0 +1,4 @@
+namespace :api do
+  get 'lheo' => 'lheo#index', defaults: { format: :xml }
+  root to: 'dashboard#index'
+end
diff --git a/config/routes/extranet.rb b/config/routes/extranet.rb
index 186b49d3851de3c596de301297fb06865336f7a7..2afd7680ebcedda96ad114d75de3886ba2a5a66f 100644
--- a/config/routes/extranet.rb
+++ b/config/routes/extranet.rb
@@ -1,6 +1,9 @@
-namespace :university do
-  resources :organizations, only: [:index, :show]
-end
-
-get 'organizations' => 'university/organizations#index', as: :organizations
-get 'organization/:id' => 'university/organizations#show', as: :organization
+get 'cohorts'           => 'extranet/cohorts#index', as: :education_cohorts
+get 'cohorts/:id'       => 'extranet/cohorts#show', as: :education_cohort
+get 'organizations'     => 'extranet/organizations#index', as: :university_organizations
+get 'organization/:id'  => 'extranet/organizations#show', as: :university_organization
+get 'persons'           => 'extranet/persons#index', as: :university_persons
+get 'persons/:id'       => 'extranet/persons#show', as: :university_person
+get 'years'             => 'extranet/academic_years#index', as: :education_academic_years
+get 'years/:id'         => 'extranet/academic_years#show', as: :education_academic_year
+root to: 'extranet/home#index'
diff --git a/db/migrate/20220316144230_create_education_academic_years.rb b/db/migrate/20220316144230_create_education_academic_years.rb
new file mode 100644
index 0000000000000000000000000000000000000000..25b594c36bf3d5573f9617668ff52f80da0ac92b
--- /dev/null
+++ b/db/migrate/20220316144230_create_education_academic_years.rb
@@ -0,0 +1,10 @@
+class CreateEducationAcademicYears < ActiveRecord::Migration[6.1]
+  def change
+    create_table :education_academic_years, id: :uuid do |t|
+      t.references :university, null: false, foreign_key: true, type: :uuid
+      t.integer :year
+
+      t.timestamps
+    end
+  end
+end
diff --git a/db/migrate/20220316153148_create_university_person_alumnus_imports.rb b/db/migrate/20220316153148_create_university_person_alumnus_imports.rb
new file mode 100644
index 0000000000000000000000000000000000000000..63518115812fad1cd450f76500aa786d0c9f50b2
--- /dev/null
+++ b/db/migrate/20220316153148_create_university_person_alumnus_imports.rb
@@ -0,0 +1,10 @@
+class CreateUniversityPersonAlumnusImports < ActiveRecord::Migration[6.1]
+  def change
+    create_table :university_person_alumnus_imports, id: :uuid do |t|
+      t.references :university, null: false, foreign_key: true, type: :uuid
+      t.references :user, null: false, foreign_key: true, type: :uuid
+
+      t.timestamps
+    end
+  end
+end
diff --git a/db/migrate/20220316155340_add_alumnus_to_university_person.rb b/db/migrate/20220316155340_add_alumnus_to_university_person.rb
new file mode 100644
index 0000000000000000000000000000000000000000..901ee02fbf82695f8fa3cc97c98aacf8665807d1
--- /dev/null
+++ b/db/migrate/20220316155340_add_alumnus_to_university_person.rb
@@ -0,0 +1,5 @@
+class AddAlumnusToUniversityPerson < ActiveRecord::Migration[6.1]
+  def change
+    add_column :university_people, :is_alumnus, :bool, default: false
+  end
+end
diff --git a/db/migrate/20220317150614_create_education_cohorts.rb b/db/migrate/20220317150614_create_education_cohorts.rb
new file mode 100644
index 0000000000000000000000000000000000000000..8794f5222f45aa882583466fb7947be7de93b112
--- /dev/null
+++ b/db/migrate/20220317150614_create_education_cohorts.rb
@@ -0,0 +1,12 @@
+class CreateEducationCohorts < ActiveRecord::Migration[6.1]
+  def change
+    create_table :education_cohorts, id: :uuid do |t|
+      t.references :university, null: false, foreign_key: true, type: :uuid
+      t.references :program, null: false, foreign_key: {to_table: :education_programs}, type: :uuid
+      t.references :academic_year, null: false, foreign_key: {to_table: :education_academic_years}, type: :uuid
+      t.string :name
+
+      t.timestamps
+    end
+  end
+end
diff --git a/db/migrate/20220317151819_create_join_table_education_cohort_university_person.rb b/db/migrate/20220317151819_create_join_table_education_cohort_university_person.rb
new file mode 100644
index 0000000000000000000000000000000000000000..1abfa95fca4f39bd95c21e8d52567f2401a6457f
--- /dev/null
+++ b/db/migrate/20220317151819_create_join_table_education_cohort_university_person.rb
@@ -0,0 +1,8 @@
+class CreateJoinTableEducationCohortUniversityPerson < ActiveRecord::Migration[6.1]
+  def change
+    create_join_table :education_cohorts, :university_people, column_options: {type: :uuid} do |t|
+      t.index [:education_cohort_id, :university_person_id], name: 'index_cohort_person'
+      t.index [:university_person_id, :education_cohort_id], name: 'index_person_cohort'
+    end
+  end
+end
diff --git a/docs/communication/wysiwyg.md b/docs/_rd/wysiwyg.md
similarity index 100%
rename from docs/communication/wysiwyg.md
rename to docs/_rd/wysiwyg.md
diff --git a/docs/administration/internships.md b/docs/administration/internships.md
deleted file mode 100644
index 994c3f168a91c06b3d8c7f0de7edc760642eb101..0000000000000000000000000000000000000000
--- a/docs/administration/internships.md
+++ /dev/null
@@ -1 +0,0 @@
-# Interships
diff --git a/docs/benchmark.md b/docs/benchmark.md
deleted file mode 100644
index 94e442e846eaf14322f410f57adb57aa473b110d..0000000000000000000000000000000000000000
--- a/docs/benchmark.md
+++ /dev/null
@@ -1,29 +0,0 @@
-# Benchmark
-
-## Ecoles
-
-### Excellent websites
-
-- https://www.harvard.edu/
-- https://www.mit.edu/
-
-### Other websites
-
-- https://www.stanford.edu/
-- https://www.brown.edu/
-- https://www.columbia.edu/
-- https://home.dartmouth.edu/
-- https://www.upenn.edu/
-- https://www.princeton.edu/
-- https://www.yale.edu/
-- https://www.ox.ac.uk/
-- https://www.cam.ac.uk/
-- https://www.caltech.edu/
-- https://www.imperial.ac.uk/
-- https://ethz.ch/
-- https://www.ucl.ac.uk/
-- https://psl.eu/
-- https://www.ip-paris.fr/
-- https://www.sorbonne-universite.fr/
-- https://www.hec.edu/fr
-- https://www.essec.edu/fr/
diff --git a/docs/communication/alumni.md b/docs/communication/alumni.md
deleted file mode 100644
index af602514ef55f29d1fa1a856172d6cea2351cf31..0000000000000000000000000000000000000000
--- a/docs/communication/alumni.md
+++ /dev/null
@@ -1 +0,0 @@
-# Alumni
diff --git a/docs/communication/blocks.md b/docs/communication/blocks.md
deleted file mode 100644
index d7251e6e01f7e53322b87b0e3a1e4b49178d2222..0000000000000000000000000000000000000000
--- a/docs/communication/blocks.md
+++ /dev/null
@@ -1,97 +0,0 @@
-# Blocks
-
-Blocs de contenus ajoutés à un objet (page, post, program...),
-avec des templates (organigramme, partenaires...).
-
-Il faut lister les dépendances des blocs et les ajouter à l'objet about.
-
-## Dev
-
-### Model
-
-```
-communication/website/Block
-- university:references
-- about:references (polymorphic)
-- template:integer (enum)
-- position:integer
-- data:jsonb
-```
-
-Pour commencer, les valeurs de l'enum seront :
-- 100, organization_chart
-- 200, partners
-
-### Partial about
-Un partial que l'on peut ajouter à un show d'objet, avec :
-- la liste des blocs utilisés (avec boutons show et edit)
-- la possibilité de les ordonner (position)
-- un bouton pour ajouter un bloc
-
-```
-views/admin/communication/blocks/_list.html.erb
-```
-
-### Show
-Le show du bloc utilise le partial de son template
-```
-views/admin/communication/blocks/templates/partners/_show.html.erb
-```
-
-### Edit
-L'edit du bloc utilise le partial de son template
-```
-views/admin/communication/blocks/templates/partners/_edit.html.erb
-```
-
-### Concern
-Tous les objets ayant des blocs utilisent le concern `WithBlocks`, qui ajoute la méthode `blocks` (la liste des blocs, dans l'ordre).
-
-### Export statique
-Les blocs sont exportés dans le frontmatter grâce au partial
-```
-views/admin/communication/blocks/_static.html.erb
-```
-qui donne ce type de résultat
-```
-blocks:
-  - template: partners
-    data:
-      - name: Partner 1
-        url: https://partner1.com
-        logo: "e09f3794-44e5-4b51-be02-0e384616e791"
-```
-Les générateurs de chaque type suivent l'organisation :
-```
-views/admin/communication/blocks/templates/partners/_static.html.erb
-```
-Attention, il faut 6 espaces pour respecter l'indentation du front-matter :
-```
-      - name: Partner 1
-        url: https://partner1.com
-        logo: "e09f3794-44e5-4b51-be02-0e384616e791"
-```
-
-### Dépendances
-
-Il faut créer une classe pour chaque template, avec le nom au singulier (partners devient partner.rb, et la class Partner) :
-
-```
-models/communication/block/partner.rb
-```
-avec une structure de type :
-```
-class Communication::Block::Partner < Communication::Block::Template
-  def git_dependencies
-    ...
-  end
-end
-```
-
-## Pour créer un bloc
-
-1. Déclarer le template dans l'enum du modèle block
-2. Créer l'edit, le show et le static dans la vue du template
-3. Créer la classe du template pour gérer les dépendances
-4. Créer la vignette dans les images
-5. Ecrire les locales fr et en
diff --git a/docs/communication/extranets.md b/docs/communication/extranets.md
deleted file mode 100644
index a3c46362a75b6821ddc360a0f21f340d81cf7b9c..0000000000000000000000000000000000000000
--- a/docs/communication/extranets.md
+++ /dev/null
@@ -1,10 +0,0 @@
-# Extranets
-
-En plus des sites statiques, il est possible d'utiliser Osuny pour gérer des extranets, avec des utilisateurs authentifiés.
-
-Un extranet permet d'utiliser les fonctions suivantes :
-- alumni
-- stages
-- trombinoscopes
-- news privées ?
-- événements privés ?
diff --git a/docs/communication/models.md b/docs/communication/models.md
deleted file mode 100644
index 7493cdb6a8717b259901acee2a0956419658d5ac..0000000000000000000000000000000000000000
--- a/docs/communication/models.md
+++ /dev/null
@@ -1,21 +0,0 @@
-# Communication
-
-## Models
-
-### communication/website/Event
-
-- university:references
-- communication_website:references
-- communication_website_event_kind:references
-- name:string
-- description:text
-- text:html
-- published:boolean
-- published_at:datetime
-
-### communication/website/event/Kind
-
-- university:references
-- communication_website:references
-- name:string
-- position:integer
diff --git a/docs/communication/websites/customs.md b/docs/communication/websites/customs.md
deleted file mode 100644
index 6dad14a7c768d87fbc433d1ebe871fd088aec5da..0000000000000000000000000000000000000000
--- a/docs/communication/websites/customs.md
+++ /dev/null
@@ -1,52 +0,0 @@
-# Customs
-
-Au-delà des pages et des actualités, les sites ont souvent besoin d'objets spécifiques, au cas par cas.
-Le site https://www.osuny.org/ présente des fonctionnalités, avec un statut.
-Le site https://cyberneticproject.eu/ présente des fiches techniques, avec des synonymes, des téléchargements, un récapitulatif, des références bibliographiques...
-
-
-Afin de permettre cette souplesse, nous utilisons des types personnalisés (custom types).
-Le type définit une nouvelle sorte d'objets (ex: feature, technical_sheet...).
-Chaque type a des propriétés (title, description, summary, status, references...), qui génèrent un formulaire à la volée.
-Le type s'ajoute au menu du site, et permet de créer des éléments.
-Les éléments s'exportent en statique en utilisant la structure définie par les propriétés.
-
-## Modèles
-
-communication/website/custom/Type
-- university:references
-- website:references
-- name:string
-- identifier:string
-- position:integer
-- order:boolean
-- tree:boolean
-- date:boolean
-
-
-Si order est true, les éléments de ce type peuvent être classés par position (js sortable).
-Si tree est true, les éléments peuvent être organisés en arbre, avec des parents et des enfants.
-Si date est true, les élément peuvent être publiés à une date donnée.
-
-
-communication/website/custom/type/Property
-- university:references
-- website:references
-- type:references
-- name:string
-- identifier:string
-- kind:integer (enum)
-- position
-
-
-communication/website/custom/Element
-- university:references
-- website:references
-- type:references
-- name:string
-- slug:string
-- published:boolean
-- published_at:datetime
-- parent:references
-- position:integer
-- data:jsonb
diff --git a/docs/communication/websites/export.md b/docs/communication/websites/export.md
deleted file mode 100644
index efc0e03e90c7229394896b52ee73b0b87cbf1d6f..0000000000000000000000000000000000000000
--- a/docs/communication/websites/export.md
+++ /dev/null
@@ -1,146 +0,0 @@
-# Export Hugo
-
-## Contexte
-
-Chaque website peut avoir un repository git.
-Tous les objets de ce website doivent être synchronisés sur le repository.
-Les publications doivent se font en asynchrone parce qu'elles peuvent être longues.
-
-
-Certains objets peuvent appartenir à plusieurs websites, donc plusieurs repositories, comme par exemple les programs.
-Certains objets ont des dépendances, par exemple les pages enfants, les auteurs, les catégories ou les médias.
-
-
-Les fichiers renommés doivent être déplacés sur git.
-Les fichiers supprimés ou dépubliés doivent être supprimés sur git.
-Il faut veiller à limiter le nombre de commits et à éviter les commits vides.
-
-## Setup
-
-### GitHub
-
-- Créer un repository à partir du template suivant : https://github.com/noesya/osuny-hugo-template
-- Une fois le repository créé, générer un personal access token ici : https://github.com/settings/tokens
-  - Permission à accorder : `repo`
-  - Durée : pour une bonne sécurité, il n'est pas recommandé de créer un token permanent, notez simplement qu'il faut le régénérer régulièrement.
-- Copier le personnal access token
-- Dans le back-office d'Osuny, créer ou modifier un website et renseignez les 2 champs relatifs à Git :
-  - Repository : `username/repo`
-  - Access token : `ghp_xxxxxxxxxxxxxxxxxxxx`
-
-### Déploiement (Netlify)
-
-- Créer un site sur Netlify lié au repository du site
-- Dans "Site settings", "Build & deploy", "Environment", "Environment variables", ajouter :
-  - `HUGO_VERSION` avec pour valeur la dernière version (ex: `0.92.1`)
-
-La récupération du thème se fait via SSH par défaut. Pour que le déploiement fonctionne correctement, vous pouvez :
-- Changer le remote du submodule pour qu'il utilise HTTPS.
-- Garder le SSH, cependant il faut :
-  - Générer et copier la deploy key du site sur Netlify (dans "Site settings", "Build & deploy" puis "Deploy key").
-  - L'ajouter dans la section "Deploy keys" du repository contenant le thème (ici : https://github.com/noesya/osuny-hugo-theme/settings/keys).
-
-## Architecture
-
-Les git::providers permettent de dialoguer avec les services comme Github et Gitlab.
-Le git::repository sert de façade et abstrait le provider.
-
-
-Chaque objet publiable utilise un objet active record Communication::Website::GitFile qui garde la trace du dernier chemin et du SHA.
-
-## Flux
-
-### Version 1
-
-Lors de l'enregistrement d'un objet, il faut, pour chaque website :
-- créer éventuellement le git_file (1 pour chaque website)
-- envoyer le git_file (add_to_batch)
-- modifier ses dépendances, qui créent leur git_files pour chaque repository
-- envoyer les git_files des dépendances aux repositories respectifs (add_to_batch)
-- pour chaque website, si au moins un fichier a été ajouté :
-    - déclencher une modification (touch), qui génère une action asynchrone :
-        - pour chaque file :
-            - générer le fichier statique
-            - calculer le SHA
-            - comparer au SHA stocké
-            - needs_sync si SHA différent ou path différent
-        - si au moins un needs_sync :
-            - créer un commit pour tout ça
-            - push
-            - mettre à jour les previous_path et les SHA des git_files
-
-Ce flux cause un problème majeur : tout ce qui est analysé disparaît en asynchrone.
-
-### Version 2
-
-Après l'enregistrement d'un objet, il faut lancer une tâche asynchrone de synchronisation.
-Cette tâche est lancée par les controllers, et intégrée dans le partial `WithGit`.
-```
-def create
-  if @page.save_and_sync
-    ...
-  end
-end
-
-def update
-  if @page.update_and_sync(page_params)
-    ...
-  end
-end
-
-def destroy
-  @page.destroy_and_sync
-end
-```
-
-
-Pour les reorder, chaque objet doit avoir ses siblings en dépendance, donc il suffit de synchroniser un objet déplacé pour qu'il gère l'ensemble :
-```
-def reorder
-  ...
-  pages.first.sync_with_git
-end
-```
-
-## Code
-
-### Website::WithRepository
-
-Le website a un trait WithRepository qui gère son rapport avec le repository Git, quel que soit le provider (Github, Gitlab...).
-
-### Objets exportables vers Git
-
-Tous les objets qui doivent être exportés vers Git :
-- doivent utiliser le concern `WithGit`, qui gère l'export vers les repositories des objets et de leurs dépendances
-- peuvent intégrer le concern `WithMedia` s'il utilise des médias (`featured_image` et/ou images dans des rich texts)
-- peuvent présenter une méthode `identifiers` qui liste les identifiants des git_files à générer, pour les objets qui créent plusieurs fichiers (le fichier par défaut s'appelle `static`)
-- peuvent présenter une méthode `git_dependencies_static` qui liste les dépendances de l'identifiant par défaut `static`
-- peuvent présenter une méthode `git_destroy_dependencies_static` qui liste les dépendances à supprimer en cascade de l'identifiant par défaut `static`
-- peuvent présenter des méthodes `git_dependencies_author` et/ou `git_destroy_dependencies_author` qui liste les dépendances de l'identifiant `author`
-
-### Modèle Communication::Website::GitFile
-
-La responsabilité de la synchronisation repose sur Communication::Website::GitFile, notamment :
-- l'information est-elle intègre, synchronisée avec le repo ? (previous_sha et previous_path cohérents avec le repo git)
-- le fichier doit-il être créé ? (pas à supprimer et (non intègre, ou pas de previous_sha/previous_path))
-- le fichier doit-il être mis à jour ? (pas à supprimer et (non intègre, ou previous_sha/previous_path différent du sha/path))
-- le fichier doit-il être supprimé ? (path nil ou marquage à détruire)
-
-
-Pour cela, le git_file dispose des propriétés suivantes :
-- previous_path (le chemin à la dernière sauvegarde, nil si pas encore créé, ou détruit)
-- previous_sha (le hash de la précédente version, utile pour savoir si le fichier a changé)
-- identifier (l'identifiant du fichier à créer, `static` par défaut, pour les objets créant plusieurs fichiers)
-
-
-Pour informer sur les actions à mener, il dispose des méthodes interrogatives suivantes :
-- synchronized_with_git? (pour évaluer l'intégrité vs le repository)
-- should_create? (pour savoir s'il faut créer ou pas)
-- should_update? (pour savoir s'il faut régénérer ou pas)
-- should_destroy? (pour savoir s'il faut supprimer)
-
-
-Pour générer les fichiers, il dispose des méthodes :
-- to_s (pour générer le fichier statique à jour)
-- sha (pour calculer le hash du fichier à jour)
-- path (pour générer le chemin à jour)
diff --git a/docs/communication/websites/import.md b/docs/communication/websites/import.md
deleted file mode 100644
index 50f270657cb2098bc2908245fac3e6b1c67a11f2..0000000000000000000000000000000000000000
--- a/docs/communication/websites/import.md
+++ /dev/null
@@ -1,94 +0,0 @@
-# Import WordPress
-
-## Contexte
-
-L'objectif est de fluidifier la transition depuis un site déjà en place, notamment WordPress. Deux approches sont possibles : interne, avec un accès BDD, et externe, en passant par le site lui-même. Nous privilégierons l'approche externe dans un premier temps, pour permettre l'approche avant-vente (présenter un site pré-migré).
-
-## Problématiques
-
-1. Détection des pages
-2. Extraction des contenus bruts
-3. Distingo entre pages et posts, et autres types d'objets
-4. Identification des menus
-
-## Approche externe
-
-Plusieurs possibilités :
-- le crawling
-- le sitemap
-- le flux RSS
-- l'api
-
-## Approche interne
-
-Plugin, connexion à la BDD, export JSON ou connexion API.
-
-## Développement
-
-Communication::Website::Imported::Website
-- university:references
-- website:references (has_one Communication::Website)
-- status:integer (enum)
-
-Communication::Website::Imported::Page
-- university:references
-- website:references (has_one Communication::Website::Imported::Website)
-- page:references (has_one Communication::Website::Page)
-- status:integer (enum)
-
-Etapes :
-1. Création du site, avec son URL
-2. Lancement de l'import (création de Communication::Website::Imported::Website)
-3. Import des sitemaps (création de Communication::Website::Imported::Page)
-4. Import du contenu brut des pages importées
-5. Analyse du contenu des pages importées et création / mise à jour des pages
-
-## Import depuis WordPress
-
-### Media
-1. On importe les media depuis l'API
-2. On crée des objets en DB (Communication::Website::Imported::Medium)
-
-### Pages
-1. On importe les pages depuis l'API
-2. On crée des objets en DB (Communication::Website::Imported::Page)
-3. Les objets importés créent ou mettent à jour les objets réels (Communication::Website::Page)
-    3.1 sans écraser de modifs locales
-    3.2 uniquement si l'import a bougé
-    3.3 Le contenu de l'html est filtré
-        3.3.1 enlever les balises problématiques
-        3.3.2 supprimer les classes
-        3.3.3 supprimer les ids
-        3.3.4 décaler les titres si h1
-    3.4 la featured image est transformée en attachment
-    3.5 si pas de featured image, la première image est enlevée du texte et devient featured
-    3.6 les medias dans le texte html sont transformés en action text attachments
-        3.6.1 lister les files dans le domaine
-        3.6.2 identifier le media master correspondant (via data:jsonb)
-        3.6.3 s'il n'existe pas, le créer (le cas se produit il ?)
-        3.6.4 crée l'attachment
-        3.6.5 on remplace le code du media par l'action text attachement
-
-### Posts
-Idem pages
-
-## Exemples
-
-### Condé
-
-- https://ecoles-conde.com/sitemap_index.xml
-- https://ecoles-conde.com/wp-json/wp/v2/posts
-- https://ecoles-conde.com/wp-json/wp/v2/pages
-
-### IUT Bordeaux Montaigne
-
-- https://www.iut.u-bordeaux-montaigne.fr/wp-sitemap.xml
-- https://www.iut.u-bordeaux-montaigne.fr/wp-sitemap-posts-post-1.xml
-- https://www.iut.u-bordeaux-montaigne.fr/wp-sitemap-posts-page-1.xml
-- https://www.iut.u-bordeaux-montaigne.fr/wp-json/wp/v2/posts
-- https://www.iut.u-bordeaux-montaigne.fr/wp-json/wp/v2/pages
-
-## Recherches
-
-- https://kinsta.com/fr/blog/api-rest-wordpress/
-- https://getshifter.io/
diff --git a/docs/communication/websites/readme.md b/docs/communication/websites/readme.md
deleted file mode 100644
index 3847269f881807fa55c1105e66bdad61d05249a4..0000000000000000000000000000000000000000
--- a/docs/communication/websites/readme.md
+++ /dev/null
@@ -1,89 +0,0 @@
-# Websites
-
-## websites/Site
-
-Attributes:
-- university:references
-- name:string
-- domain:string
-
-## websites/Post
-
-Attributes:
-- university:references
-- website:references
-- title:string
-- description:text
-- text:text
-- published:boolean
-- published_at:datetime
-
-## websites/Page
-
-Attributes:
-- university:references
-- website:references
-- title:string
-- description:text
-- text:text
-- parent:references
-- published:boolean
-
-## websites/Document
-
-Est-ce vraiment dans ce namespace ? Peut-être un DAM ?
-
-1 document peut concerner :
-- toute l'université (charte numérique)
-- 1 école (règlement intérieur de l'école)
-- 1 campus (règlement intérieur du campus)
-- 1 formation (contrat d'alternance type)
-- 1 session (calendrier pédagogique de l'année en cours)
-
-Attributes:
-- university:references
-- name:string
-- school:references (optional)
-- campus:references (optional)
-- program:references (optional)
-- session:references (optional)
-
-## websites/Menu
-
-Attributes:
-- university:references
-- website:references
-- title:string
-- identifier:string
-
-## websites/menu/Item
-
-Attributes:
-- university:references
-- website:references
-- menu:references
-- title:string
-- parent:references
-- position:integer
-- kind:integer (enum: page, url)
-- about:references (polymorphic)
-
-## Export du menu
-
-/_data/menus.yml
-
-```yaml
-primary:
-    - title: Accueil
-      target: /
-    - title: Formations
-      target: /formations
-      children:
-          - title: DUT
-            target: /formations/dut
-    - title: ENT
-      target: https://ent.u-bordeaux3.fr
-legal:
-    - title: Mentions légales
-      target: /mentions-legales
-```
diff --git a/docs/communication/websites/templates.md b/docs/communication/websites/templates.md
deleted file mode 100644
index 1d03ffb132ec9ce1ee7c0fc6805275f85fdbf162..0000000000000000000000000000000000000000
--- a/docs/communication/websites/templates.md
+++ /dev/null
@@ -1,29 +0,0 @@
-# Templates
-
-## Thème
-
-https://github.com/noesya/osuny-hugo-theme
-
-## Template
-
-https://github.com/noesya/osuny-hugo-template
-
-Pour faire la mise à jour :
-
-```
-git remote add template git@github.com:noesya/osuny-hugo-template.git
-git fetch --all
-git merge template/master --allow-unrelated-histories
-```
-
-## Site d'un journal
-
-https://github.com/osuny-org/clermontauvergne-journal-degrowth
-
-Pour faire la mise à jour :
-
-```
-git remote add template git@github.com:noesya/osuny-hugo-template.git
-git fetch --all
-git merge template/master --allow-unrelated-histories
-```
diff --git a/docs/communication/websites/wysiwyg.md b/docs/communication/websites/wysiwyg.md
deleted file mode 100644
index 43a4073f8ce6b5f9dba18a129d93a3c481119621..0000000000000000000000000000000000000000
--- a/docs/communication/websites/wysiwyg.md
+++ /dev/null
@@ -1,135 +0,0 @@
-# WYSIWYG
-
-## Quels enjeux ?
-
-Permettre l'édition, mais limiter les options graphiques (ni couleurs, ni tailles, ni typos).
-
-Fonctionnalités :
-- intégration d'images dans le corps du texte, en gardant la trace du blob active storage, et en les intégrant dans la liste des dépendances.
-- intégration de vidéos.
-- intégration d'autres formats (Tweets...).
-
-## Solutions techniques
-
-### ActionText
-
-Avantages :
-- active storage intégré
-- Trix intégré
-
-Inconvénients :
-- Pas de modèle (polymorphic)
-
-### Trix
-
-Avantages :
-- intégré à ActionText
-- très limité
-
-Inconvénients :
-- très limité (target blank, 1 seul niveau de titre, pas d'embed, pas de code source)
-- pas extensible
-
-### Summernote
-
-Avantages :
-- vaste
-- extensible
-
-Inconvénients :
-- dépendance jQuery
-- pas intégré à ActionText
-- moyennement robuste quand on le torture
-
-### Page builder custom
-
-Avantages :
-- puissant
-- souple
-
-Inconvénients :
-- compliqué à construire et maintenir
-- compliqué à utiliser
-
-
-## Benchmark
-
-[Refinery](https://www.refinerycms.com/)
-
-
-[Spina](https://spinacms.com/)
-
-
-[Alchemy CMS](https://alchemy-cms.com/)
-
-
-[Locomotive CMS](https://www.locomotivecms.com/)
-
-
-## Méthode
-
-### action-text-attachment
-
-Dans la BDD, on stocke cette balise :
-```
-<action-text-attachment sgid="BAh[...]1df3"
-                        content-type="image/jpeg"
-                        url="http://demo.osuny:3000/rails/active_storage/blobs/redirect/eyJf[...]0f4a1/domenico_bruno_de_lobkowitz_watchingwindows_com_08.jpg" filename="domenico_bruno_de_lobkowitz_watchingwindows_com_08.jpg"
-                        filesize="352931"
-                        width="588"
-                        height="746"
-                        previewable="true"
-                        presentation="gallery">
-</action-text-attachment>
-```
-
-A l'édition, la balise est "remplie" avant affichage, pour avoir une preview.
-A l'enregistrement, la balise est vidée.
-
-Etapes normales
--[x] A l'import d'une image, ajouter l'action-text-attachement autour de l'img
--[ ] A la suppression d'une image dans l'éditeur, supprimer l'action-text-attachement autour de l'img
--[x] A l'enregistrement, déshydrater les action-text-attachements
--[x] A l'édition, réhydrater les action-text-attachements
--[ ] Après l'enregistrement mettre à jour les blobs attachés à l'objet parent (le post, par exemple)
-
-Si un programme a 5 champs Summernote avec 3 images dans chaque champ, cela fait 15 attachments à lier au programme.
-Si on enlève une image d'un champ, il faut mettre à jour la liste pour avoir les 14 bons attachments.
-
-Actions de dev
--[ ] Coder les ajouts aux modèles dans Osuny
--[ ] Coder le JS dans Osuny
--[ ] Une fois que c'est fait, déplacer le Ruby et le JS dans summernote-rails
-
-Migration phase 1
--[x] Ajouter des champs _new
--[x] Au rails app:fix, transformer le markup Trix en markup Summernote (application_record) dans les propriétés _new
-Migration phase 2
--[ ] Supprimer les champs ActionText dans les modèles
--[ ] Supprimer la table d'ActionText
--[ ] Renommer les champs en enlevant _new
-
-### Le pdf
-
-Autoriser dans summernote
-
-### Code HTML cible
-
-```
-<h2>Titre</h2>
-<p>
-  Texte normal<br>
-  <b>Texte en gras</b><br>
-  <i>Texte en italique</i>
-  <action-text-attachment sgid="BAh7CEkiCGdpZAY6BkVUSSJUZ2lkOi8vb3N1bnkvQWN0aXZlU3RvcmFnZTo6QmxvYi9hYWUyNDI5OC1kNDE2LTQ2YWMtYTRlNS02ZjY4ZGU2MjFiZDE_ZXhwaXJlc19pbgY7AFRJIgxwdXJwb3NlBjsAVEkiD2F0dGFjaGFibGUGOwBUSSIPZXhwaXJlc19hdAY7AFQw--7e8ead4d79f455499f1d73e8d53a4b8e81a21df3" content-type="image/jpeg" url="http://demo.osuny:3000/rails/active_storage/blobs/redirect/eyJfcmFpbHMiOnsibW...df8140070f4a1/domenico_bruno_de_lobkowitz_watchingwindows_com_08.jpg?website_id=6d8fb0bb-0445-46f0-8954-0e25143e7a58" filename="domenico_bruno_de_lobkowitz_watchingwindows_com_08.jpg" filesize="352931" width="588" height="746" previewable="true" presentation="gallery">
-    <figure class="attachment attachment--preview attachment--jpg">
-      <picture>
-        <source srcset="/rails/active_storage/representations/redirect/eyJfcmFpbHMiOns...XJpYXRpb24ifX0=--7d11fdd26322fef8959415f46d5e2c6d6763b4c0/domenico_bruno_de_lobkowitz_watchingwindows_com_08.jpg 100w, /rails/active_storage/representations/redirect/eyJfcmFpbHMiOnsibWVzc2FnZSI6IkJBaE...527eb11f95949a389acb1c/domenico_bruno_de_lobkowitz_watchingwindows_com_08.jpg 200w" type="image/webp">
-        <source srcset="/rails/active_storage/representations/redirect/eyJfcmFpbHMiOnsibW...9fd77765da7c4f647d453b2/domenico_bruno_de_lobkowitz_watchingwindows_com_08.jpg?website_id=6d8fb0bb-0445-46f0-8954-0e25143e7a58 100w, /rails/active_storage/representations/redirect/eyJfcmFpbHMiOnsibW...bb3bc14127bc06ce0d1e32/domenico_bruno_de_lobkowitz_watchingwindows_com_08.jpg?website_id=6d8fb0bb-0445-46f0-8954-0e25143e7a58 200w" type="image/jpeg">
-        <img src="/rails/active_storage/representations/redirect/eyJfcmFpbHMiOnsi...190ffccd8/domenico_bruno_de_lobkowitz_watchingwindows_com_08.jpg?website_id=6d8fb0bb-0445-46f0-8954-0e25143e7a58" loading="lazy" decoding="async" width="800">
-      </picture>
-    </figure>
-  </action-text-attachment>
-  <a href="https://www.u-bordeaux-montaigne.fr/fr/actualites/vie-etudiante/soutenir-les-etudiant-e-s-les-aides-de-l-universite.html">Lien</a>
-</p>
-```
diff --git a/docs/education/qualiopi/guide-lecture-referentiel-qualite.pdf b/docs/education/qualiopi/guide-lecture-referentiel-qualite.pdf
deleted file mode 100644
index c1f804a1460335cc081147c47400352d1dcc1893..0000000000000000000000000000000000000000
Binary files a/docs/education/qualiopi/guide-lecture-referentiel-qualite.pdf and /dev/null differ
diff --git a/docs/education/qualiopi/qualiopi.md b/docs/education/qualiopi/qualiopi.md
deleted file mode 100644
index 1c8f921f4589565b052723abafbafd2c29524f1f..0000000000000000000000000000000000000000
--- a/docs/education/qualiopi/qualiopi.md
+++ /dev/null
@@ -1,142 +0,0 @@
-# Qualiopi
-
-Les vérifications se font pour chaque programme.
-
-## Indicateur 1
-
-Vérification de la présence des champs dans les programmes
-
-## Indicateur 2
-
-- 1 champ libre + documents joints dans le programme dont on vérifie la présence
-- des stats extraites de l'alumni
-
-## Indicateur 3
-
-- 1 champ libre + documents joints dans le programme dont on vérifie la présence
-- des stats extraites de l'alumni
-
-## Indicateur 4
-
-1 champ libre + documents joints dans le programme dont on vérifie la présence
-
-## Indicateur 5
-
-1 champ libre + documents joints dans le programme dont on vérifie la présence
-
-## Indicateur 6
-
-Présence de cours suffisamment détaillés et complets
-
-## Indicateur 7
-
-1 champ libre + documents joints analysant l'adéquation avec la certification (il faudra modéliser la certification vs RNCP)
-
-## Indicateur 8
-
-- 1 champ libre + documents joints dans le programme dont on vérifie la présence
-- procédure directement intégrée dans la gestion des admissions
-
-## Indicateur 9
-
-1 champ libre + documents joints dans le programme dont on vérifie la présence
-
-## Indicateur 10
-
-? Différence avec 6 ?
-
-## Indicateur 11
-
-- 1 champ libre + documents joints dans le programme dont on vérifie la présence
-- Alumni
-
-## Indicateur 12
-
-- 1 champ libre + documents joints dans le programme dont on vérifie la présence
-- Statistiques d'abandon extraites des profils étudiants
-
-## Indicateur 13
-
-- 1 champ libre + documents joints dans le programme dont on vérifie la présence
-- Statistiques et outils de suivi issus des profils étudiants (module alternance à fabriquer)
-
-## Indicateur 14
-
-1 champ libre + documents joints dans le programme dont on vérifie la présence
-
-## Indicateur 15
-
-Proche 9
-
-Qualifier les documents en fonction de leur contenu vs indicateurs, puis évaluer la présence
-
-## Indicateur 16
-
-Lier les programmes aux certifications, documenter les conditions de présentation
-
-## Indicateur 17
-
-Documents confidentiels à fournir.
-Créer un espace sécurisé de dialogue avec l'auditeur ?
-
-## Indicateur 18
-
-Organigramme avec les postes et les champs (pédagogique, technique, commercial, social)
-
-## Indicateur 19
-
-Ressources éducatives libres, accès à des ressources privées pour les étudiants, bibliographies
-
-## Indicateur 20
-
-Documents et personnes dans l'organigramme
-
-## Indicateur 21
-
-1. Définition des compétences nécessaires
-2. CV et compétences des formateurs
-3. Plan de formation des formateurs
-
-## Indicateur 22
-
-Preuves de formation des formateurs
-
-## Indicateur 23
-
-Preuves de veille légale et réglementaire
-
-## Indicateur 24
-
-Preuves de veille compétences, métiers, emplois
-
-## Indicateur 25
-
-Preuves de veille technologique et pédagogique
-
-## Indicateur 26
-
-Liste des partenaires handicap
-
-## Indicateur 27
-
-Preuves de conformité des sous-traitants
-
-## Indicateur 28
-
-Réseau de partenaires socio-économiques et preuves de coopération
-
-## Indicateur 29
-
-Preuves d'actions orientées insertion professionnelle et poursuite d'études
-
-## Indicateur 30
-
-Feedback des parties prenantes
-
-## Indicateur 31
-
-Gestion des réclamations et suivi de la satisfaction
-
-## Indicateur 32
-
-Preuve d'action correctives sur 31
diff --git a/docs/education/readme.md b/docs/education/readme.md
deleted file mode 100644
index 4e2018713cc993c9c7b6d9c7f4efc0c9cc775da0..0000000000000000000000000000000000000000
--- a/docs/education/readme.md
+++ /dev/null
@@ -1,62 +0,0 @@
-# Education
-
-## education/Program
-
-https://schema.org/EducationalOccupationalProgram
-
-Bachelor Universitaire de Technologie Métiers du Multimédia et de l'Internet
-
-- university:references
-- name:string
-- level:integer (enum)
-- capacity:integer
-- ects:integer
-- continuing:boolean
-- prerequisites:text
-- objectives:text
-- duration:text
-- registration:text
-- pedagogy:text
-- evaluation:text
-- accessibility:text
-+ schools:habtm
-+ campuses:habtm
-
-## education/Course
-
-30 heures d'histoire de l'art en année 1
-
-- program:references
-- year:references
-- name:string
-- syllabus:text
-- hours:integer
-
-## education/Year
-
-- university:references
-- program:references
-- year:integer
-
-## education/Session
-
-- name:string
-- university:references
-- program:references
-- from:date
-- to:date
-
-## qualiopi/Criterion
-
-- number:integer
-- name:text
-- description:text
-
-## qualiopi/Indicator
-- criterion:references
-- number:integer
-- name:text
-- level_expected:text
-- proof:text
-- requirement:text
-- non_conformity:text
diff --git a/docs/education/rncp/README.md b/docs/education/rncp/README.md
deleted file mode 100644
index 585382a4d2948e7b597ba6509468e7b5d5b03a80..0000000000000000000000000000000000000000
--- a/docs/education/rncp/README.md
+++ /dev/null
@@ -1,3 +0,0 @@
-# RNCP
-
-https://www.data.gouv.fr/fr/datasets/repertoire-national-des-certifications-professionnelles-et-repertoire-specifique/
diff --git a/docs/research/journals.md b/docs/research/journals.md
deleted file mode 100644
index a897003057699a5108c411017e5c435b5376c446..0000000000000000000000000000000000000000
--- a/docs/research/journals.md
+++ /dev/null
@@ -1,8 +0,0 @@
-# Journals
-
-## References
-
-http://infolit.be/CoMLiS/ch02s03.html
-
-
-Authors biography is fixed per article. The biography does not evolve, it is set in time (by the time of publication).
diff --git a/docs/research/laboratories.md b/docs/research/laboratories.md
deleted file mode 100644
index 927694980d3284c99284f84e600fe9c280fd6f85..0000000000000000000000000000000000000000
--- a/docs/research/laboratories.md
+++ /dev/null
@@ -1,98 +0,0 @@
-# Laboratories
-
-## Axes (research axis/axes) ou équipes
-
-Projet scientifique sur plusieurs années, avec des thèmes (ou thématiques) de recherche
-
-## Personnes
-
-- Prénom
-- Nom
-- Photo
-- Poste : Professeur des Universités
-- Responsabilités : Co-Directeur du Laboratoire, Responsable de l'axe ICIN
-- Statut : statutaire ou doctorant ou associé
-- Axe : ICIN
-- Disciplines : Sciences de l'info (71e CNU)
-- Année de rattachement : 2015
-- Université de rattachement : Université de Bordeaux
-- Mail
-- Thèmes de recherche
-- Responsabilités (administratives et d'enseignement)
-- Publications
-- HDR ou pas
-
-### Doctorant
-
-- Sujet de thèse
-- Direction de thèse
-- Inscription en thèse
-
-## Evénements
-
-Pour le MICA, autour d'une dizaine par mois, avec un pic en novembre
-
-### Appels à contribution
-
-### Parutions
-
-### Séminaires
-
-### Soutenances
-
-Thèses et HDR
-
-Import depuis theses.fr ?
-
-## Programmes de recherche
-
-- Nom : CyberNeTic. Se prémunir des mécanismes du cyberharcèlement.
-- Portée : région, national, international
-- Nature : Projet Région Nouvelle-Aquitaine
-- Durée : 3 ans (2020-2022)
-- Pilotage : Marlène Dulaurans
-- Co-Responsabilité : personnes et postes
-- Partenaires extérieurs : Orange...
-- Présentation
-- URL
-
-## Partenaires
-
-- Catégorie (pays par ex)
-- Nom
-- Logo (éventuellement, si dispo)
-- URL
-
-Alphabétique parfois, classé parfois
-
-## Bibliographies
-
-DOI, HAL, researchgate
-
-
-Import depuis HAL
-
-## Publications
-
-Avec décompte pour l'Hcéres
-
-
-Publier automatiquement sur :
-- HAL
-- Arxiv
-- DOI
-- Researchgate
-- Academia
-- autres sites mineurs
-
-## Newsletter
-
-Collecte de mails, gestion avec Sendinblue
-
-## Don
-
-Fonctionnalité nouvelle au MICA, page d'explication (pas de don en ligne)
-
-## SEO
-
-Plan de redirection
diff --git a/docs/research/readme.md b/docs/research/readme.md
deleted file mode 100644
index 36b8e6529ac3dd09e0c50097fcc4bbc16d553964..0000000000000000000000000000000000000000
--- a/docs/research/readme.md
+++ /dev/null
@@ -1,48 +0,0 @@
-## Models
-
-### university/person/Researcher (extends university/Person)
-
-... ajouté à university/Person
-- habilitation:boolean
-- tenure:boolean
-
-### research/Laboratory
-
-- university:references
-- name:string
-- address:string
-- zipcode:string
-- city:string
-- country:string
-
-### research/laboratory/Axis
-
-- university:references
-- research_laboratory:references
-- name:string
-- short_name:string
-- position:integer
-- description:text
-- text:html
-
-### research/Thesis
-
-- university:references
-- research_laboratory:references
-- author:references (person)
-- director:references (person)
-- title:string
-- abstract:text
-- started_at:date
-- completed:boolean
-- completed_at:date
-
-### research/laboratory/Involvement (tbc)
-
-- university:references
-- research_laboratory:references
-- university_person:references
-- research_axis:references
-- status:enum (statutaire, doctorant, associé)
-- description:string
-- themes:html
diff --git a/docs/university/readme.md b/docs/university/readme.md
deleted file mode 100644
index 94ba0222e61688bccc46f1288a9b7447c9e03dfc..0000000000000000000000000000000000000000
--- a/docs/university/readme.md
+++ /dev/null
@@ -1,74 +0,0 @@
-# University
-
-## Models
-
-### University
-
-- name:string
-- address:string
-- zipcode:string
-- city:string
-- country:string
-- private:boolean
-
-### university/Person
-
-- university:references
-- user:references(optional)
-- last_name:string
-- first_name
-- slug:string
-- is_researcher:boolean
-- is_teacher:boolean
-- is_administrator:boolean
-- phone:string
-- email:string
-- description:text
-- habilitation:boolean
-- tenure:boolean
-
-### university/person/Involvement
-
-- university:references
-- university_person:references
-- kind (administrator, teacher, researcher)
-- target:references (poly)
-  - "Education::Program" if teacher
-  - "Research::Laboratory" if researcher
-  - "University::Role" if administrator
-- description:text
-- position:integer
-
-### university/Role
-
-- university:references
-- target:references (poly)
-- description:text
-- position:integer
-- (parent:references => role) ?
-
-### university/School
-
-- university:references
-- name:string
-- address:string
-- zipcode:string
-- city:string
-- country:string
-- latitude:float
-- longitude:float
-
-### university/Campus
-
-- university:references
-- name:string
-- address:string
-- zipcode:string
-- city:string
-- country:string
-
-### university/Section
-cf https://conseil-national-des-universites.fr/cnu/
-
-- name:string
-- number:integer
diff --git a/docs/university/role.md b/docs/university/role.md
deleted file mode 100644
index 4c6d8d6f55404c228906938887e82b78e06867a8..0000000000000000000000000000000000000000
--- a/docs/university/role.md
+++ /dev/null
@@ -1,60 +0,0 @@
-# University
-
-## Roles
-
-Les personnes d'une université peuvent avoir plusieurs rôles, que ce soit au niveau de l'université ou d'un objet spécifique telle qu'une formation ou un laboratoire.
-
-Ces rôles peuvent être intrinsèques ou non. Par exemple, être enseignant dans une formation est un lien intrinsèque, alors qu'un rôle « directeur des études » dans une formation peut ne pas exister.
-
-Pour cela, on a 2 façons de créer ces liens, à partir d'un modèle commun
-
-### University::Person::Involvement
-
-Ce modèle permet de lier une personne à une cible polymorphique. On définit au niveau de l'objet si l'involvement est de type administratif, enseignant ou chercheur. On renseigne également une description et une position.
-
-### University::Role
-
-Ce modèle sert pour les liens non intrinsèques. On crée un rôle au niveau d'une cible polymorphique, description et position, et possiblement un rôle parent pour définir la hiérarchie au sein d'un organigramme.
-
-Ensuite, on connecte une personne à ce rôle en utilisant le modèle Involvement avec pour target, le rôle en question.
-
-### Exemples
-
-Soient :
-- `mmi_program` : l'objet `Education::Program` représentant le BUT MMI
-- `teacher` : l'objet `University::Person` représentant un enseignant
-- `director` : l'objet `University::Person` représentant la cheffe de département
-- `program_manager` : l'objet `University::Person` représentant le directeur des études
-- `secretary` : l'objet `University::Person` représentant le secrétaire
-
-Pour l'enseignant on crée un objet `University::Person::Involvement`:
-- target: `mmi_program`
-- person: `teacher`
-- kind: teacher
-
-Pour la cheffe de département on crée :
-- Un objet `University::Role`, qu'on nomme `director_role`
-  - target: `mmi_program`
-  - description: "Cheffe de département"
-- Un objet `University::Person::Involvement`
-  - target: `director_role`
-  - person: `director`
-  - kind: administrator
-
-Pour le directeur des études on crée :
-- Un objet `University::Role`, qu'on nomme `program_manager_role`
-  - target: `mmi_program`
-  - description: "Directeur des études"
-- Un objet `University::Person::Involvement`
-  - target: `program_manager_role`
-  - person: `program_manager`
-  - kind: administrator
-
-Pour le secrétaire on crée :
-- Un objet `University::Role`, qu'on nomme `secretary_role`
-  - target: `mmi_program`
-  - description: "Secrétaire"
-- Un objet `University::Person::Involvement`
-  - target: `secretary_role`
-  - person: `secretary`
-  - kind: administrator
diff --git a/docs/user/readme.md b/docs/user/readme.md
deleted file mode 100644
index e3e46ccad9ab7cffe667ed09c9fd01d6f781513a..0000000000000000000000000000000000000000
--- a/docs/user/readme.md
+++ /dev/null
@@ -1,15 +0,0 @@
-# User
-
-Users can manage their profiles as:
-- teachers
-- students
-- alumni
-
-https://www.iut.u-bordeaux-montaigne.fr/mon-compte
-
-## Model
-
-- university:references
-- first_name:string
-- last_name:string
-- role:integer (enum: superadmin, admin, visitor)
diff --git a/test/controllers/education/academic_years_controller_test.rb b/test/controllers/education/academic_years_controller_test.rb
new file mode 100644
index 0000000000000000000000000000000000000000..3e5bf7bdceceffbac113d450dac8ad2c3dfb1b54
--- /dev/null
+++ b/test/controllers/education/academic_years_controller_test.rb
@@ -0,0 +1,48 @@
+require "test_helper"
+
+class Education::AcademicYearsControllerTest < ActionDispatch::IntegrationTest
+  setup do
+    @education_academic_year = education_academic_years(:one)
+  end
+
+  test "should get index" do
+    get education_academic_years_url
+    assert_response :success
+  end
+
+  test "should get new" do
+    get new_education_academic_year_url
+    assert_response :success
+  end
+
+  test "should create education_academic_year" do
+    assert_difference('Education::AcademicYear.count') do
+      post education_academic_years_url, params: { education_academic_year: { university_id: @education_academic_year.university_id, year: @education_academic_year.year } }
+    end
+
+    assert_redirected_to education_academic_year_url(Education::AcademicYear.last)
+  end
+
+  test "should show education_academic_year" do
+    get education_academic_year_url(@education_academic_year)
+    assert_response :success
+  end
+
+  test "should get edit" do
+    get edit_education_academic_year_url(@education_academic_year)
+    assert_response :success
+  end
+
+  test "should update education_academic_year" do
+    patch education_academic_year_url(@education_academic_year), params: { education_academic_year: { university_id: @education_academic_year.university_id, year: @education_academic_year.year } }
+    assert_redirected_to education_academic_year_url(@education_academic_year)
+  end
+
+  test "should destroy education_academic_year" do
+    assert_difference('Education::AcademicYear.count', -1) do
+      delete education_academic_year_url(@education_academic_year)
+    end
+
+    assert_redirected_to education_academic_years_url
+  end
+end
diff --git a/test/controllers/education/cohorts_controller_test.rb b/test/controllers/education/cohorts_controller_test.rb
new file mode 100644
index 0000000000000000000000000000000000000000..e8e527ad1bbef72912c6509b815a3c842c169ab5
--- /dev/null
+++ b/test/controllers/education/cohorts_controller_test.rb
@@ -0,0 +1,48 @@
+require "test_helper"
+
+class Education::CohortsControllerTest < ActionDispatch::IntegrationTest
+  setup do
+    @education_cohort = education_cohorts(:one)
+  end
+
+  test "should get index" do
+    get education_cohorts_url
+    assert_response :success
+  end
+
+  test "should get new" do
+    get new_education_cohort_url
+    assert_response :success
+  end
+
+  test "should create education_cohort" do
+    assert_difference('Education::Cohort.count') do
+      post education_cohorts_url, params: { education_cohort: { academic_year_id: @education_cohort.academic_year_id, name: @education_cohort.name, program_id: @education_cohort.program_id, university_id: @education_cohort.university_id } }
+    end
+
+    assert_redirected_to education_cohort_url(Education::Cohort.last)
+  end
+
+  test "should show education_cohort" do
+    get education_cohort_url(@education_cohort)
+    assert_response :success
+  end
+
+  test "should get edit" do
+    get edit_education_cohort_url(@education_cohort)
+    assert_response :success
+  end
+
+  test "should update education_cohort" do
+    patch education_cohort_url(@education_cohort), params: { education_cohort: { academic_year_id: @education_cohort.academic_year_id, name: @education_cohort.name, program_id: @education_cohort.program_id, university_id: @education_cohort.university_id } }
+    assert_redirected_to education_cohort_url(@education_cohort)
+  end
+
+  test "should destroy education_cohort" do
+    assert_difference('Education::Cohort.count', -1) do
+      delete education_cohort_url(@education_cohort)
+    end
+
+    assert_redirected_to education_cohorts_url
+  end
+end
diff --git a/test/controllers/university/person/alumnus/imports_controller_test.rb b/test/controllers/university/person/alumnus/imports_controller_test.rb
new file mode 100644
index 0000000000000000000000000000000000000000..8585a49c7011147fefa9cbd786ca351bb596c732
--- /dev/null
+++ b/test/controllers/university/person/alumnus/imports_controller_test.rb
@@ -0,0 +1,48 @@
+require "test_helper"
+
+class University::Person::Alumnus::ImportsControllerTest < ActionDispatch::IntegrationTest
+  setup do
+    @university_person_alumnus_import = university_person_alumnus_imports(:one)
+  end
+
+  test "should get index" do
+    get university_person_alumnus_imports_url
+    assert_response :success
+  end
+
+  test "should get new" do
+    get new_university_person_alumnus_import_url
+    assert_response :success
+  end
+
+  test "should create university_person_alumnus_import" do
+    assert_difference('University::Person::Alumnus::Import.count') do
+      post university_person_alumnus_imports_url, params: { university_person_alumnus_import: { university_id: @university_person_alumnus_import.university_id, user_id: @university_person_alumnus_import.user_id } }
+    end
+
+    assert_redirected_to university_person_alumnus_import_url(University::Person::Alumnus::Import.last)
+  end
+
+  test "should show university_person_alumnus_import" do
+    get university_person_alumnus_import_url(@university_person_alumnus_import)
+    assert_response :success
+  end
+
+  test "should get edit" do
+    get edit_university_person_alumnus_import_url(@university_person_alumnus_import)
+    assert_response :success
+  end
+
+  test "should update university_person_alumnus_import" do
+    patch university_person_alumnus_import_url(@university_person_alumnus_import), params: { university_person_alumnus_import: { university_id: @university_person_alumnus_import.university_id, user_id: @university_person_alumnus_import.user_id } }
+    assert_redirected_to university_person_alumnus_import_url(@university_person_alumnus_import)
+  end
+
+  test "should destroy university_person_alumnus_import" do
+    assert_difference('University::Person::Alumnus::Import.count', -1) do
+      delete university_person_alumnus_import_url(@university_person_alumnus_import)
+    end
+
+    assert_redirected_to university_person_alumnus_imports_url
+  end
+end
diff --git a/test/fixtures/education/academic_years.yml b/test/fixtures/education/academic_years.yml
new file mode 100644
index 0000000000000000000000000000000000000000..2a7ea54e6ec792f1e370cda239c158f95f011a32
--- /dev/null
+++ b/test/fixtures/education/academic_years.yml
@@ -0,0 +1,26 @@
+# == Schema Information
+#
+# Table name: education_academic_years
+#
+#  id            :uuid             not null, primary key
+#  year          :integer
+#  created_at    :datetime         not null
+#  updated_at    :datetime         not null
+#  university_id :uuid             not null, indexed
+#
+# Indexes
+#
+#  index_education_academic_years_on_university_id  (university_id)
+#
+# Foreign Keys
+#
+#  fk_rails_7d376afe35  (university_id => universities.id)
+#
+
+one:
+  university: one
+  year: 1
+
+two:
+  university: two
+  year: 1
diff --git a/test/fixtures/education/cohorts.yml b/test/fixtures/education/cohorts.yml
new file mode 100644
index 0000000000000000000000000000000000000000..48a543db806f22f20ffa9dc40bd512758f52a9e5
--- /dev/null
+++ b/test/fixtures/education/cohorts.yml
@@ -0,0 +1,36 @@
+# == Schema Information
+#
+# Table name: education_cohorts
+#
+#  id               :uuid             not null, primary key
+#  name             :string
+#  created_at       :datetime         not null
+#  updated_at       :datetime         not null
+#  academic_year_id :uuid             not null, indexed
+#  program_id       :uuid             not null, indexed
+#  university_id    :uuid             not null, indexed
+#
+# Indexes
+#
+#  index_education_cohorts_on_academic_year_id  (academic_year_id)
+#  index_education_cohorts_on_program_id        (program_id)
+#  index_education_cohorts_on_university_id     (university_id)
+#
+# Foreign Keys
+#
+#  fk_rails_0f4a4f43d9  (university_id => universities.id)
+#  fk_rails_72528c3d76  (program_id => education_programs.id)
+#  fk_rails_c2d725cabd  (academic_year_id => education_academic_years.id)
+#
+
+one:
+  university: one
+  program: one
+  academic_year: one
+  name: MyString
+
+two:
+  university: two
+  program: two
+  academic_year: two
+  name: MyString
diff --git a/test/fixtures/university/person/alumnus/imports.yml b/test/fixtures/university/person/alumnus/imports.yml
new file mode 100644
index 0000000000000000000000000000000000000000..f8533f0e62b8768e40b987dc28aad4f609d1950a
--- /dev/null
+++ b/test/fixtures/university/person/alumnus/imports.yml
@@ -0,0 +1,28 @@
+# == Schema Information
+#
+# Table name: university_person_alumnus_imports
+#
+#  id            :uuid             not null, primary key
+#  created_at    :datetime         not null
+#  updated_at    :datetime         not null
+#  university_id :uuid             not null, indexed
+#  user_id       :uuid             not null, indexed
+#
+# Indexes
+#
+#  index_university_person_alumnus_imports_on_university_id  (university_id)
+#  index_university_person_alumnus_imports_on_user_id        (user_id)
+#
+# Foreign Keys
+#
+#  fk_rails_3ff74ac195  (user_id => users.id)
+#  fk_rails_d14eb003f9  (university_id => universities.id)
+#
+
+one:
+  university: one
+  user: one
+
+two:
+  university: two
+  user: two
diff --git a/test/models/education/academic_year_test.rb b/test/models/education/academic_year_test.rb
new file mode 100644
index 0000000000000000000000000000000000000000..f5535a748800fef1a826d92880c4969576a154c5
--- /dev/null
+++ b/test/models/education/academic_year_test.rb
@@ -0,0 +1,25 @@
+# == Schema Information
+#
+# Table name: education_academic_years
+#
+#  id            :uuid             not null, primary key
+#  year          :integer
+#  created_at    :datetime         not null
+#  updated_at    :datetime         not null
+#  university_id :uuid             not null, indexed
+#
+# Indexes
+#
+#  index_education_academic_years_on_university_id  (university_id)
+#
+# Foreign Keys
+#
+#  fk_rails_7d376afe35  (university_id => universities.id)
+#
+require "test_helper"
+
+class Education::AcademicYearTest < ActiveSupport::TestCase
+  # test "the truth" do
+  #   assert true
+  # end
+end
diff --git a/test/models/education/cohort_test.rb b/test/models/education/cohort_test.rb
new file mode 100644
index 0000000000000000000000000000000000000000..2bb945a4269462f09d294da1e8c6b96f50241a05
--- /dev/null
+++ b/test/models/education/cohort_test.rb
@@ -0,0 +1,31 @@
+# == Schema Information
+#
+# Table name: education_cohorts
+#
+#  id               :uuid             not null, primary key
+#  name             :string
+#  created_at       :datetime         not null
+#  updated_at       :datetime         not null
+#  academic_year_id :uuid             not null, indexed
+#  program_id       :uuid             not null, indexed
+#  university_id    :uuid             not null, indexed
+#
+# Indexes
+#
+#  index_education_cohorts_on_academic_year_id  (academic_year_id)
+#  index_education_cohorts_on_program_id        (program_id)
+#  index_education_cohorts_on_university_id     (university_id)
+#
+# Foreign Keys
+#
+#  fk_rails_0f4a4f43d9  (university_id => universities.id)
+#  fk_rails_72528c3d76  (program_id => education_programs.id)
+#  fk_rails_c2d725cabd  (academic_year_id => education_academic_years.id)
+#
+require "test_helper"
+
+class Education::CohortTest < ActiveSupport::TestCase
+  # test "the truth" do
+  #   assert true
+  # end
+end
diff --git a/test/models/university/person/alumnus/import_test.rb b/test/models/university/person/alumnus/import_test.rb
new file mode 100644
index 0000000000000000000000000000000000000000..e8757e980ee1e9afff1818e585439882863a3246
--- /dev/null
+++ b/test/models/university/person/alumnus/import_test.rb
@@ -0,0 +1,27 @@
+# == Schema Information
+#
+# Table name: university_person_alumnus_imports
+#
+#  id            :uuid             not null, primary key
+#  created_at    :datetime         not null
+#  updated_at    :datetime         not null
+#  university_id :uuid             not null, indexed
+#  user_id       :uuid             not null, indexed
+#
+# Indexes
+#
+#  index_university_person_alumnus_imports_on_university_id  (university_id)
+#  index_university_person_alumnus_imports_on_user_id        (user_id)
+#
+# Foreign Keys
+#
+#  fk_rails_3ff74ac195  (user_id => users.id)
+#  fk_rails_d14eb003f9  (university_id => universities.id)
+#
+require "test_helper"
+
+class University::Person::Alumnus::ImportTest < ActiveSupport::TestCase
+  # test "the truth" do
+  #   assert true
+  # end
+end
diff --git a/test/system/education/academic_years_test.rb b/test/system/education/academic_years_test.rb
new file mode 100644
index 0000000000000000000000000000000000000000..68b9d54ab2803acbe86d5c3c79293011e9077836
--- /dev/null
+++ b/test/system/education/academic_years_test.rb
@@ -0,0 +1,45 @@
+require "application_system_test_case"
+
+class Education::AcademicYearsTest < ApplicationSystemTestCase
+  setup do
+    @education_academic_year = education_academic_years(:one)
+  end
+
+  test "visiting the index" do
+    visit education_academic_years_url
+    assert_selector "h1", text: "Education/Academic Years"
+  end
+
+  test "creating a Academic year" do
+    visit education_academic_years_url
+    click_on "New Education/Academic Year"
+
+    fill_in "University", with: @education_academic_year.university_id
+    fill_in "Year", with: @education_academic_year.year
+    click_on "Create Academic year"
+
+    assert_text "Academic year was successfully created"
+    click_on "Back"
+  end
+
+  test "updating a Academic year" do
+    visit education_academic_years_url
+    click_on "Edit", match: :first
+
+    fill_in "University", with: @education_academic_year.university_id
+    fill_in "Year", with: @education_academic_year.year
+    click_on "Update Academic year"
+
+    assert_text "Academic year was successfully updated"
+    click_on "Back"
+  end
+
+  test "destroying a Academic year" do
+    visit education_academic_years_url
+    page.accept_confirm do
+      click_on "Destroy", match: :first
+    end
+
+    assert_text "Academic year was successfully destroyed"
+  end
+end
diff --git a/test/system/education/cohorts_test.rb b/test/system/education/cohorts_test.rb
new file mode 100644
index 0000000000000000000000000000000000000000..2ed293473e1a44cbffdddfd1b614b8848a703711
--- /dev/null
+++ b/test/system/education/cohorts_test.rb
@@ -0,0 +1,49 @@
+require "application_system_test_case"
+
+class Education::CohortsTest < ApplicationSystemTestCase
+  setup do
+    @education_cohort = education_cohorts(:one)
+  end
+
+  test "visiting the index" do
+    visit education_cohorts_url
+    assert_selector "h1", text: "Education/Cohorts"
+  end
+
+  test "creating a Cohort" do
+    visit education_cohorts_url
+    click_on "New Education/Cohort"
+
+    fill_in "Academic year", with: @education_cohort.academic_year_id
+    fill_in "Name", with: @education_cohort.name
+    fill_in "Program", with: @education_cohort.program_id
+    fill_in "University", with: @education_cohort.university_id
+    click_on "Create Cohort"
+
+    assert_text "Cohort was successfully created"
+    click_on "Back"
+  end
+
+  test "updating a Cohort" do
+    visit education_cohorts_url
+    click_on "Edit", match: :first
+
+    fill_in "Academic year", with: @education_cohort.academic_year_id
+    fill_in "Name", with: @education_cohort.name
+    fill_in "Program", with: @education_cohort.program_id
+    fill_in "University", with: @education_cohort.university_id
+    click_on "Update Cohort"
+
+    assert_text "Cohort was successfully updated"
+    click_on "Back"
+  end
+
+  test "destroying a Cohort" do
+    visit education_cohorts_url
+    page.accept_confirm do
+      click_on "Destroy", match: :first
+    end
+
+    assert_text "Cohort was successfully destroyed"
+  end
+end
diff --git a/test/system/university/person/alumnus/imports_test.rb b/test/system/university/person/alumnus/imports_test.rb
new file mode 100644
index 0000000000000000000000000000000000000000..1f10cd1aaaa7814bb3fa822293d0faf11efa69ae
--- /dev/null
+++ b/test/system/university/person/alumnus/imports_test.rb
@@ -0,0 +1,45 @@
+require "application_system_test_case"
+
+class University::Person::Alumnus::ImportsTest < ApplicationSystemTestCase
+  setup do
+    @university_person_alumnus_import = university_person_alumnus_imports(:one)
+  end
+
+  test "visiting the index" do
+    visit university_person_alumnus_imports_url
+    assert_selector "h1", text: "University/Person/Alumnus/Imports"
+  end
+
+  test "creating a Import" do
+    visit university_person_alumnus_imports_url
+    click_on "New University/Person/Alumnus/Import"
+
+    fill_in "University", with: @university_person_alumnus_import.university_id
+    fill_in "User", with: @university_person_alumnus_import.user_id
+    click_on "Create Import"
+
+    assert_text "Import was successfully created"
+    click_on "Back"
+  end
+
+  test "updating a Import" do
+    visit university_person_alumnus_imports_url
+    click_on "Edit", match: :first
+
+    fill_in "University", with: @university_person_alumnus_import.university_id
+    fill_in "User", with: @university_person_alumnus_import.user_id
+    click_on "Update Import"
+
+    assert_text "Import was successfully updated"
+    click_on "Back"
+  end
+
+  test "destroying a Import" do
+    visit university_person_alumnus_imports_url
+    page.accept_confirm do
+      click_on "Destroy", match: :first
+    end
+
+    assert_text "Import was successfully destroyed"
+  end
+end