diff --git a/greg/management/commands/import_usernames_from_cerebrum.py b/greg/management/commands/import_usernames_from_cerebrum.py
new file mode 100644
index 0000000000000000000000000000000000000000..fc27a115614a8c501f136bab7de43027db6bff0c
--- /dev/null
+++ b/greg/management/commands/import_usernames_from_cerebrum.py
@@ -0,0 +1,104 @@
+"""
+Command for running or scheduling the django-q task for importing
+usernames from Cerebrum
+"""
+import logging
+
+from django.conf import settings
+from django.core.management.base import BaseCommand, CommandParser
+from django.db import transaction
+from django_q.tasks import schedule
+
+from greg.tasks.usernames import UsernameImporter
+from greg.models import Identity, Person
+
+logger = logging.getLogger(__name__)
+
+
+class Command(BaseCommand):
+    help = __doc__
+
+    def add_arguments(self, parser: CommandParser) -> None:
+        group = parser.add_mutually_exclusive_group()
+        group.add_argument(
+            "--all-persons",
+            action="store_true",
+            default=False,
+            help="Import username for all persons in GREG",
+        )
+        group.add_argument(
+            "--without-usernames",
+            action="store_true",
+            default=False,
+            help="Only import those without usernames in GREG",
+        )
+        group.add_argument(
+            "--with-usernames",
+            action="store_true",
+            default=False,
+            help="Only import those with usernames in GREG",
+        )
+        parser.add_argument(
+            "--schedule",
+            action="store_true",
+            default=False,
+            help="Add a scheduled task for running the import",
+        )
+        parser.add_argument(
+            "--commit",
+            action="store_true",
+            default=False,
+            help="Commit the changes done by running the import once",
+        )
+
+    def schedule_task(self, all_persons, schedule_type, logger_msg):
+        logger.info(logger_msg)
+        schedule(
+            func="greg.tasks.usernames.import_usernames",
+            persons=all_persons,
+            schedule_type=schedule_type,
+        )
+        logger.info("Username import scheduled")
+
+    def handle(self, *args, **options):
+        """Schedule or run import of usernames"""
+
+        if options["with_usernames"]:
+            all_persons = Person.objects.filter(
+                identities__type=Identity.IdentityType.FEIDE_ID,
+                identities__source="cerebrum",
+            )
+            schedule_type = settings.USERNAME_SCHEDULE_WITH_USERNAME
+            logger_msg = "Scheduling import of usernames for persons with username..."
+        elif options["without_usernames"]:
+            all_persons = Person.objects.exclude(
+                identities__type=Identity.IdentityType.FEIDE_ID,
+                identities__source="cerebrum",
+            )
+            schedule_type = settings.USERNAME_SCHEDULE_WITHOUT_USERNAME
+            logger_msg = (
+                "Scheduling import of usernames for persons without username..."
+            )
+        else:
+            all_persons = Person.objects.all()
+            schedule_type = settings.USERNAME_SCHEDULE_ALL
+            logger_msg = "Scheduling import of usernames for all persons..."
+
+        if options["schedule"]:
+            self.schedule_task(
+                all_persons=all_persons,
+                schedule_type=schedule_type,
+                logger_msg=logger_msg,
+            )
+            return
+
+        # Will assume to run once if schedule is not given
+        with transaction.atomic():
+            logger.info("Importing usernames...")
+            importer = UsernameImporter()
+            importer.update_usernames(all_persons)
+            if not options["commit"]:
+                logger.info("Rolling back changes...")
+                transaction.set_rollback(True)
+            else:
+                logger.info("Changes committed")
diff --git a/greg/migrations/0030_remove_identity_identity_type_value_unique_and_more.py b/greg/migrations/0030_remove_identity_identity_type_value_unique_and_more.py
new file mode 100644
index 0000000000000000000000000000000000000000..676362b1671fd3b228b1889798f700afa9fb00da
--- /dev/null
+++ b/greg/migrations/0030_remove_identity_identity_type_value_unique_and_more.py
@@ -0,0 +1,22 @@
+# Generated by Django 4.1.4 on 2023-08-25 12:10
+
+from django.db import migrations, models
+
+
+class Migration(migrations.Migration):
+    dependencies = [
+        ("greg", "0029_alter_identity_type_add_national_id_card_number"),
+    ]
+
+    operations = [
+        migrations.RemoveConstraint(
+            model_name="identity",
+            name="identity_type_value_unique",
+        ),
+        migrations.AddConstraint(
+            model_name="identity",
+            constraint=models.UniqueConstraint(
+                fields=("type", "value", "source"), name="identity_type_value_unique"
+            ),
+        ),
+    ]
diff --git a/greg/models.py b/greg/models.py
index 7149144d31a91c5b239acaa522f432c4fae9ffd2..70a3af830cad656fc4de3e6723888413a5c52445 100644
--- a/greg/models.py
+++ b/greg/models.py
@@ -344,7 +344,7 @@ class Identity(BaseModel):
         verbose_name_plural = "identities"
         constraints = (
             models.UniqueConstraint(
-                fields=["type", "value"],
+                fields=["type", "value", "source"],
                 name="identity_type_value_unique",
             ),
         )
diff --git a/greg/tasks/usernames.py b/greg/tasks/usernames.py
new file mode 100644
index 0000000000000000000000000000000000000000..537173042a2172a5564eb74ca2f9e0163bf42fbd
--- /dev/null
+++ b/greg/tasks/usernames.py
@@ -0,0 +1,116 @@
+import logging
+
+from cerebrum_client import CerebrumClient
+from django.conf import settings
+
+from greg.models import Identity
+
+logger = logging.getLogger(__name__)
+
+
+class UsernameImporter:
+    def __init__(self):
+        self.client = CerebrumClient(**settings.CEREBRUM_CLIENT)
+
+        self.source = "GREG"
+        self.cerebrum_id_type = "gregPersonId"
+        self.id_type = Identity.IdentityType.FEIDE_ID
+
+    def get_username(self, person):
+        person_id = person.person_id
+        username = self.client.get_person_primary_account_name(person_id)
+        return username
+
+    def update_or_create_person_feide_id(self, person, username):
+        existing_feide_id = person.identities.filter(
+            type=self.id_type, source="cerebrum"
+        )
+        feide_id = f"{username}@{settings.INSTANCE_NAME}.no"
+        if not existing_feide_id:
+            logger.info("Creating Feide-ID for person_id=%s", person.pk)
+            Identity.objects.create(
+                person=person,
+                type=self.id_type,
+                source="cerebrum",
+                value=feide_id,
+                verified=Identity.Verified.AUTOMATIC,
+            )
+            return
+        existing_feide_id_obj = existing_feide_id.first()
+        existing_feide_id_value = existing_feide_id_obj.value
+        old_username, _ = existing_feide_id_value.split("@")
+        if old_username != username:
+            logger.info(
+                "Updating Feide-ID for person_id=%s from %s to %s",
+                person.pk,
+                existing_feide_id_value,
+                feide_id,
+            )
+            existing_feide_id.update(value=feide_id)
+
+    def update_usernames(self, all_persons):
+        for person in all_persons:
+            logger.info(
+                "Looking up username for person_id=%s from cerebrum",
+                person.pk,
+            )
+            try:
+                persons = list(
+                    self.client.search_person_external_ids(
+                        source_system=self.source,
+                        id_type=self.cerebrum_id_type,
+                        external_id=person.pk,
+                    )
+                )
+            except ConnectionError as e:  # noqa: W0703
+                logger.exception(
+                    "Failed to get person with person_id=%s. Got error=%s. Skipping person",
+                    person.pk,
+                    e,
+                )
+                continue
+            if not persons:
+                logger.info(
+                    "Found no person in cerebrum matching id_type=%s and person_id=%s from source=%s. Skipping person",
+                    self.cerebrum_id_type,
+                    person.pk,
+                    self.source,
+                )
+                continue
+            if len(persons) > 1:
+                logger.info(
+                    "Found %s persons with same id_type=%s and person_id=%s in source_system=%s. Skipping person",
+                    len(persons),
+                    self.cerebrum_id_type,
+                    person.pk,
+                    self.source,
+                )
+                continue
+
+            try:
+                username = self.get_username(persons[0])
+            except ConnectionError as e:
+                logger.exception(
+                    "Failed to get username for person_id=%s. Got error=%s. Skipping person",
+                    person.pk,
+                    e,
+                )
+                continue
+            if not username:
+                logger.info(
+                    "Found no username in cerebrum for person_id=%s. Skipping person",
+                    person.pk,
+                )
+                continue
+
+            logger.info(
+                "Found username=%s for person_id=%s in cerebrum.",
+                username,
+                person.pk,
+            )
+            self.update_or_create_person_feide_id(person, username)
+
+
+def update_usernames(persons):
+    importer = UsernameImporter()
+    importer.update_usernames(persons)
diff --git a/greg/tests/conftest.py b/greg/tests/conftest.py
index 22f0ddff674a7c86721619384d01767d38006152..bcf3ba58062ff5cc633392219f86f73c57210077 100644
--- a/greg/tests/conftest.py
+++ b/greg/tests/conftest.py
@@ -113,6 +113,25 @@ def person_bar() -> Person:
     return Person.objects.get(id=person.id)
 
 
+@pytest.fixture
+def person_foobar() -> Person:
+    with transaction.atomic():
+        person = Person.objects.create(
+            first_name="Foo", last_name="Bar", date_of_birth="1999-02-05"
+        )
+        Identity.objects.create(
+            person=person,
+            type=Identity.IdentityType.PRIVATE_EMAIL,
+            value="foobar@example.org",
+        )
+        Identity.objects.create(
+            person=person,
+            type=Identity.IdentityType.PRIVATE_MOBILE_NUMBER,
+            value="+4712345689",
+        )
+    return Person.objects.get(id=person.id)
+
+
 @pytest.fixture
 def sponsor_guy() -> Sponsor:
     sp = Sponsor.objects.create(
diff --git a/greg/tests/management/test_import_usernames.py b/greg/tests/management/test_import_usernames.py
new file mode 100644
index 0000000000000000000000000000000000000000..16735e1aded43d69507834df39e615754cc36255
--- /dev/null
+++ b/greg/tests/management/test_import_usernames.py
@@ -0,0 +1,291 @@
+import pytest
+
+from django.core.management import call_command
+from django.core.management.base import CommandError
+from django.test import override_settings
+from django_q.tasks import Schedule
+
+from greg.models import Person, Identity
+
+
+@pytest.fixture
+def search_response():
+    return {
+        "external_ids": [
+            {
+                "person_id": 1,
+                "source_system": "GREG",
+                "external_id": 1,
+                "id_type": "gregPersonId",
+            }
+        ]
+    }
+
+
+@pytest.fixture
+def search_response_bar():
+    return {
+        "external_ids": [
+            {
+                "person_id": 2,
+                "source_system": "GREG",
+                "external_id": 2,
+                "id_type": "gregPersonId",
+            }
+        ]
+    }
+
+
+@pytest.fixture
+def search_response_foobar():
+    return {
+        "external_ids": [
+            {
+                "person_id": 3,
+                "source_system": "GREG",
+                "external_id": 3,
+                "id_type": "gregPersonId",
+            }
+        ]
+    }
+
+
+@pytest.fixture
+def person_one(person_foo):
+    Identity.objects.create(
+        person=person_foo,
+        type=Identity.IdentityType.PASSPORT_NUMBER,
+        value="HHH111111",
+    )
+    return Person.objects.get(id=person_foo.id)
+
+
+@pytest.fixture
+def person_two(person_bar):
+    Identity.objects.create(
+        person=person_bar,
+        type=Identity.IdentityType.FEIDE_ID,
+        source="cerebrum",
+        value="not_bar@inst.no",
+    )
+    return Person.objects.get(id=person_bar.id)
+
+
+@pytest.fixture
+def person_three(person_foobar):
+    Identity.objects.create(
+        person=person_foobar,
+        type=Identity.IdentityType.FEIDE_ID,
+        source="feide",
+        value="foobar@test.no",
+    )
+    return Person.objects.get(id=person_foobar.id)
+
+
+@pytest.fixture
+def person_account_response():
+    return {
+        "accounts": [
+            {
+                "href": "",
+                "primary": True,
+                "id": 1,
+                "name": "foo",
+            }
+        ]
+    }
+
+
+@pytest.fixture
+def person_account_response_bar():
+    return {
+        "accounts": [
+            {
+                "href": "",
+                "primary": True,
+                "id": 1,
+                "name": "bar",
+            }
+        ]
+    }
+
+
+@pytest.fixture
+def person_account_response_foobar():
+    return {
+        "accounts": [
+            {
+                "href": "",
+                "primary": True,
+                "id": 2,
+                "name": "foobar",
+            }
+        ]
+    }
+
+
+@pytest.mark.django_db
+def test_schedule(person_one, person_two, person_three):
+    assert Schedule.objects.all().count() == 0
+    call_command("import_usernames_from_cerebrum", "--all-persons", "--schedule")
+    assert Schedule.objects.all().count() == 1
+    sc = Schedule.objects.first()
+    assert sc.func == "greg.tasks.usernames.import_usernames"
+    call_command("import_usernames_from_cerebrum", "--without-usernames", "--schedule")
+    assert Schedule.objects.all().count() == 2
+    call_command("import_usernames_from_cerebrum", "--with-username", "--schedule")
+    assert Schedule.objects.all().count() == 3
+
+
+@pytest.mark.django_db
+def test_only_one_import_argument_allowed():
+    with pytest.raises(CommandError):
+        call_command(
+            "import_usernames_from_cerebrum",
+            "--without-usernames",
+            "--with-username",
+            "--schedule",
+        )
+
+
+@override_settings(INSTANCE_NAME="inst")
+@pytest.mark.django_db
+def test_import_wo_feide_id(
+    requests_mock,
+    search_response,
+    person_account_response,
+    person_one,
+):
+    requests_mock.get(
+        "http://example.com/cerebrum/v1/search/persons/external-ids?source_system=GREG&id_type=gregPersonId&external_id=1",
+        json=search_response,
+    )
+    requests_mock.get(
+        "http://example.com/cerebrum/v1/persons/1/accounts",
+        json=person_account_response,
+    )
+    usr_id_type = Identity.IdentityType.FEIDE_ID
+    assert not person_one.identities.filter(type=usr_id_type)
+
+    call_command("import_usernames_from_cerebrum", "--without-usernames", "--commit")
+    person_identities = person_one.identities.filter(type=usr_id_type)
+    assert len(person_identities) == 1
+    assert person_identities[0].value == "foo@inst.no"
+
+
+@override_settings(INSTANCE_NAME="inst")
+@pytest.mark.django_db
+def test_import_w_feide_id(
+    requests_mock,
+    search_response_bar,
+    person_account_response_bar,
+    person_two,
+):
+    requests_mock.get(
+        "http://example.com/cerebrum/v1/search/persons/external-ids?source_system=GREG&id_type=gregPersonId&external_id=1",
+        json=search_response_bar,
+    )
+    requests_mock.get(
+        "http://example.com/cerebrum/v1/persons/2/accounts",
+        json=person_account_response_bar,
+    )
+    usr_id_type = Identity.IdentityType.FEIDE_ID
+    person_identities = person_two.identities.filter(type=usr_id_type)
+    assert len(person_identities) == 1
+    assert person_identities[0].value == "not_bar@inst.no"
+    call_command("import_usernames_from_cerebrum", "--with-usernames", "--commit")
+    person_identities = person_two.identities.filter(type=usr_id_type)
+    assert len(person_identities) == 1
+    assert person_identities[0].value == "bar@inst.no"
+
+
+@override_settings(INSTANCE_NAME="inst")
+@pytest.mark.django_db
+def test_different_feide_id_already_exists(
+    requests_mock,
+    search_response_foobar,
+    person_account_response_foobar,
+    person_three,
+):
+    requests_mock.get(
+        "http://example.com/cerebrum/v1/search/persons/external-ids?source_system=GREG&id_type=gregPersonId&external_id=1",
+        json=search_response_foobar,
+    )
+    requests_mock.get(
+        "http://example.com/cerebrum/v1/persons/3/accounts",
+        json=person_account_response_foobar,
+    )
+    usr_id_type = Identity.IdentityType.FEIDE_ID
+    person_identities = person_three.identities.filter(type=usr_id_type)
+    assert len(person_identities) == 1
+    assert person_identities[0].value == "foobar@test.no"
+    call_command("import_usernames_from_cerebrum", "--without-usernames", "--commit")
+    person_identities = person_three.identities.filter(type=usr_id_type)
+    print(person_identities)
+    assert len(person_identities) == 2
+    assert person_identities[0].value == "foobar@test.no"
+    assert person_identities[1].value == "foobar@inst.no"
+
+
+@override_settings(INSTANCE_NAME="inst")
+@pytest.mark.django_db
+def test_import_usernames(
+    requests_mock,
+    search_response,
+    search_response_bar,
+    search_response_foobar,
+    person_account_response,
+    person_account_response_bar,
+    person_account_response_foobar,
+    person_one,
+    person_two,
+    person_three,
+):
+    """
+    This test is mostly just all the other three tests in one to check that all-persons does all the same stuff
+    """
+    requests_mock.get(
+        "http://example.com/cerebrum/v1/search/persons/external-ids?source_system=GREG&id_type=gregPersonId&external_id=1",
+        json=search_response,
+    )
+    requests_mock.get(
+        "http://example.com/cerebrum/v1/search/persons/external-ids?source_system=GREG&id_type=gregPersonId&external_id=2",
+        json=search_response_bar,
+    )
+    requests_mock.get(
+        "http://example.com/cerebrum/v1/search/persons/external-ids?source_system=GREG&id_type=gregPersonId&external_id=3",
+        json=search_response_foobar,
+    )
+    requests_mock.get(
+        "http://example.com/cerebrum/v1/persons/1/accounts",
+        json=person_account_response,
+    )
+    requests_mock.get(
+        "http://example.com/cerebrum/v1/persons/2/accounts",
+        json=person_account_response_bar,
+    )
+    requests_mock.get(
+        "http://example.com/cerebrum/v1/persons/3/accounts",
+        json=person_account_response_foobar,
+    )
+    usr_id_type = Identity.IdentityType.FEIDE_ID
+    person_identities_two = person_two.identities.filter(type=usr_id_type)
+    person_identities_three = person_three.identities.filter(type=usr_id_type)
+    assert not person_one.identities.filter(type=usr_id_type)
+    assert len(person_identities_two) == 1
+    assert person_identities_two[0].value == "not_bar@inst.no"
+    assert len(person_identities_three) == 1
+    assert person_identities_three[0].value == "foobar@test.no"
+
+    call_command("import_usernames_from_cerebrum", "--all-persons", "--commit")
+    person_identities_one = person_one.identities.filter(type=usr_id_type)
+    person_identities_two = person_two.identities.filter(type=usr_id_type)
+    person_identities_three = person_three.identities.filter(type=usr_id_type)
+
+    assert len(person_identities_one) == 1
+    assert person_identities_one[0].value == "foo@inst.no"
+    assert len(person_identities_two) == 1
+    assert person_identities_two[0].value == "bar@inst.no"
+    assert len(person_identities_three) == 2
+    assert person_identities_three[0].value == "foobar@test.no"
+    assert person_identities_three[1].value == "foobar@inst.no"
diff --git a/gregsite/settings/base.py b/gregsite/settings/base.py
index dc8e9ae533542a38f19eeac3a4c5522d19d4085e..f9b878b4e44e016d15f2c0e3a961dfa90fcab698 100644
--- a/gregsite/settings/base.py
+++ b/gregsite/settings/base.py
@@ -317,6 +317,11 @@ ORGREG_ACRONYMS = []
 # scheduled by running the management command
 ORGREG_SCHEDULE_TYPE = "D"
 
+# Controls how often the usernames import should be scheduled
+USERNAME_SCHEDULE_ALL = "M"
+USERNAME_SCHEDULE_WITHOUT_USERNAME = "H"
+USERNAME_SCHEDULE_WITH_USERNAME = "D"
+
 # List of stedkode for units where Sponsors are handled manually
 CEREBRUM_MANUAL_SPONSOR_UNITS = []