 53435579b3
			
		
	
	
		53435579b3
		
			
		
	
	
	
	
		
			
			* fbt, faploader: minimal app module implementation * faploader, libs: moved API hashtable core to flipper_application * example: compound api * lib: flipper_application: naming fixes, doxygen comments * fbt: changed `requires` manifest field behavior for app extensions * examples: refactored plugin apps; faploader: changed new API naming; fbt: changed PLUGIN app type meaning * loader: dropped support for debug apps & plugin menus * moved applications/plugins -> applications/external * Restored x bit on chiplist_convert.py * git: fixed free-dap submodule path * pvs: updated submodule paths * examples: example_advanced_plugins.c: removed potential memory leak on errors * examples: example_plugins: refined requires * fbt: not deploying app modules for debug/sample apps; extra validation for .PLUGIN-type apps * apps: removed cdefines for external apps * fbt: moved ext app path definition * fbt: reworked fap_dist handling; f18: synced api_symbols.csv * fbt: removed resources_paths for extapps * scripts: reworked storage * scripts: reworked runfap.py & selfupdate.py to use new api * wip: fal runner * fbt: moved file packaging into separate module * scripts: storage: fixes * scripts: storage: minor fixes for new api * fbt: changed internal artifact storage details for external apps * scripts: storage: additional fixes and better error reporting; examples: using APP_DATA_PATH() * fbt, scripts: reworked launch_app to deploy plugins; moved old runfap.py to distfap.py * fbt: extra check for plugins descriptors * fbt: additional checks in emitter * fbt: better info message on SDK rebuild * scripts: removed requirements.txt * loader: removed remnants of plugins & debug menus * post-review fixes
		
			
				
	
	
		
			88 lines
		
	
	
		
			2.6 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			88 lines
		
	
	
		
			2.6 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| #!/usr/bin/env python3
 | |
| 
 | |
| from flipper.app import App
 | |
| from flipper.storage import FlipperStorage, FlipperStorageOperations
 | |
| from flipper.utils.cdc import resolve_port
 | |
| 
 | |
| import os
 | |
| import posixpath
 | |
| from functools import reduce
 | |
| import operator
 | |
| 
 | |
| 
 | |
| class Main(App):
 | |
|     def init(self):
 | |
|         self.parser.add_argument("-p", "--port", help="CDC Port", default="auto")
 | |
|         self.parser.add_argument(
 | |
|             "--sources",
 | |
|             "-s",
 | |
|             nargs="+",
 | |
|             action="append",
 | |
|             default=[],
 | |
|             help="Files to send",
 | |
|         )
 | |
|         self.parser.add_argument(
 | |
|             "--targets",
 | |
|             "-t",
 | |
|             nargs="+",
 | |
|             action="append",
 | |
|             default=[],
 | |
|             help="File destinations (must be same length as -s)",
 | |
|         )
 | |
|         self.parser.add_argument(
 | |
|             "--host-app",
 | |
|             "-a",
 | |
|             help="Host app to launch",
 | |
|         )
 | |
| 
 | |
|         self.parser.set_defaults(func=self.install)
 | |
| 
 | |
|     @staticmethod
 | |
|     def flatten(l):
 | |
|         return reduce(operator.concat, l, [])
 | |
| 
 | |
|     def install(self):
 | |
|         self.args.sources = self.flatten(self.args.sources)
 | |
|         self.args.targets = self.flatten(self.args.targets)
 | |
| 
 | |
|         if len(self.args.sources) != len(self.args.targets):
 | |
|             self.logger.error(
 | |
|                 f"Error: sources ({self.args.sources}) and targets ({self.args.targets}) must be same length"
 | |
|             )
 | |
|             return 1
 | |
| 
 | |
|         if not (port := resolve_port(self.logger, self.args.port)):
 | |
|             return 2
 | |
| 
 | |
|         try:
 | |
|             with FlipperStorage(port) as storage:
 | |
|                 storage_ops = FlipperStorageOperations(storage)
 | |
|                 for fap_local_path, fap_dst_path in zip(
 | |
|                     self.args.sources, self.args.targets
 | |
|                 ):
 | |
|                     self.logger.info(f'Installing "{fap_local_path}" to {fap_dst_path}')
 | |
| 
 | |
|                     storage_ops.recursive_send(fap_dst_path, fap_local_path, False)
 | |
| 
 | |
|                 fap_host_app = self.args.targets[0]
 | |
|                 startup_command = f'"Applications" {fap_host_app}'
 | |
|                 if self.args.host_app:
 | |
|                     startup_command = self.args.host_app
 | |
| 
 | |
|                 self.logger.info(f"Launching app: {startup_command}")
 | |
|                 storage.send_and_wait_eol(f"loader open {startup_command}\r")
 | |
| 
 | |
|                 if len(result := storage.read.until(storage.CLI_EOL)):
 | |
|                     self.logger.error(f"Unexpected response: {result.decode('ascii')}")
 | |
|                     return 3
 | |
|                 return 0
 | |
| 
 | |
|         except Exception as e:
 | |
|             self.logger.error(f"Error: {e}")
 | |
|             # raise
 | |
|             return 4
 | |
| 
 | |
| 
 | |
| if __name__ == "__main__":
 | |
|     Main()()
 |