Make storage providers more configurable
							parent
							
								
									3cb2dabaad
								
							
						
					
					
						commit
						0af5dc63a8
					
				|  | @ -16,6 +16,8 @@ | |||
| from ._base import Config, ConfigError | ||||
| from collections import namedtuple | ||||
| 
 | ||||
| from synapse.util.module_loader import load_module | ||||
| 
 | ||||
| 
 | ||||
| MISSING_NETADDR = ( | ||||
|     "Missing netaddr library. This is required for URL preview API." | ||||
|  | @ -36,6 +38,10 @@ ThumbnailRequirement = namedtuple( | |||
|     "ThumbnailRequirement", ["width", "height", "method", "media_type"] | ||||
| ) | ||||
| 
 | ||||
| MediaStorageProviderConfig = namedtuple( | ||||
|     "MediaStorageProviderConfig", ("store_local", "store_remote", "store_synchronous",) | ||||
| ) | ||||
| 
 | ||||
| 
 | ||||
| def parse_thumbnail_requirements(thumbnail_sizes): | ||||
|     """ Takes a list of dictionaries with "width", "height", and "method" keys | ||||
|  | @ -73,16 +79,65 @@ class ContentRepositoryConfig(Config): | |||
| 
 | ||||
|         self.media_store_path = self.ensure_directory(config["media_store_path"]) | ||||
| 
 | ||||
|         self.backup_media_store_path = config.get("backup_media_store_path") | ||||
|         if self.backup_media_store_path: | ||||
|             self.backup_media_store_path = self.ensure_directory( | ||||
|         backup_media_store_path = config.get("backup_media_store_path") | ||||
|         if backup_media_store_path: | ||||
|             backup_media_store_path = self.ensure_directory( | ||||
|                 self.backup_media_store_path | ||||
|             ) | ||||
| 
 | ||||
|         self.synchronous_backup_media_store = config.get( | ||||
|         synchronous_backup_media_store = config.get( | ||||
|             "synchronous_backup_media_store", False | ||||
|         ) | ||||
| 
 | ||||
|         storage_providers = config.get("media_storage_providers", []) | ||||
| 
 | ||||
|         if backup_media_store_path: | ||||
|             if storage_providers: | ||||
|                 raise ConfigError( | ||||
|                     "Cannot use both 'backup_media_store_path' and 'storage_providers'" | ||||
|                 ) | ||||
| 
 | ||||
|             storage_providers = [{ | ||||
|                 "module": "file_system", | ||||
|                 "store_local": True, | ||||
|                 "store_synchronous": synchronous_backup_media_store, | ||||
|                 "store_remote": True, | ||||
|                 "config": { | ||||
|                     "directory": backup_media_store_path, | ||||
|                 } | ||||
|             }] | ||||
| 
 | ||||
|         # This is a list of config that can be used to create the storage | ||||
|         # providers. The entries are tuples of (Class, class_config, | ||||
|         # MediaStorageProviderConfig), where Class is the class of the provider, | ||||
|         # the class_config the config to pass to it, and | ||||
|         # MediaStorageProviderConfig are options for StorageProviderWrapper. | ||||
|         # | ||||
|         # We don't create the storage providers here as not all workers need | ||||
|         # them to be started. | ||||
|         self.media_storage_providers = [] | ||||
| 
 | ||||
|         for provider_config in storage_providers: | ||||
|             # We special case the module "file_system" so as not to need to | ||||
|             # expose FileStorageProviderBackend | ||||
|             if provider_config["module"] == "file_system": | ||||
|                 provider_config["module"] = ( | ||||
|                     "synapse.rest.media.v1.storage_provider" | ||||
|                     ".FileStorageProviderBackend" | ||||
|                 ) | ||||
| 
 | ||||
|             provider_class, parsed_config = load_module(provider_config) | ||||
| 
 | ||||
|             wrapper_config = MediaStorageProviderConfig( | ||||
|                 provider_config.get("store_local", False), | ||||
|                 provider_config.get("store_remote", False), | ||||
|                 provider_config.get("store_synchronous", False), | ||||
|             ) | ||||
| 
 | ||||
|             self.media_storage_providers.append( | ||||
|                 (provider_class, provider_config, wrapper_config,) | ||||
|             ) | ||||
| 
 | ||||
|         self.uploads_path = self.ensure_directory(config["uploads_path"]) | ||||
|         self.dynamic_thumbnails = config["dynamic_thumbnails"] | ||||
|         self.thumbnail_requirements = parse_thumbnail_requirements( | ||||
|  | @ -127,13 +182,19 @@ class ContentRepositoryConfig(Config): | |||
|         # Directory where uploaded images and attachments are stored. | ||||
|         media_store_path: "%(media_store)s" | ||||
| 
 | ||||
|         # A secondary directory where uploaded images and attachments are | ||||
|         # stored as a backup. | ||||
|         # backup_media_store_path: "%(media_store)s" | ||||
| 
 | ||||
|         # Whether to wait for successful write to backup media store before | ||||
|         # returning successfully. | ||||
|         # synchronous_backup_media_store: false | ||||
|         # Media storage providers allow media to be stored in different | ||||
|         # locations. | ||||
|         # media_storage_providers: | ||||
|         # - module: file_system | ||||
|         #   # Whether to write new local files. | ||||
|         #   store_local: false | ||||
|         #   # Whether to write new remote media | ||||
|         #   store_remote: false | ||||
|         #   # Whether to block upload requests waiting for write to this | ||||
|         #   # provider to complete | ||||
|         #   store_synchronous: false | ||||
|         #   config: | ||||
|         #     directory: /mnt/some/other/directory | ||||
| 
 | ||||
|         # Directory where in-progress uploads are stored. | ||||
|         uploads_path: "%(uploads_path)s" | ||||
|  |  | |||
|  | @ -27,9 +27,7 @@ from .identicon_resource import IdenticonResource | |||
| from .preview_url_resource import PreviewUrlResource | ||||
| from .filepath import MediaFilePaths | ||||
| from .thumbnailer import Thumbnailer | ||||
| from .storage_provider import ( | ||||
|     StorageProviderWrapper, FileStorageProviderBackend, | ||||
| ) | ||||
| from .storage_provider import StorageProviderWrapper | ||||
| from .media_storage import MediaStorage | ||||
| 
 | ||||
| from synapse.http.matrixfederationclient import MatrixFederationHttpClient | ||||
|  | @ -80,17 +78,13 @@ class MediaRepository(object): | |||
|         # potentially upload to. | ||||
|         storage_providers = [] | ||||
| 
 | ||||
|         # TODO: Move this into config and allow other storage providers to be | ||||
|         # defined. | ||||
|         if hs.config.backup_media_store_path: | ||||
|             backend = FileStorageProviderBackend( | ||||
|                 self.primary_base_path, hs.config.backup_media_store_path, | ||||
|             ) | ||||
|         for clz, provider_config, wrapper_config in hs.config.media_storage_providers: | ||||
|             backend = clz(hs, provider_config) | ||||
|             provider = StorageProviderWrapper( | ||||
|                 backend, | ||||
|                 store=True, | ||||
|                 store_synchronous=hs.config.synchronous_backup_media_store, | ||||
|                 store_remote=True, | ||||
|                 store_local=wrapper_config.store_local, | ||||
|                 store_remote=wrapper_config.store_remote, | ||||
|                 store_synchronous=wrapper_config.store_synchronous, | ||||
|             ) | ||||
|             storage_providers.append(provider) | ||||
| 
 | ||||
|  |  | |||
|  | @ -17,6 +17,7 @@ from twisted.internet import defer, threads | |||
| 
 | ||||
| from .media_storage import FileResponder | ||||
| 
 | ||||
| from synapse.config._base import Config | ||||
| from synapse.util.logcontext import preserve_fn | ||||
| 
 | ||||
| import logging | ||||
|  | @ -64,14 +65,14 @@ class StorageProviderWrapper(StorageProvider): | |||
| 
 | ||||
|     Args: | ||||
|         backend (StorageProvider) | ||||
|         store (bool): Whether to store new files or not. | ||||
|         store_local (bool): Whether to store new local files or not. | ||||
|         store_synchronous (bool): Whether to wait for file to be successfully | ||||
|             uploaded, or todo the upload in the backgroud. | ||||
|         store_remote (bool): Whether remote media should be uploaded | ||||
|     """ | ||||
|     def __init__(self, backend, store, store_synchronous, store_remote): | ||||
|     def __init__(self, backend, store_local, store_synchronous, store_remote): | ||||
|         self.backend = backend | ||||
|         self.store = store | ||||
|         self.store_local = store_local | ||||
|         self.store_synchronous = store_synchronous | ||||
|         self.store_remote = store_remote | ||||
| 
 | ||||
|  | @ -97,13 +98,13 @@ class FileStorageProviderBackend(StorageProvider): | |||
|     """A storage provider that stores files in a directory on a filesystem. | ||||
| 
 | ||||
|     Args: | ||||
|         cache_directory (str): Base path of the local media repository | ||||
|         base_directory (str): Base path to store new files | ||||
|         hs (HomeServer) | ||||
|         config: The config returned by `parse_config`, i | ||||
|     """ | ||||
| 
 | ||||
|     def __init__(self, cache_directory, base_directory): | ||||
|         self.cache_directory = cache_directory | ||||
|         self.base_directory = base_directory | ||||
|     def __init__(self, hs, config): | ||||
|         self.cache_directory = hs.config.media_store_path | ||||
|         self.base_directory = config | ||||
| 
 | ||||
|     def store_file(self, path, file_info): | ||||
|         """See StorageProvider.store_file""" | ||||
|  | @ -125,3 +126,14 @@ class FileStorageProviderBackend(StorageProvider): | |||
|         backup_fname = os.path.join(self.base_directory, path) | ||||
|         if os.path.isfile(backup_fname): | ||||
|             return FileResponder(open(backup_fname, "rb")) | ||||
| 
 | ||||
|     def parse_config(config): | ||||
|         """Called on startup to parse config supplied. This should parse | ||||
|         the config and raise if there is a problem. | ||||
| 
 | ||||
|         The returned value is passed into the constructor. | ||||
| 
 | ||||
|         In this case we only care about a single param, the directory, so lets | ||||
|         just pull that out. | ||||
|         """ | ||||
|         return Config.ensure_directory(config["directory"]) | ||||
|  |  | |||
		Loading…
	
		Reference in New Issue
	
	 Erik Johnston
						Erik Johnston