diff --git a/cms/djangoapps/contentstore/views/course.py b/cms/djangoapps/contentstore/views/course.py index 9f6cfb7c430e..5bc01f6414bb 100644 --- a/cms/djangoapps/contentstore/views/course.py +++ b/cms/djangoapps/contentstore/views/course.py @@ -673,15 +673,40 @@ def library_listing(request): def _format_library_for_view(library, request): """ Return a dict of the data which the view requires for each library - """ + @@TODO This is a hacky prototype implementation. In a real implementation, we'd probably want to change the schema + of this API to include both the old an (if migrate) new library metadata, and then leave the messaging and + URL-building logic to frontend-app-authoring. + """ + # @@TODO: Either put ContentLibraryMigration behind a Python API, or move it to the contentstore app. + # That way, this app doesn't need to import from another app's models. + from openedx.core.djangoapps.content_libraries.models import ContentLibraryMigration + try: + migration = ContentLibraryMigration.objects.select_related( + "target", "target__learning_package", "target_collection" + ).get(source_key=library.id) + except ContentLibraryMigration.DoesNotExist: + # Library is not yet migrated. Point to legacy legacyy. + display_name = library.display_name + url = reverse_library_url('library_handler', str(library.location.library_key)) + key_for_access_check = library.context_key + else: + url = f"{settings.COURSE_AUTHORING_MICROFRONTEND_URL}/library/{migration.target.library_key}" + new_library_title = migration.target.learning_package.title + if migration.target_collection: + url = f"{url}/collection/{migration.target_collection.key}" + collection_title = migration.target_collection.title + display_name = f"{library.display_name} (migrated to '{collection_title}' in '{new_library_title}')" + else: + display_name = f"{library.display_name} (migrated to '{new_library_title}')" + key_for_access_check = migration.target.library_key return { - 'display_name': library.display_name, + 'display_name': display_name, + 'url': url, 'library_key': str(library.location.library_key), - 'url': reverse_library_url('library_handler', str(library.location.library_key)), 'org': library.display_org_with_default, 'number': library.display_number_with_default, - 'can_edit': has_studio_write_access(request.user, library.location.library_key), + 'can_edit': has_studio_write_access(request.user, key_for_access_check), } diff --git a/cms/lib/xblock/upstream_sync.py b/cms/lib/xblock/upstream_sync.py index 0d95931ce29d..fb74cc80c0ad 100644 --- a/cms/lib/xblock/upstream_sync.py +++ b/cms/lib/xblock/upstream_sync.py @@ -146,16 +146,47 @@ def get_for_block(cls, downstream: XBlock) -> t.Self: If link exists, is supported, and is followable, returns UpstreamLink. Otherwise, raises an UpstreamLinkException. """ - if not downstream.upstream: + if downstream.upstream: + if not isinstance(downstream.usage_key.context_key, CourseKey): + raise BadDownstream(_("Cannot update content because it does not belong to a course.")) + if downstream.has_children: + raise BadDownstream(_("Updating content with children is not yet supported.")) + + # We need to determine the usage key of this block's upstream. + upstream_key: LibraryUsageLocatorV2 + version_synced: int | None + version_available: int | None + # A few different scenarios... + + # Do we have an upstream explicitly defined on the block? If so, use that. + if downstream.upstream: + try: + upstream_key = LibraryUsageLocatorV2.from_string(downstream.upstream) + except InvalidKeyError as exc: + raise BadUpstream(_("Reference to linked library item is malformed")) from exc + version_synced = downstream.upstream_version + version_declined = downstream.upstream_version_declined + + # Otherwise, is this the child of a LegacyLibraryContentBlock? + # If so, then we know that this block was derived from block in a legacy (v1) content library. + # Try to get that block's migrated (v2) content library equivalent and use it as our upstream. + elif downstream.parent and downstream.parent.block_type == "library_content": + from xmodule.library_content_block import LegacyLibraryContentBlock + parent: LegacyLibraryContentBlock = downstream.get_parent() + # Next line will raise UpstreamLinkException if no matching V2 library block. + upstream_key = parent.get_migrated_upstream_for_child(downstream.usage_key.block_id) + # If we are here, then there is indeed a migrated V2 library block, but we have not yet synced from it + # (otherwise `.upstream` would have been explicitly set). So, it is fair to set the version information + # to "None". That way, as soon as an updated version of the migrated upstream is published, it will be + # available to the course author. + version_synced = None + version_declined = None + + # Otherwise, we don't have an upstream. Raise. + else: raise NoUpstream() - if not isinstance(downstream.usage_key.context_key, CourseKey): - raise BadDownstream(_("Cannot update content because it does not belong to a course.")) - if downstream.has_children: - raise BadDownstream(_("Updating content with children is not yet supported.")) - try: - upstream_key = LibraryUsageLocatorV2.from_string(downstream.upstream) - except InvalidKeyError as exc: - raise BadUpstream(_("Reference to linked library item is malformed")) from exc + + # Ensure that the upstream block is of a compatible type. downstream_type = downstream.usage_key.block_type if upstream_key.block_type != downstream_type: # Note: Currently, we strictly enforce that the downstream and upstream block_types must exactly match. @@ -178,8 +209,8 @@ def get_for_block(cls, downstream: XBlock) -> t.Self: except XBlockNotFoundError as exc: raise BadUpstream(_("Linked library item was not found in the system")) from exc return cls( - upstream_ref=downstream.upstream, - version_synced=downstream.upstream_version, + upstream_ref=str(upstream_key), + version_synced=downstream.upstream_version if downstream.upstream else 0, version_available=(lib_meta.published_version_num if lib_meta else None), version_declined=downstream.upstream_version_declined, error_message=None, @@ -201,6 +232,13 @@ def sync_from_upstream(downstream: XBlock, user: User) -> None: _update_tags(upstream=upstream, downstream=downstream) downstream.upstream_version = link.version_available + # Explicitly set the `upstream` setting of the downstream block from the upstream's usage key. + # In most cases, this is a no-op, since that is normally how we'd spefically an upstream. + # However, it is also possible for a block to have implicitly-defined upstream-- particularly, if it is the child of + # a LegacyLibraryContentBlock, whose source library was recently migrated from a V1 library to a V2 library. + # In that case, we want to "migrate" the downstream to the new schema by explicitly setting its `upstream` setting. + downstream.upstream = str(upstream.usage_key) + def fetch_customizable_fields(*, downstream: XBlock, user: User, upstream: XBlock | None = None) -> None: """ @@ -213,6 +251,9 @@ def fetch_customizable_fields(*, downstream: XBlock, user: User, upstream: XBloc _link, upstream = _load_upstream_link_and_block(downstream, user) _update_customizable_fields(upstream=upstream, downstream=downstream, only_fetch=True) + # (see comment in sync_from_upstream) + downstream.upstream = str(upstream.usage_key) + def _load_upstream_link_and_block(downstream: XBlock, user: User) -> tuple[UpstreamLink, XBlock]: """ @@ -227,14 +268,16 @@ def _load_upstream_link_and_block(downstream: XBlock, user: User) -> tuple[Upstr # We import load_block here b/c UpstreamSyncMixin is used by cms/envs, which loads before the djangoapps are ready. from openedx.core.djangoapps.xblock.api import load_block, CheckPerm, LatestVersion # pylint: disable=wrong-import-order try: + # We know that upstream_ref cannot be None, since get_for_block returned successfully. + upstream_ref: str = link.upstream_ref # type: ignore[assignment] lib_block: XBlock = load_block( - LibraryUsageLocatorV2.from_string(downstream.upstream), + LibraryUsageLocatorV2.from_string(upstream_ref), user, check_permission=CheckPerm.CAN_READ_AS_AUTHOR, version=LatestVersion.PUBLISHED, ) except (NotFound, PermissionDenied) as exc: - raise BadUpstream(_("Linked library item could not be loaded: {}").format(downstream.upstream)) from exc + raise BadUpstream(_("Linked library item could not be loaded: {}").format(link.upstream_ref)) from exc return link, lib_block diff --git a/common/djangoapps/split_modulestore_django/admin.py b/common/djangoapps/split_modulestore_django/admin.py index 4dbddce9fa08..a2c1e3a2cfcf 100644 --- a/common/djangoapps/split_modulestore_django/admin.py +++ b/common/djangoapps/split_modulestore_django/admin.py @@ -1,12 +1,90 @@ """ Admin registration for Split Modulestore Django Backend """ -from django.contrib import admin +import logging + +from django import forms +from django.contrib import admin, messages +from django.contrib.admin.helpers import ActionForm +from opaque_keys import InvalidKeyError +from opaque_keys.edx.locator import LibraryLocator as LegacyLibraryLocator, LibraryLocatorV2, LibraryCollectionLocator from simple_history.admin import SimpleHistoryAdmin +from openedx.core.djangoapps.content_libraries.migration_api import migrate_legacy_library + from .models import SplitModulestoreCourseIndex +logger = logging.getLogger(__name__) + + +@admin.action(description="Migrate Legacy Library to new Library or Collection") +def migrate(modeladmin, request, queryset): + """ + Migrate legacy modulestore index entries to Learning Core, based on `migration_target_key`. + + Currently, this only works for LEGACY LIBRARY (library-v1:...) index entries. + Will fail if used on any other course entry. + + The only valid targets are currently V2 Libraries and their Collections. + Will fail on any other type of target key. + + WARNING: This does not delete the remaining legacy index item! It's up to Studio to recognize that an item has been + migrated, and that the legacy entry should be ignored. + """ + target_key_string = request.POST['migration_target_key'] + target_library_key: LibraryLocatorV2 + target_collection_slug: str | None + try: + target_library_key = LibraryLocatorV2.from_string(target_key_string) + target_collection_slug = None + except InvalidKeyError: + try: + target_collection_key = LibraryCollectionLocator.from_string(target_key_string) + target_library_key = target_collection_key.library_key + target_collection_slug = target_collection_key.collection_id + except InvalidKeyError: + modeladmin.message_user( + request, + f"Migration target key is not a valid V2 Library or Collection key: {target_key_string}", + level=messages.ERROR, + ) + return + for obj in queryset: + if not isinstance(obj.course_id, LegacyLibraryLocator): + modeladmin.message_user( + request, + f"Selected entry is not a Legacy Library: {obj.course_id}. Skipping.", + level=messages.WARNING, + ) + continue + try: + migrate_legacy_library( + source_key=obj.course_id, + target_key=target_library_key, + collection_slug=target_collection_slug, + user=request.user, + ) + except Exception as exc: # pylint: disable=broad-except + modeladmin.message_user( + request, + f"Failed to migrate {obj.course_id} to {target_key_string}: {exc}. See logs for details.", + level=messages.ERROR, + ) + logger.exception(exc) + continue + else: + modeladmin.message_user( + request, + f"Migrated {obj.course_id} to {target_key_string}", + level=messages.SUCCESS, + ) + + +class MigrationTargetForm(ActionForm): + migration_target_key = forms.CharField() + + @admin.register(SplitModulestoreCourseIndex) class SplitModulestoreCourseIndexAdmin(SimpleHistoryAdmin): """ @@ -15,4 +93,5 @@ class SplitModulestoreCourseIndexAdmin(SimpleHistoryAdmin): list_display = ('course_id', 'draft_version', 'published_version', 'library_version', 'wiki_slug', 'last_update') search_fields = ('course_id', 'wiki_slug') ordering = ('course_id', ) - readonly_fields = ('id', 'objectid', 'course_id', 'org', ) + actions = [migrate] + action_form = MigrationTargetForm diff --git a/openedx/core/djangoapps/content_libraries/admin.py b/openedx/core/djangoapps/content_libraries/admin.py index f84cac7f62e2..0eb3912a3751 100644 --- a/openedx/core/djangoapps/content_libraries/admin.py +++ b/openedx/core/djangoapps/content_libraries/admin.py @@ -2,7 +2,9 @@ Admin site for content libraries """ from django.contrib import admin -from .models import ContentLibrary, ContentLibraryPermission +from .models import ( + ContentLibrary, ContentLibraryPermission, ContentLibraryMigration, ContentLibraryBlockMigration +) class ContentLibraryPermissionInline(admin.TabularInline): @@ -39,3 +41,20 @@ def get_readonly_fields(self, request, obj=None): return ["library_key", "org", "slug"] else: return ["library_key", ] + + +class ContentLibraryBlockMigrationInline(admin.TabularInline): + """ + Django admin UI for content library block migrations + """ + model = ContentLibraryBlockMigration + list_display = ("library_migration", "block_type", "source_block_id", "target_block_id") + + +@admin.register(ContentLibraryMigration) +class ContentLibraryMigrationAdmin(admin.ModelAdmin): + """ + Django admin UI for content library migrations + """ + list_display = ("source_key", "target", "target_collection") + inlines = (ContentLibraryBlockMigrationInline,) diff --git a/openedx/core/djangoapps/content_libraries/management/commands/migrate_legacy_library.py b/openedx/core/djangoapps/content_libraries/management/commands/migrate_legacy_library.py new file mode 100644 index 000000000000..8d6e599c193e --- /dev/null +++ b/openedx/core/djangoapps/content_libraries/management/commands/migrate_legacy_library.py @@ -0,0 +1,49 @@ +""" +Implements ./manage.py cms migrate_legacy_library +""" +import logging + +from django.contrib.auth.models import User # pylint: disable=imported-auth-user +from django.core.management import BaseCommand + +from opaque_keys.edx.locator import LibraryLocator, LibraryLocatorV2 +from openedx.core.djangoapps.content_libraries.migration_api import migrate_legacy_library + + +log = logging.getLogger(__name__) + + +class Command(BaseCommand): + """ + @TODO + """ + + def add_arguments(self, parser): + """ + Add arguments to the argument parser. + """ + parser.add_argument( + 'legacy_library', + type=LibraryLocator.from_string, + ) + parser.add_argument( + 'new_library', + type=LibraryLocatorV2.from_string, + ) + parser.add_argument( + 'collection', + type=str, + ) + + def handle( # pylint: disable=arguments-differ + self, + legacy_library: LibraryLocator, + new_library: LibraryLocatorV2, + collection: str | None, + **kwargs, + ) -> None: + """ + Handle the command. + """ + user = User.objects.filter(is_superuser=True)[0] + migrate_legacy_library(legacy_library, new_library, collection_slug=collection, user=user) diff --git a/openedx/core/djangoapps/content_libraries/migration_api.py b/openedx/core/djangoapps/content_libraries/migration_api.py new file mode 100644 index 000000000000..05dfd61d5c13 --- /dev/null +++ b/openedx/core/djangoapps/content_libraries/migration_api.py @@ -0,0 +1,136 @@ +""" +@@TODO +""" +from __future__ import annotations + +import logging +from django.contrib.auth.models import User # pylint: disable=imported-auth-user +from django.db import transaction +from opaque_keys.edx.locator import LibraryLocator, LibraryLocatorV2 +from openedx_learning.api.authoring import add_to_collection, get_collection +from openedx_learning.api.authoring_models import PublishableEntity, Component +from openedx_tagging.core.tagging.api import tag_object +from openedx_tagging.core.tagging.models import Taxonomy +from organizations.models import Organization +from xblock.fields import Scope + +from openedx.core.djangoapps.xblock.api import load_block +from openedx.core.djangoapps.content_libraries.api import create_library_block +from xmodule.util.keys import BlockKey +from xmodule.modulestore.django import modulestore + +from .models import ContentLibrary, ContentLibraryMigration, ContentLibraryBlockMigration + + +log = logging.getLogger(__name__) + + +def migrate_legacy_library( + source_key: LibraryLocator, + target_key: LibraryLocatorV2, + *, + collection_slug: str | None, + user: User, + tags_to_add: dict[Taxonomy, list[str]] | None = None, +) -> None: + """ + Migrate a v1 (legacy) library into a v2 (learning core) library, optionally within a collection. + + Use a single transaction so that if any step fails, nothing happens. + + @@TODO handle or document various exceptions + """ + source = modulestore().get_library(source_key) + target = ContentLibrary.objects.get(org=Organization.objects.get(short_name=target_key.org), slug=target_key.slug) + assert target.learning_package_id + collection = get_collection(target.learning_package_id, collection_slug) if collection_slug else None + + # We need to be careful not to conflict with any existing block keys in the target library. + # This is unlikely to happen, since legacy library block ids are genreally randomly-generated GUIDs. + # Howevever, there are a couple scenarios where it could arise: + # * An instance has two legacy libraries which were imported from the same source legacy library (and thus share + # block GUIDs) which the author now wants to merge together into one big new library. + # * A library was imported from handcrafted OLX, and thus has human-readable block IDs which are liable to overlap. + # When there is conflict, we'll append "-1" to the end of the id (or "-2", "-3", etc., until we find a free ID). + all_target_block_keys: set[BlockKey] = { + BlockKey(*block_type_and_id) + for block_type_and_id + in Component.objects.filter( + learning_package=target.learning_package, + component_type__namespace="xblock.v1", + ).values_list("component_type__name", "local_key") + } + + # We also need to be careful not to conflict with other block IDs which we are moving in from the *source* library + # This is very unlikely, but it could happen if, for example: + # * the source library has a problem "foo", and + # * the target library also has a problem "foo", and + # * the source library ALSO has a problem "foo-1", thus + # * the source library's "foo" must be moved to the target as "foo-2". + all_source_block_keys: set[BlockKey] = { + BlockKey.from_usage_key(child_key) + for child_key in source.children + } + + target_block_entity_keys: set[str] = set() + + with transaction.atomic(): + migration = ContentLibraryMigration.objects.create( + source_key=source_key, + target=target, + target_collection=collection, + ) + + for source_block in source.get_children(): + block_type: str = source_block.usage_key.block_type + + # Determine an available block_id... + target_block_key = BlockKey(block_type, source_block.usage_key.block_id) + if target_block_key in all_target_block_keys: + suffix = 0 + while target_block_key in all_target_block_keys | all_source_block_keys: + suffix += 1 + target_block_key = BlockKey(block_type, f"{source_block.usage_key.block_id}-{suffix}") + + # Create the block in the v2 library + target_block_meta = create_library_block( + library_key=target_key, + block_type=block_type, + definition_id=target_block_key.id, + user_id=user.id, + ) + target_block_entity_keys.add(f"xblock.v1:{block_type}:{target_block_key.id}") + + # Copy its content over from the v1 library + target_block = load_block(target_block_meta.usage_key, user) + for field_name, field in source_block.__class__.fields.items(): + if field.scope not in [Scope.settings, Scope.content]: + continue + if not hasattr(target_block, field_name): + continue + source_value = getattr(source_block, field_name) + if getattr(target_block, field_name) != source_value: + setattr(target_block, field_name, source_value) + target_block.save() + + # If requested, add tags + for taxonomy, taxonomy_tags in (tags_to_add or {}).items(): + tag_object(str(target_block_meta.usage_key), taxonomy, taxonomy_tags) + + # Make a record of the migration + ContentLibraryBlockMigration.objects.create( + library_migration=migration, + block_type=block_type, + source_block_id=source_block.usage_key.block_id, + target_block_id=target_block_key.id, + ) + + # If requested, add to a collection, and add tags + if collection_slug: + add_to_collection( + target.learning_package_id, + collection_slug, + PublishableEntity.objects.filter( + key__in=target_block_entity_keys, + ), + ) diff --git a/openedx/core/djangoapps/content_libraries/migrations/0012_contentlibrarymigration_contentlibraryblockmigration.py b/openedx/core/djangoapps/content_libraries/migrations/0012_contentlibrarymigration_contentlibraryblockmigration.py new file mode 100644 index 000000000000..0dc5efb747a6 --- /dev/null +++ b/openedx/core/djangoapps/content_libraries/migrations/0012_contentlibrarymigration_contentlibraryblockmigration.py @@ -0,0 +1,38 @@ +# Generated by Django 4.2.16 on 2024-11-01 19:07 + +from django.db import migrations, models +import django.db.models.deletion +import opaque_keys.edx.django.models + + +class Migration(migrations.Migration): + + dependencies = [ + ('oel_collections', '0005_alter_collection_options_alter_collection_enabled'), + ('content_libraries', '0011_remove_contentlibrary_bundle_uuid_and_more'), + ] + + operations = [ + migrations.CreateModel( + name='ContentLibraryMigration', + fields=[ + ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), + ('source_key', opaque_keys.edx.django.models.LearningContextKeyField(max_length=255, unique=True)), + ('target', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to='content_libraries.contentlibrary')), + ('target_collection', models.ForeignKey(null=True, on_delete=django.db.models.deletion.SET_NULL, to='oel_collections.collection')), + ], + ), + migrations.CreateModel( + name='ContentLibraryBlockMigration', + fields=[ + ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), + ('block_type', models.SlugField()), + ('source_block_id', models.SlugField()), + ('target_block_id', models.SlugField()), + ('library_migration', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='block_migrations', to='content_libraries.contentlibrarymigration')), + ], + options={ + 'unique_together': {('library_migration', 'block_type', 'source_block_id')}, + }, + ), + ] diff --git a/openedx/core/djangoapps/content_libraries/models.py b/openedx/core/djangoapps/content_libraries/models.py index 61e28b944851..34afe78909d8 100644 --- a/openedx/core/djangoapps/content_libraries/models.py +++ b/openedx/core/djangoapps/content_libraries/models.py @@ -46,17 +46,19 @@ from django.utils.translation import gettext_lazy as _ from opaque_keys.edx.django.models import CourseKeyField -from opaque_keys.edx.locator import LibraryLocatorV2 +from opaque_keys.edx.locator import ( + BlockUsageLocator, LibraryUsageLocatorV2, LibraryLocatorV2, LibraryCollectionLocator +) from pylti1p3.contrib.django import DjangoDbToolConf from pylti1p3.contrib.django import DjangoMessageLaunch from pylti1p3.contrib.django.lti1p3_tool_config.models import LtiTool from pylti1p3.grade import Grade -from opaque_keys.edx.django.models import UsageKeyField from openedx.core.djangoapps.content_libraries.constants import ( LICENSE_OPTIONS, ALL_RIGHTS_RESERVED, ) -from openedx_learning.api.authoring_models import LearningPackage +from opaque_keys.edx.django.models import LearningContextKeyField, UsageKeyField +from openedx_learning.api.authoring_models import LearningPackage, Collection from organizations.models import Organization # lint-amnesty, pylint: disable=wrong-import-order from .apps import ContentLibrariesConfig @@ -227,6 +229,60 @@ def __str__(self): return f"ContentLibraryPermission ({self.access_level} for {who})" +class ContentLibraryMigration(models.Model): + """ + Record of a legacy (v1) content library that has been migrated into a new (v2) content library. + """ + source_key = LearningContextKeyField(unique=True, max_length=255) + target = models.ForeignKey(ContentLibrary, on_delete=models.CASCADE) + target_collection = models.ForeignKey(Collection, on_delete=models.SET_NULL, null=True) + + @property + def target_key(self) -> LibraryLocatorV2: + return self.target.library_key + + @property + def target_library_collection_key(self) -> LibraryCollectionLocator | None: + return ( + LibraryCollectionLocator(self.target_key, self.target_collection.key) + if self.target_collection + else None + ) + + def __str__(self) -> str: + return f"{self.source_key} -> {self.target_library_collection_key or self.target_key}" + + +class ContentLibraryBlockMigration(models.Model): + """ + Record of a legacy (v1) content library block that has been migrated into a new (v) content library block. + """ + library_migration = models.ForeignKey( + ContentLibraryMigration, on_delete=models.CASCADE, related_name="block_migrations" + ) + block_type = models.SlugField() + source_block_id = models.SlugField() + target_block_id = models.SlugField() + + @property + def source_usage_key(self) -> BlockUsageLocator: + return self.library_migration.source_key.make_usage_key(self.block_type, self.source_block_id) + + @property + def target_usage_key(self) -> LibraryUsageLocatorV2: + return LibraryUsageLocatorV2( # type: ignore[abstract] # (we are missing an annotation in opaque-keys) + lib_key=self.library_migration.target_key, + usage_id=self.target_block_id, + block_type=self.block_type, + ) + + def __str__(self): + return f"{self.source_usage_key} -> {self.target_usage_key}" + + class Meta: + unique_together = [('library_migration', 'block_type', 'source_block_id')] + + class ContentLibraryBlockImportTask(models.Model): """ Model of a task to import blocks from an external source (e.g. modulestore). diff --git a/xmodule/library_content_block.py b/xmodule/library_content_block.py index 52e33108027c..ae6af327af41 100644 --- a/xmodule/library_content_block.py +++ b/xmodule/library_content_block.py @@ -16,7 +16,7 @@ import nh3 from django.core.exceptions import ObjectDoesNotExist, PermissionDenied -from opaque_keys.edx.locator import LibraryLocator +from opaque_keys.edx.locator import LibraryLocator, LibraryUsageLocatorV2 from web_fragments.fragment import Fragment from webob import Response from xblock.core import XBlock @@ -104,6 +104,91 @@ class LegacyLibraryContentBlock(ItemBankMixin, XModuleToXBlockMixin, XBlock): scope=Scope.settings, ) + def get_migrated_upstream_for_child(self, child_block_id: str) -> LibraryUsageLocatorV2: + """ + Return the key of the V2 library block to which the indicated child's source block was migrated. + + May raise UpstreamLink for a variety of reasons (@@TODO -- describe here). + """ + # @@TODO Move these imports up to the top. + # @@TODO Factor the ContentLibraryMigration fetching into a nice Python API with data objects. + from openedx.core.djangoapps.content_libraries.models import ContentLibraryMigration + from cms.lib.xblock.upstream_sync import NoUpstream, BadUpstream + from xmodule.util.keys import BlockKey, derive_key + from opaque_keys import InvalidKeyError + + if not self.source_library_id: + raise NoUpstream() + try: + source_library_key = self.source_library_key + except InvalidKeyError as exc: + # Source v1 library key is malformed. + # The legacy library content block UI will inform the user of this, so we don't need to. + # @@TODO Should we actually move error messaging here, so that we can remove that bit of the legacy UI? + raise NoUpstream() from exc + try: + library_migration = ContentLibraryMigration.objects.get(source_key=source_library_key) + except ContentLibraryMigration.DoesNotExist as exc: + # Source v1 library has not (yet?) been migrated to a v2 library. + # Alternatively, we are on an instance that doesn't have the source v1 library-- in that case, the legacy + # library content block UI will tell the user such. + # @@TODO Should we actually move error messaging here, so that we can remove that bit of the legacy UI? + raise NoUpstream() from exc + + # In order identify the new v2 library block, we need to know the v1 library block that this child came from. + # Unfortunately, there's no straightforward mapping from these children back to their v1 library source blocks. + # (ModuleStore does have a get_original_usage function that inspects edit_info, but we can't count on + # that always working, particularly if this block's course was imported from another instance.) + # However, we can work around this by just looping through every block in the legacy library, and testing to see + # if it's our source block. + + logger.info( + "Within context '%s'...\n" + " we are searching for the new upstream of block '%s'\n" + " by looking at its parent legacy library_content block at '%s',\n" + " which points at source library %s,\n" + " and whose children are: %s.", + self.usage_key.context_key, child_block_id, self.usage_key.block_id, source_library_key, + ' '.join(child_key.block_id for child_key in self.children), + ) + + # So, for each block in the migrated legacy library... + for block_migration in library_migration.block_migrations.all(): + + # IF we were to have used the legacy library block as a legacy library_content block child, + # then what would its block_id be? + derived_child_block_id = derive_key( + source=block_migration.source_usage_key.for_branch("library"), + dest_parent=BlockKey.from_usage_key(self.usage_key), + ).id + logger.info( + "Within legacy library '%s'...\n" + " there is a block at '%s',\n" + " whose child would be '%s'.", + library_migration.source_key, + block_migration.source_block_id, + derived_child_block_id, + ) + + # If that derived block_id matches the child_block_id we're after, then we've found our legacy library + # source block! So, just return the usage key of the v2 library block that it's been migrated to. + if child_block_id == derived_child_block_id: + logger.info( + "Within context '%s'....\n" + " we have MATCHED block '%s'" + " with upstream '%s'", + self.usage_key.context_key, + child_block_id, + block_migration.target_usage_key, + ) + return block_migration.target_usage_key + + # The v1 library was migrated to a v2 library, but this particular child was never migrated to said v2 library. + # This can happen if a legacy LC block child was derived from a v1 library block which was later deleted from + # said v1 library, and the legacy LC block never synced the update which deleted said child. + logger.info("Did not find matching upstream for %s", child_block_id) + raise NoUpstream() + @property def source_library_key(self): """