* 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
		
			
				
	
	
		
			72 lines
		
	
	
		
			2.2 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			72 lines
		
	
	
		
			2.2 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
 | 
						|
 | 
						|
 | 
						|
class Main(App):
 | 
						|
    def init(self):
 | 
						|
        self.parser.add_argument("-p", "--port", help="CDC Port", default="auto")
 | 
						|
        self.parser.add_argument(
 | 
						|
            "-n",
 | 
						|
            "--no-launch",
 | 
						|
            dest="launch_app",
 | 
						|
            action="store_false",
 | 
						|
            help="Don't launch app",
 | 
						|
        )
 | 
						|
 | 
						|
        self.parser.add_argument("fap_src_path", help="App file to upload")
 | 
						|
        self.parser.add_argument(
 | 
						|
            "--fap_dst_dir", help="Upload path", default="/ext/apps", required=False
 | 
						|
        )
 | 
						|
        self.parser.set_defaults(func=self.install)
 | 
						|
 | 
						|
    def install(self):
 | 
						|
        if not (port := resolve_port(self.logger, self.args.port)):
 | 
						|
            return 1
 | 
						|
 | 
						|
        try:
 | 
						|
            with FlipperStorage(port) as storage:
 | 
						|
                storage_ops = FlipperStorageOperations(storage)
 | 
						|
                fap_local_path = self.args.fap_src_path
 | 
						|
                self.args.fap_dst_dir = self.args.fap_dst_dir.rstrip("/\\")
 | 
						|
 | 
						|
                if not os.path.isfile(fap_local_path):
 | 
						|
                    self.logger.error(
 | 
						|
                        f"Error: source .fap ({fap_local_path}) not found"
 | 
						|
                    )
 | 
						|
                    return 2
 | 
						|
 | 
						|
                fap_dst_path = posixpath.join(
 | 
						|
                    self.args.fap_dst_dir, os.path.basename(fap_local_path)
 | 
						|
                )
 | 
						|
 | 
						|
                self.logger.info(f'Installing "{fap_local_path}" to {fap_dst_path}')
 | 
						|
 | 
						|
                storage_ops.recursive_send(fap_dst_path, fap_local_path, False)
 | 
						|
 | 
						|
                if not self.args.launch_app:
 | 
						|
                    return 0
 | 
						|
 | 
						|
                storage.send_and_wait_eol(
 | 
						|
                    f'loader open "Applications" {fap_dst_path}\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()()
 |