diff options
-rwxr-xr-x | git-gmi/cgi | 2 | ||||
-rw-r--r-- | git-gmi/const.py | 6 | ||||
-rw-r--r-- | git-gmi/gateway.py (renamed from git-gmi/cgi.py) | 20 | ||||
-rw-r--r-- | git-gmi/git.py | 55 |
4 files changed, 56 insertions, 27 deletions
diff --git a/git-gmi/cgi b/git-gmi/cgi index c0e0d25..7ef8259 100755 --- a/git-gmi/cgi +++ b/git-gmi/cgi @@ -1,3 +1,3 @@ #!/home/fakefred/p/git.gmi/venv/bin/python3.8 # gotta change the executable path before running -import cgi.py
\ No newline at end of file +import gateway
\ No newline at end of file diff --git a/git-gmi/const.py b/git-gmi/const.py index 9a1bddd..dcf97b1 100644 --- a/git-gmi/const.py +++ b/git-gmi/const.py @@ -1,5 +1,7 @@ -GIT_CATALOG = "/home/fakefred/p/gemini/git/" -STATUS_SUCCESS = "20 SUCCESS" +GIT_CATALOG = "/home/fakefred/p/gemini/repos/" +CGI_PATH = "/git/cgi/" +GIT_GMI_SITE_TITLE = "git.gmi demo instance" +STATUS_SUCCESS = "20" STATUS_NOT_FOUND = "51 NOT FOUND" STATUS_TEMPORARY_FAILURE = "40 TEMPORARY FAILURE" META_GEMINI = "text/gemini" diff --git a/git-gmi/cgi.py b/git-gmi/gateway.py index 76f1522..0ae963f 100644 --- a/git-gmi/cgi.py +++ b/git-gmi/gateway.py @@ -10,7 +10,7 @@ def generate_navigation(repo_name: str): pass # TODO -def handle_cgi_request(path: str): +def handle_cgi_request(path: str, query: str): # intended to work with Jetforce. # url: gemini://git.gemini.site/cgi-bin/cgi.py/repo/src/static/css/[index.css] # path: /repo/src/static/css/[index.css] @@ -18,8 +18,8 @@ def handle_cgi_request(path: str): path_trace = path[1:].split("/") if path_trace == [""]: # empty path print(f"{STATUS_SUCCESS} {META_GEMINI}") # welcome page - print("Welcome to the git.gmi demo") - print("Available repositories:") + print(f"# Welcome to {GIT_GMI_SITE_TITLE}") + print("## Available repositories:") print("\n".join([f"=> {dir}/" for dir in listdir(GIT_CATALOG)])) return @@ -51,18 +51,18 @@ def handle_cgi_request(path: str): if len(path_trace) > 2: branch = path_trace[2] - if len(path_trace) == 3: - location = [] - else: - location = path_trace[3:] + location = path_trace[3:] try: # is dir print(repo.view_tree(branch, location)) except FileNotFoundError: # is file try: - print(repo.view_blob(branch, location)) + if query == "raw": + print(repo.view_raw_blob(branch, location)) + else: + print(repo.view_blob(branch, location)) except FileNotFoundError: - print("50 Error locating content") + print(STATUS_NOT_FOUND) elif view == "log": try: @@ -73,4 +73,4 @@ def handle_cgi_request(path: str): return -handle_cgi_request(environ.get("PATH_INFO")) +handle_cgi_request(environ.get("PATH_INFO"), environ.get("QUERY_STRING")) diff --git a/git-gmi/git.py b/git-gmi/git.py index 9a0ab0d..990ca0c 100644 --- a/git-gmi/git.py +++ b/git-gmi/git.py @@ -2,6 +2,9 @@ from pygit2 import * import mimetypes from const import * +mimetypes.add_type("text/gemini", ".gmi") +mimetypes.add_type("text/gemini", ".gemini") + class GitGmiRepo: def __init__(self, name: str, path: str): @@ -9,25 +12,37 @@ class GitGmiRepo: self.path = path try: self.repo = Repository(path) - except FileNotFoundError: - print(f"Error: repository {path} not found") + except GitError: + raise FileNotFoundError(f"Error: no such repo: {name}") + + def generate_header(self): + header = ( + f"# {self.name}\n" + f"=> {CGI_PATH} {GIT_GMI_SITE_TITLE}\n" + f"=> {CGI_PATH}{self.name}/summary summary\n" + f"=> {CGI_PATH}{self.name}/tree/master/ tree\n" + f"=> {CGI_PATH}{self.name}/log log\n\n" + ) + return header def view_summary(self) -> str: + response = f"{STATUS_SUCCESS} {META_GEMINI}\n" + self.generate_header() tree = self.get_tree("master") trls = self.list_tree(tree) + found_readme = False for item in trls: - if item["type"] == "file" and item["name"].lower().split(".")[0] == ( - "readme" + if ( + item["type"] == "file" + and item["name"].lower().split(".")[0] == ("readme") + and not found_readme ): - # mimetypes.guess_type() returns tuple (type, encoding) - # only the first one of which we care about - response = ( - f"{STATUS_SUCCESS} {mimetypes.guess_type(item['name'])[0]}\n" - "=> tree/master/ tree\n" - "=> log/ log\n\n" + found_readme = True + response += ( + f"## {item['name']} | {item['size']} bytes\n" + f"{item['blob'].data.decode('utf-8')}" ) - response += item["blob"].data.decode("utf-8") - break + if not found_readme: + response += "## No readme found." return response def get_commit_log(self) -> list: @@ -48,7 +63,7 @@ class GitGmiRepo: return log # reverse chronical order def view_log(self) -> str: - response = f"{STATUS_SUCCESS} {META_GEMINI}\n" + response = f"{STATUS_SUCCESS} {META_GEMINI}\n" + self.generate_header() log = self.get_commit_log() for cmt in log: response += f"## {cmt['short_id']} - {cmt['author']}\n{cmt['msg']}\n\n" @@ -123,7 +138,7 @@ class GitGmiRepo: def view_tree(self, branch: str, location=[]) -> str: # actual Gemini response # consists of a header and a body - response = f"{STATUS_SUCCESS} {META_GEMINI}\n" + response = f"{STATUS_SUCCESS} {META_GEMINI}\n" + self.generate_header() tree = self.get_tree(branch) contents = self.list_tree(tree, location) for item in contents: @@ -151,6 +166,18 @@ class GitGmiRepo: def view_blob(self, branch: str, location=[]) -> str: blob = self.get_blob(branch, location) + response = ( + f"{STATUS_SUCCESS} {META_GEMINI}\n" + + self.generate_header() + + f"## {self.name}/{'/'.join(location)} | {blob.size} bytes\n\n" + f"=> {blob.name}?raw view raw\n\n" + f"```\n" + ) + response += blob.data.decode("utf-8") + "\n```" + return response + + def view_raw_blob(self, branch: str, location=[]) -> str: + blob = self.get_blob(branch, location) guessed_mimetype = mimetypes.guess_type(blob.name)[0] or "text/plain" response = f"{STATUS_SUCCESS} {guessed_mimetype}\n" response += blob.data.decode("utf-8") |