* 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.4 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			72 lines
		
	
	
		
			2.4 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
#!/usr/bin/env python3
 | 
						|
 | 
						|
from typing import final
 | 
						|
from flipper.app import App
 | 
						|
from flipper.storage import FlipperStorage, FlipperStorageOperations
 | 
						|
from flipper.utils.cdc import resolve_port
 | 
						|
 | 
						|
import logging
 | 
						|
import os
 | 
						|
import pathlib
 | 
						|
import serial.tools.list_ports as list_ports
 | 
						|
 | 
						|
 | 
						|
class Main(App):
 | 
						|
    def init(self):
 | 
						|
        self.parser.add_argument("-p", "--port", help="CDC Port", default="auto")
 | 
						|
 | 
						|
        self.parser.add_argument("manifest_path", help="Manifest path")
 | 
						|
        self.parser.add_argument(
 | 
						|
            "--pkg_dir_name", help="Update dir name", default="pcbundle", required=False
 | 
						|
        )
 | 
						|
        self.parser.set_defaults(func=self.install)
 | 
						|
 | 
						|
        # logging
 | 
						|
        self.logger = logging.getLogger()
 | 
						|
 | 
						|
    def install(self):
 | 
						|
        if not (port := resolve_port(self.logger, self.args.port)):
 | 
						|
            return 1
 | 
						|
 | 
						|
        if not os.path.isfile(self.args.manifest_path):
 | 
						|
            self.logger.error("Error: manifest not found")
 | 
						|
            return 2
 | 
						|
 | 
						|
        manifest_path = pathlib.Path(os.path.abspath(self.args.manifest_path))
 | 
						|
        manifest_name, pkg_name = manifest_path.parts[-1], manifest_path.parts[-2]
 | 
						|
 | 
						|
        pkg_dir_name = self.args.pkg_dir_name or pkg_name
 | 
						|
        update_root = "/ext/update"
 | 
						|
        flipper_update_path = f"{update_root}/{pkg_dir_name}"
 | 
						|
 | 
						|
        self.logger.info(f'Installing "{pkg_name}" from {flipper_update_path}')
 | 
						|
 | 
						|
        try:
 | 
						|
            with FlipperStorage(port) as storage:
 | 
						|
                storage_ops = FlipperStorageOperations(storage)
 | 
						|
                storage_ops.mkpath(update_root)
 | 
						|
                storage_ops.mkpath(flipper_update_path)
 | 
						|
                storage_ops.recursive_send(
 | 
						|
                    flipper_update_path, manifest_path.parents[0]
 | 
						|
                )
 | 
						|
 | 
						|
                storage.send_and_wait_eol(
 | 
						|
                    f"update install {flipper_update_path}/{manifest_name}\r"
 | 
						|
                )
 | 
						|
                result = storage.read.until(storage.CLI_EOL)
 | 
						|
                if not b"Verifying" in result:
 | 
						|
                    self.logger.error(f"Unexpected response: {result.decode('ascii')}")
 | 
						|
                    return 3
 | 
						|
                result = storage.read.until(storage.CLI_EOL)
 | 
						|
                if not result.startswith(b"OK"):
 | 
						|
                    self.logger.error(result.decode("ascii"))
 | 
						|
                    return 4
 | 
						|
                return 0
 | 
						|
        except Exception as e:
 | 
						|
            self.logger.error(e)
 | 
						|
            return 5
 | 
						|
 | 
						|
 | 
						|
if __name__ == "__main__":
 | 
						|
    Main()()
 |