hmml_to_youtube.c: First functional build [#9]
Using insofaras's hmmlib.h to produce even nicer output
This commit is contained in:
		
							parent
							
								
									dedb407dde
								
							
						
					
					
						commit
						e973b8376d
					
				|  | @ -0,0 +1,324 @@ | |||
| #if 0 | ||||
| ctime -begin ${0%.*}.ctm | ||||
| clang -g -fsanitize=address $0 -o ${0%.*} hmml.a | ||||
| ctime -end ${0%.*}.ctm | ||||
| exit | ||||
| #endif | ||||
| 
 | ||||
| #include "hmmlib.h" | ||||
| #include <stdio.h> | ||||
| #include <stdlib.h> | ||||
| 
 | ||||
| #ifndef bool | ||||
| typedef unsigned int bool; | ||||
| #endif | ||||
| 
 | ||||
| #define TRUE 1 | ||||
| #define FALSE 0 | ||||
| 
 | ||||
| typedef struct | ||||
| { | ||||
|     char *Location; | ||||
|     char *Ptr; | ||||
|     int Size; | ||||
| } buffer; | ||||
| 
 | ||||
| int | ||||
| StringLength(char *String) | ||||
| { | ||||
|     int i = 0; | ||||
|     while(String[i]) | ||||
|     { | ||||
|         ++i; | ||||
|     } | ||||
|     return i; | ||||
| } | ||||
| 
 | ||||
| bool | ||||
| StringsDiffer(char A[], char B[]) | ||||
| { | ||||
|     char APtr, BPtr; | ||||
|     while(*A) | ||||
|     { | ||||
|         APtr = *A++; | ||||
|         BPtr = *B++; | ||||
|         if(APtr >= 'A' && APtr <= 'Z') | ||||
|         { | ||||
|             APtr += 32; | ||||
|         } | ||||
|         if(BPtr >= 'A' && BPtr <= 'Z') | ||||
|         { | ||||
|             BPtr += 32; | ||||
|         } | ||||
|         if(APtr != BPtr) | ||||
|         { | ||||
|             return TRUE; | ||||
|         } | ||||
|     } | ||||
|     return FALSE; | ||||
| } | ||||
| 
 | ||||
| int | ||||
| main(int ArgC, char **Args) | ||||
| { | ||||
|     if(ArgC < 2) | ||||
|     { | ||||
|         fprintf(stderr, "Usage: %s filename(s)\n", Args[0]); | ||||
|         return 1; | ||||
|     } | ||||
| 
 | ||||
|     FILE *InFile; | ||||
|     for(int FileIndex = 1; FileIndex < ArgC; ++FileIndex) | ||||
|     { | ||||
|         if(!(InFile = fopen(Args[FileIndex], "r"))) | ||||
|         { | ||||
|             perror(Args[0]); | ||||
|             return 1; | ||||
|         } | ||||
| 
 | ||||
|         // Init MemoryArena
 | ||||
|         int ArenaSize = 1024 * 32; | ||||
|         char *MemoryArena; | ||||
|         if(!(MemoryArena = calloc(ArenaSize, 1))) | ||||
|         { | ||||
|             perror(Args[0]); | ||||
|             return 1; | ||||
|         } | ||||
|         int ClaimedMemory = 0; | ||||
| 
 | ||||
|         // Buffers and Pointers
 | ||||
|         char *InPtr; // Associated buffer is allocated by hmml_parse_file()
 | ||||
|         buffer Temp; // Temp.Ptr may be used independently of Temp.Location
 | ||||
|         buffer Out; | ||||
| 
 | ||||
|         //printf("Reading %s\n", Args[FileIndex]);
 | ||||
|         HMML_Output HMML = hmml_parse_file(InFile); | ||||
|         fclose(InFile); | ||||
| 
 | ||||
|         if(HMML.well_formed) | ||||
|         { | ||||
|             Out.Location = MemoryArena + ClaimedMemory;  | ||||
|             Out.Size = 1024*16; | ||||
|             ClaimedMemory += Out.Size; | ||||
| 
 | ||||
|             bool WritingChat = TRUE; | ||||
| 
 | ||||
|             char *Member = HMML.metadata.twitch ? | ||||
|                 HMML.metadata.twitch : | ||||
|                 HMML.metadata.member; | ||||
| 
 | ||||
|             for(int BuildPass = 0; BuildPass < 2; ++BuildPass) | ||||
|             { | ||||
|                 if(WritingChat == FALSE) | ||||
|                 { | ||||
|                     printf("%s: %ld characters over budget. Shrinking...\n", Args[FileIndex], (Out.Ptr - Out.Location) - 5000); | ||||
|                 } | ||||
| 
 | ||||
|                 Out.Ptr = Out.Location; | ||||
|                 for(int AnnotationIndex = 0; AnnotationIndex < HMML.annotation_count; ++AnnotationIndex) | ||||
|                 { | ||||
|                     if(HMML.annotations[AnnotationIndex].author && !WritingChat) | ||||
|                     { | ||||
|                         goto skip; | ||||
|                     } | ||||
| 
 | ||||
|                     InPtr = HMML.annotations[AnnotationIndex].time; | ||||
| 
 | ||||
|                     while(*InPtr) | ||||
|                     { | ||||
|                         *Out.Ptr++ = *InPtr++; | ||||
|                     } | ||||
|                     *Out.Ptr++ = ' '; | ||||
| 
 | ||||
|                     InPtr = HMML.annotations[AnnotationIndex].text; | ||||
| 
 | ||||
| 
 | ||||
|                     // TODO(matt): Get this logic correct
 | ||||
|                     if(HMML.annotations[AnnotationIndex].author) | ||||
|                     { | ||||
|                         Temp.Location = "Chat comment: \""; | ||||
|                         Temp.Ptr = Temp.Location; | ||||
|                         while(*Temp.Ptr) | ||||
|                         { | ||||
|                             *Out.Ptr++ = *Temp.Ptr++; | ||||
|                         } | ||||
| 
 | ||||
|                         if(*HMML.annotations[AnnotationIndex].text) | ||||
|                         { | ||||
|                             InPtr = HMML.annotations[AnnotationIndex].text; | ||||
|                             if(!StringsDiffer(Member, InPtr)) | ||||
|                             { | ||||
|                                 InPtr += StringLength(Member); | ||||
| 
 | ||||
|                                 while(*InPtr && | ||||
|                                         !(*InPtr >= '0' && *InPtr <= '9') && | ||||
|                                         !(*InPtr >= 'A' && *InPtr <= 'Z') && | ||||
|                                         !(*InPtr >= 'a' && *InPtr <= 'z')) | ||||
|                                 { | ||||
|                                     ++InPtr; | ||||
|                                 } | ||||
| 
 | ||||
|                                 if(*InPtr && *InPtr >= 'a' && *InPtr <= 'z') | ||||
|                                 { | ||||
|                                     *InPtr -= 32; | ||||
|                                 } | ||||
|                             } | ||||
|                         } | ||||
|                     } | ||||
| 
 | ||||
|                     if(HMML.annotations[AnnotationIndex].reference_count) | ||||
|                     { | ||||
|                         for(int RefIndex = 0; RefIndex < HMML.annotations[AnnotationIndex].reference_count; ++RefIndex) | ||||
|                         { | ||||
|                             while(InPtr - HMML.annotations[AnnotationIndex].text < HMML.annotations[AnnotationIndex].references[RefIndex].offset) | ||||
|                             { | ||||
|                                 *Out.Ptr++ = *InPtr++; | ||||
|                             } | ||||
| 
 | ||||
|                             if(HMML.annotations[AnnotationIndex].references[RefIndex].offset == 0) | ||||
|                             { | ||||
|                                 if(HMML.annotations[AnnotationIndex].references[RefIndex].page) | ||||
|                                 { | ||||
|                                     Temp.Ptr = HMML.annotations[AnnotationIndex].references[RefIndex].page; | ||||
|                                 } | ||||
|                                 else if(HMML.annotations[AnnotationIndex].references[RefIndex].site) | ||||
|                                 { | ||||
|                                     Temp.Ptr = HMML.annotations[AnnotationIndex].references[RefIndex].site; | ||||
|                                 } | ||||
|                                 else if(HMML.annotations[AnnotationIndex].references[RefIndex].title) | ||||
|                                 { | ||||
|                                     Temp.Ptr = HMML.annotations[AnnotationIndex].references[RefIndex].title; | ||||
|                                 } | ||||
| 
 | ||||
|                                 if(Out.Ptr[-1] != '"' && Out.Ptr[-1] != ' ') | ||||
|                                 { | ||||
|                                     *Out.Ptr++ = ' '; | ||||
|                                 } | ||||
| 
 | ||||
|                                 while(*Temp.Ptr) | ||||
|                                 { | ||||
|                                     *Out.Ptr++ = *Temp.Ptr++; | ||||
|                                 } | ||||
|                             } | ||||
| 
 | ||||
|                             if(HMML.annotations[AnnotationIndex].references[RefIndex].url) | ||||
|                             { | ||||
|                                 Temp.Ptr = HMML.annotations[AnnotationIndex].references[RefIndex].url; | ||||
| 
 | ||||
|                                 if(HMML.annotations[AnnotationIndex].references[RefIndex].offset < StringLength(HMML.annotations[AnnotationIndex].text) || | ||||
|                                         RefIndex < HMML.annotations[AnnotationIndex].reference_count-1) | ||||
|                                 { | ||||
|                                     if(Out.Ptr[-1] != ' ') | ||||
|                                     { | ||||
|                                         *Out.Ptr++ = ' '; | ||||
|                                     } | ||||
|                                     *Out.Ptr++ = '-'; | ||||
|                                     *Out.Ptr++ = ' '; | ||||
|                                     while(*Temp.Ptr) | ||||
|                                     { | ||||
|                                         *Out.Ptr++ = *Temp.Ptr++; | ||||
|                                     } | ||||
|                                     *Out.Ptr++ = ' '; | ||||
|                                     *Out.Ptr++ = '-'; | ||||
|                                 } | ||||
|                                 else | ||||
|                                 { | ||||
|                                     if(Out.Ptr[-1] == ' ') | ||||
|                                     { | ||||
|                                         --Out.Ptr; | ||||
|                                     } | ||||
|                                     if(InPtr[-3] != ':') | ||||
|                                     { | ||||
|                                         *Out.Ptr++ = ':'; | ||||
|                                         *Out.Ptr++ = ' '; | ||||
|                                     } | ||||
|                                     while(*Temp.Ptr) | ||||
|                                     { | ||||
|                                         *Out.Ptr++ = *Temp.Ptr++; | ||||
|                                     } | ||||
|                                 } | ||||
|                             } | ||||
|                         } | ||||
|                     } | ||||
| 
 | ||||
|                     while(*InPtr) | ||||
|                     { | ||||
|                         *Out.Ptr++ = *InPtr++; | ||||
|                     } | ||||
| 
 | ||||
|                     if(HMML.annotations[AnnotationIndex].author && | ||||
|                             WritingChat == TRUE) | ||||
|                     { | ||||
|                         *Out.Ptr++ = '\"'; | ||||
|                     } | ||||
| 
 | ||||
|                     *Out.Ptr++ = '\n'; | ||||
| skip: {}; | ||||
|                 } | ||||
| 
 | ||||
|                 Temp.Location = MemoryArena + ClaimedMemory; | ||||
|                 Temp.Size = 256; | ||||
|                 ClaimedMemory += Temp.Size; | ||||
|                 Temp.Ptr = Temp.Location; | ||||
| 
 | ||||
|                 sprintf(Temp.Location, "\nAnnotated by %s - https://handmade.network/m/%s\n", | ||||
|                         HMML.metadata.annotator, HMML.metadata.annotator); | ||||
| 
 | ||||
|                 while(*Temp.Ptr) | ||||
|                 { | ||||
|                     *Out.Ptr++ = *Temp.Ptr++; | ||||
|                 } | ||||
| 
 | ||||
|                 *Temp.Location = '\0'; | ||||
|                 ClaimedMemory -= Temp.Size; | ||||
| 
 | ||||
|                 if(Out.Ptr - Out.Location > 5000) | ||||
|                 { | ||||
|                     WritingChat = FALSE; | ||||
|                 } | ||||
|                 else | ||||
|                 { | ||||
|                     break; | ||||
|                 } | ||||
|             } | ||||
| 
 | ||||
|             if(Out.Ptr - Out.Location > 5000) | ||||
|             { | ||||
|                 fprintf(stderr, "%s: %ld characters over budget. Requires manual shrinking!\n", Args[FileIndex], (Out.Ptr - Out.Location) - 5000); | ||||
|             } | ||||
| 
 | ||||
|             // NOTE(matt): At this point we should have filled a buffer with the stuff
 | ||||
|             hmml_free(&HMML); | ||||
| 
 | ||||
|             // NOTE(matt): Open a file for writing out to
 | ||||
|             Temp.Location = MemoryArena + ClaimedMemory; | ||||
|             Temp.Size = StringLength(Args[FileIndex]) + 5; | ||||
|             ClaimedMemory += Temp.Size; | ||||
| 
 | ||||
|             sprintf(Temp.Location, "%s.txt", Args[FileIndex]); | ||||
|             //printf("Writing to %s\n", Temp.Location);
 | ||||
|             FILE *OutFile; | ||||
|             if(!(OutFile = fopen(Temp.Location, "w"))) | ||||
|             { | ||||
|                 perror(Args[0]); | ||||
|                 free(MemoryArena); | ||||
|                 return 1; | ||||
|             } | ||||
| 
 | ||||
|             *Temp.Location = '\0'; | ||||
|             ClaimedMemory -= Temp.Size; | ||||
| 
 | ||||
|             fwrite(Out.Location, Out.Ptr - Out.Location, 1, OutFile); | ||||
|             fclose(OutFile); | ||||
|         } | ||||
|         else | ||||
|         { | ||||
|             fprintf(stderr, "%s:%d: %s\n", Args[FileIndex], HMML.error.line, HMML.error.message); | ||||
|             hmml_free(&HMML); | ||||
|         } | ||||
| 
 | ||||
|         free(MemoryArena); | ||||
|     } | ||||
| 
 | ||||
|     return 0; | ||||
| } | ||||
		Loading…
	
		Reference in New Issue