hmml_to_html.c: Pull from insobot's quote store
#25 Also fix the forced integration
This commit is contained in:
		
							parent
							
								
									ad8fb22211
								
							
						
					
					
						commit
						5f21a9544d
					
				|  | @ -1,6 +1,6 @@ | |||
| #if 0 | ||||
| ctime -begin ${0%.*}.ctm | ||||
| gcc -g -no-pie -fsanitize=address -Wall -std=c99 -pipe $0 -o ${0%.*} hmml.a | ||||
| gcc -g -no-pie -fsanitize=address -Wall -std=c99 -pipe $0 -o ${0%.*} hmml.a -lcurl | ||||
| ctime -end ${0%.*}.ctm | ||||
| exit | ||||
| #endif | ||||
|  | @ -18,6 +18,7 @@ typedef unsigned int bool; | |||
| #include "hmmlib.h" | ||||
| #include <getopt.h> // NOTE(matt): getopts | ||||
| //#include "config.h" // TODO(matt): Implement config.h
 | ||||
| #include <curl/curl.h> | ||||
| 
 | ||||
| #define Kilobytes(Bytes) Bytes << 10 | ||||
| #define Megabytes(Bytes) Bytes << 20 | ||||
|  | @ -809,78 +810,98 @@ StringToInt(char *String) | |||
|     return Result; | ||||
| } | ||||
| 
 | ||||
| int | ||||
| BuildQuote(quote_info *Info, char *Speaker, int ID, char *QuoteDir) | ||||
| size_t | ||||
| CurlIntoBuffer(char *InPtr, size_t CharLength, size_t Chars, char **OutputPtr) | ||||
| { | ||||
|     char Path[255] = {0}; | ||||
|     sprintf(Path, "%s/#%s", QuoteDir, Speaker); | ||||
|     FILE *File; | ||||
|     if(!(File = fopen(Path, "r"))) | ||||
|     int Length = CharLength * Chars; | ||||
|     int i; | ||||
|     for(i = 0; InPtr[i] && i < Length; ++i) | ||||
|     { | ||||
|         perror(Path); | ||||
|         return 1; | ||||
|         *((*OutputPtr)++) = InPtr[i]; | ||||
|     } | ||||
|     **OutputPtr = '\0'; | ||||
|     return Length; | ||||
| }; | ||||
| 
 | ||||
|     fseek(File, 0, SEEK_END); | ||||
|     int Length = ftell(File); | ||||
|     fseek(File, 0, SEEK_SET); | ||||
|     char *Buffer; | ||||
|     if(!(Buffer = malloc(Length))) | ||||
| #include <time.h> | ||||
| 
 | ||||
| int | ||||
| BuildQuote(quote_info *Info, char *Speaker, int ID) | ||||
| { | ||||
|     char QuotesURL[256]; | ||||
|     CopyString(QuotesURL, "https://dev.abaines.me.uk/quotes/%s.raw", Speaker); | ||||
| 
 | ||||
|     buffer QuoteStaging; | ||||
|     QuoteStaging.ID = "QuoteStaging"; | ||||
|     QuoteStaging.Size = Kilobytes(256); | ||||
|     if(!(QuoteStaging.Location = malloc(QuoteStaging.Size))) | ||||
|     { | ||||
|         perror("BuildQuote"); | ||||
|         perror("malloc"); | ||||
|     } | ||||
|     QuoteStaging.Ptr = QuoteStaging.Location; | ||||
| 
 | ||||
|     CURL *curl = curl_easy_init(); | ||||
|     if(curl) { | ||||
|         CURLcode res; | ||||
|         curl_easy_setopt(curl, CURLOPT_WRITEDATA, &QuoteStaging.Ptr); | ||||
|         curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, CurlIntoBuffer); | ||||
|         curl_easy_setopt(curl, CURLOPT_URL, QuotesURL); | ||||
|         if((res = curl_easy_perform(curl))) | ||||
|         { | ||||
|             fprintf(stderr, "%s", curl_easy_strerror(res)); | ||||
|         } | ||||
|         curl_easy_cleanup(curl); | ||||
|     } | ||||
|     fread(Buffer, Length, 1, File); | ||||
|     fclose(File); | ||||
| 
 | ||||
|     // TODO(matt): Search the quote store in reverse
 | ||||
|     char *InPtr = Buffer; | ||||
|     QuoteStaging.Ptr = QuoteStaging.Location; | ||||
| 
 | ||||
|     while(InPtr - Buffer < Length) | ||||
|     while(*QuoteStaging.Ptr) | ||||
|     { | ||||
|         char InID[4] = { 0 }; | ||||
|         char InTime[16] = { 0 }; | ||||
|         char *OutPtr = InID; | ||||
|         while(*InPtr != ',') | ||||
|         while(*QuoteStaging.Ptr != ',') | ||||
|         { | ||||
|             *OutPtr++ = *InPtr++; | ||||
|             *OutPtr++ = *QuoteStaging.Ptr++; | ||||
|         } | ||||
|         *OutPtr = '\0'; | ||||
| 
 | ||||
|         if(StringToInt(InID) == ID) | ||||
|         { | ||||
|             InPtr += 2; | ||||
|             OutPtr = Info->Date; | ||||
|             while(*InPtr != '"') | ||||
|             QuoteStaging.Ptr += 1; | ||||
|             OutPtr = InTime; | ||||
|             while(*QuoteStaging.Ptr != ',') | ||||
|             { | ||||
|                 *OutPtr++ = *InPtr++; | ||||
|                 *OutPtr++ = *QuoteStaging.Ptr++; | ||||
|             } | ||||
|             *OutPtr = '\0'; | ||||
| 
 | ||||
|             InPtr += 3; | ||||
|             long int Time = StringToInt(InTime); | ||||
|             strftime(Info->Date, 32, "%d %B, %Y", gmtime(&Time)); | ||||
| 
 | ||||
|             QuoteStaging.Ptr += 1; | ||||
|             OutPtr = Info->Text; | ||||
|             while(*InPtr != '\n') | ||||
|             while(*QuoteStaging.Ptr != '\n') | ||||
|             { | ||||
|                 if(*InPtr == '\\') | ||||
|                 { | ||||
|                     ++InPtr; | ||||
|                 } | ||||
|                 *OutPtr++ = *InPtr++; | ||||
|                 *OutPtr++ = *QuoteStaging.Ptr++; | ||||
|             } | ||||
|             *--OutPtr = '\0'; | ||||
| 
 | ||||
|             free(Buffer); | ||||
|             FreeBuffer(&QuoteStaging); | ||||
|             return 0; | ||||
|         } | ||||
|         else | ||||
|         { | ||||
|             while(*InPtr != '\n') | ||||
|             while(*QuoteStaging.Ptr != '\n') | ||||
|             { | ||||
|                 ++InPtr; | ||||
|                 ++QuoteStaging.Ptr; | ||||
|             } | ||||
|             ++InPtr; | ||||
|             ++QuoteStaging.Ptr; | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     free(Buffer); | ||||
|     FreeBuffer(&QuoteStaging); | ||||
|     return 1; | ||||
| } | ||||
| 
 | ||||
|  | @ -1215,7 +1236,7 @@ ParseConfig(buffer *Buffer, char *Username) | |||
| #endif | ||||
| 
 | ||||
| void | ||||
| PrintUsage(char *BinaryLocation, char *DefaultCSSDir, char *DefaultImagesDir, char *DefaultJSDir, char *DefaultDefaultMedium, char *DefaultOutLocation, char *DefaultQuoteDir, char *DefaultTemplateLocation) | ||||
| PrintUsage(char *BinaryLocation, char *DefaultCSSDir, char *DefaultImagesDir, char *DefaultJSDir, char *DefaultDefaultMedium, char *DefaultOutLocation, char *DefaultTemplateLocation) | ||||
| { | ||||
|     fprintf(stderr, "Usage: %s [option(s)] filename(s)\n" | ||||
|             "\n" | ||||
|  | @ -1232,8 +1253,6 @@ PrintUsage(char *BinaryLocation, char *DefaultCSSDir, char *DefaultImagesDir, ch | |||
|             "        Override default default medium (\"%s\")\n" | ||||
|             "    -o <output location>\n" | ||||
|             "        Override default output location (\"%s\")\n" | ||||
|             "    -q <quotes directory path>\n" | ||||
|             "        Override default quotes directory (\"%s\")\n" | ||||
|             "    -t <template location>\n" | ||||
|             "        Override default template location (\"%s\")\n" | ||||
|             "        and automatically enable integration\n" | ||||
|  | @ -1254,7 +1273,7 @@ PrintUsage(char *BinaryLocation, char *DefaultCSSDir, char *DefaultImagesDir, ch | |||
|             "       <!-- __CINERA_SCRIPT__ --> (must come after <!-- __CINERA_PLAYER__ -->)\n" | ||||
|             "   Other available tags include:\n" | ||||
|             "       <!-- __CINERA_TITLE__ -->\n", | ||||
|             BinaryLocation, DefaultCSSDir, DefaultImagesDir, DefaultJSDir, DefaultDefaultMedium, DefaultQuoteDir, DefaultOutLocation, DefaultTemplateLocation); | ||||
|             BinaryLocation, DefaultCSSDir, DefaultImagesDir, DefaultJSDir, DefaultDefaultMedium, DefaultOutLocation, DefaultTemplateLocation); | ||||
| } | ||||
| 
 | ||||
| int | ||||
|  | @ -1284,9 +1303,6 @@ main(int ArgC, char **Args) | |||
|     char *DefaultOutIntegratedLocation = "out_integrated.html"; | ||||
|     char *OutIntegratedLocation = DefaultOutIntegratedLocation; | ||||
| 
 | ||||
|     char *DefaultQuoteDir = "/home/matt/git/GitHub/insofaras/25fc16d58a297a486334"; | ||||
|     char *QuoteDir = DefaultQuoteDir; | ||||
| 
 | ||||
|     char *CINERA_MODE = getenv("CINERA_MODE"); | ||||
| 
 | ||||
|     bool DefaultForceIntegration = FALSE; | ||||
|  | @ -1294,7 +1310,7 @@ main(int ArgC, char **Args) | |||
| 
 | ||||
|     if(ArgC < 2) | ||||
|     { | ||||
|         PrintUsage(Args[0], DefaultCSSDir, DefaultImagesDir, DefaultJSDir, DefaultDefaultMedium, DefaultQuoteDir, DefaultOutLocation, DefaultTemplateLocation); | ||||
|         PrintUsage(Args[0], DefaultCSSDir, DefaultImagesDir, DefaultJSDir, DefaultDefaultMedium, DefaultOutLocation, DefaultTemplateLocation); | ||||
|         return 1; | ||||
|     } | ||||
| 
 | ||||
|  | @ -1322,8 +1338,6 @@ main(int ArgC, char **Args) | |||
|                 OutLocation = optarg; | ||||
|                 OutIntegratedLocation = optarg; | ||||
|                 break; | ||||
|             case 'q': | ||||
|                 QuoteDir = optarg; | ||||
|             case 't': | ||||
|                 TemplateLocation = optarg; | ||||
|                 CINERA_MODE="INTEGRATE"; | ||||
|  | @ -1332,7 +1346,7 @@ main(int ArgC, char **Args) | |||
|             // Override config path, once we even have a default!
 | ||||
|             case 'h': | ||||
|             default: | ||||
|                 PrintUsage(Args[0], DefaultCSSDir, DefaultImagesDir, DefaultJSDir, DefaultDefaultMedium, DefaultQuoteDir, DefaultOutLocation, DefaultTemplateLocation); | ||||
|                 PrintUsage(Args[0], DefaultCSSDir, DefaultImagesDir, DefaultJSDir, DefaultDefaultMedium, DefaultOutLocation, DefaultTemplateLocation); | ||||
|                 return 1; | ||||
|         } | ||||
|     } | ||||
|  | @ -1340,7 +1354,7 @@ main(int ArgC, char **Args) | |||
|     if(optind == ArgC) | ||||
|     { | ||||
|         fprintf(stderr, "%s: requires at least one input .hmml file\n", Args[0]); | ||||
|         PrintUsage(Args[0], DefaultCSSDir, DefaultImagesDir, DefaultJSDir, DefaultDefaultMedium, DefaultQuoteDir, DefaultOutLocation, DefaultTemplateLocation); | ||||
|         PrintUsage(Args[0], DefaultCSSDir, DefaultImagesDir, DefaultJSDir, DefaultDefaultMedium, DefaultOutLocation, DefaultTemplateLocation); | ||||
|         return 1; | ||||
|     } | ||||
| 
 | ||||
|  | @ -1377,21 +1391,6 @@ main(int ArgC, char **Args) | |||
|     { | ||||
|         JSDir[StringLength(JSDir) - 1] = '\0'; | ||||
|     } | ||||
|     if(QuoteDir[StringLength(QuoteDir) - 1] == '/') | ||||
|     { | ||||
|         QuoteDir[StringLength(QuoteDir) - 1] = '\0'; | ||||
|     } | ||||
| 
 | ||||
|     // TODO(matt): Put the Errors in the MemoryArena
 | ||||
|     buffer Errors; | ||||
|     Errors.ID = "Errors"; | ||||
|     Errors.Size = Kilobytes(1); | ||||
|     if(!(Errors.Location = malloc(Errors.Size))) | ||||
|     { | ||||
|         perror(Args[0]); return 1; | ||||
|     } | ||||
|     Errors.Ptr = Errors.Location; | ||||
|     bool HaveErrors = FALSE; | ||||
| 
 | ||||
|     buffer Template; | ||||
|     Template.ID = "Template"; | ||||
|  | @ -1399,6 +1398,17 @@ main(int ArgC, char **Args) | |||
|     { | ||||
|         if(CINERA_MODE && !StringsDiffer(CINERA_MODE, "INTEGRATE")) | ||||
|         { | ||||
|             // TODO(matt): Put the Errors in the MemoryArena
 | ||||
|             buffer Errors; | ||||
|             Errors.ID = "Errors"; | ||||
|             Errors.Size = Kilobytes(1); | ||||
|             if(!(Errors.Location = malloc(Errors.Size))) | ||||
|             { | ||||
|                 perror(Args[0]); return 1; | ||||
|             } | ||||
|             Errors.Ptr = Errors.Location; | ||||
|             bool HaveErrors = FALSE; | ||||
| 
 | ||||
|             FILE *TemplateFile; | ||||
|             if(!(TemplateFile = fopen(TemplateLocation, "r"))) | ||||
|             { | ||||
|  | @ -1548,12 +1558,15 @@ main(int ArgC, char **Args) | |||
|             } | ||||
|             else | ||||
|             { | ||||
|                 if(!FoundIncludes){ CopyStringToBuffer(&Errors, "Template must include one <!-- __CINERA_INCLUDES__ --> tag\n"); }; | ||||
|                 if(!FoundMenus){ CopyStringToBuffer(&Errors, "Template must include one <!-- __CINERA_MENUS__ --> tag\n"); }; | ||||
|                 if(!FoundPlayer){ CopyStringToBuffer(&Errors, "Template must include one <!-- __CINERA_PLAYER__ --> tag\n"); }; | ||||
|                 if(!FoundScript){ CopyStringToBuffer(&Errors, "Template must include one <!-- __CINERA_SCRIPT__ --> tag\n"); }; | ||||
|                 fprintf(stderr, "%s", Errors.Location); | ||||
|                 free(Template.Location); free(Errors.Location); return 1; | ||||
|                 if(!ForceIntegration) | ||||
|                 { | ||||
|                     if(!FoundIncludes){ CopyStringToBuffer(&Errors, "Template must include one <!-- __CINERA_INCLUDES__ --> tag\n"); }; | ||||
|                     if(!FoundMenus){ CopyStringToBuffer(&Errors, "Template must include one <!-- __CINERA_MENUS__ --> tag\n"); }; | ||||
|                     if(!FoundPlayer){ CopyStringToBuffer(&Errors, "Template must include one <!-- __CINERA_PLAYER__ --> tag\n"); }; | ||||
|                     if(!FoundScript){ CopyStringToBuffer(&Errors, "Template must include one <!-- __CINERA_SCRIPT__ --> tag\n"); }; | ||||
|                     fprintf(stderr, "%s", Errors.Location); | ||||
|                     free(Template.Location); free(Errors.Location); return 1; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | @ -1564,6 +1577,7 @@ main(int ArgC, char **Args) | |||
|     if(!(MemoryArena.Location = calloc(MemoryArena.Size, 1))) | ||||
|     { | ||||
|         perror(Args[0]); | ||||
|         free(Template.Location); | ||||
|         return 1; | ||||
|     } | ||||
|     int ClaimedMemory = 0; | ||||
|  | @ -1627,6 +1641,7 @@ main(int ArgC, char **Args) | |||
|         { | ||||
|             perror(Args[FileIndex]); | ||||
|             free(MemoryArena.Location); | ||||
|             free(Template.Location); | ||||
|             return 1; | ||||
|         } | ||||
| 
 | ||||
|  | @ -1719,6 +1734,7 @@ main(int ArgC, char **Args) | |||
|                 fprintf(stderr, "%s: Missing annotator in [video] node\n", Args[FileIndex]); | ||||
|                 hmml_free(&HMML); | ||||
|                 free(MemoryArena.Location); | ||||
|                 free(Template.Location); | ||||
|                 return 1; | ||||
|             } | ||||
| 
 | ||||
|  | @ -1893,6 +1909,7 @@ main(int ArgC, char **Args) | |||
|                                         "appear in the references menu\n", Args[FileIndex], Anno->line); | ||||
|                                 hmml_free(&HMML); | ||||
|                                 free(MemoryArena.Location); | ||||
|                                 free(Template.Location); | ||||
|                                 return 1; | ||||
|                             } | ||||
|                             ++ReferencesArray[RefIdentifier - 1].IdentifierCount; | ||||
|  | @ -1909,6 +1926,7 @@ main(int ArgC, char **Args) | |||
|                                     fprintf(stderr, "%s:%d: Too many timecodes associated with one reference (increase REF_MAX_IDENTIFIER)\n", Args[FileIndex], Anno->line); | ||||
|                                     hmml_free(&HMML); | ||||
|                                     free(MemoryArena.Location); | ||||
|                                     free(Template.Location); | ||||
|                                     return 1; | ||||
|                                 } | ||||
|                                 if(CurrentRef->isbn) | ||||
|  | @ -1936,6 +1954,7 @@ main(int ArgC, char **Args) | |||
|                                     fprintf(stderr, "%s:%d: Reference must have an ISBN or URL\n", Args[FileIndex], Anno->line); | ||||
|                                     hmml_free(&HMML); | ||||
|                                     free(MemoryArena.Location); | ||||
|                                     free(Template.Location); | ||||
|                                     return 1; | ||||
|                                 } | ||||
|                             } | ||||
|  | @ -1949,6 +1968,7 @@ main(int ArgC, char **Args) | |||
|                                         "appear in the references menu\n", Args[FileIndex], Anno->line); | ||||
|                                 hmml_free(&HMML); | ||||
|                                 free(MemoryArena.Location); | ||||
|                                 free(Template.Location); | ||||
|                                 return 1; | ||||
|                             } | ||||
|                             ++ReferencesArray[UniqueRefs].IdentifierCount; | ||||
|  | @ -1970,6 +1990,7 @@ main(int ArgC, char **Args) | |||
|                                 fprintf(stderr, "%s:%d: Reference must have an ISBN or URL\n", Args[FileIndex], Anno->line); | ||||
|                                 hmml_free(&HMML); | ||||
|                                 free(MemoryArena.Location); | ||||
|                                 free(Template.Location); | ||||
|                                 return 1; | ||||
|                             } | ||||
| 
 | ||||
|  | @ -1990,6 +2011,7 @@ main(int ArgC, char **Args) | |||
|                                 fprintf(stderr, "%s:%d: Reference must have an ISBN or URL", Args[FileIndex], Anno->line); | ||||
|                                 hmml_free(&HMML); | ||||
|                                 free(MemoryArena.Location); | ||||
|                                 free(Template.Location); | ||||
|                                 return 1; | ||||
|                             } | ||||
|                         } | ||||
|  | @ -2062,15 +2084,16 @@ main(int ArgC, char **Args) | |||
| 
 | ||||
|                     HasQuote = TRUE; | ||||
| 
 | ||||
|                     if(BuildQuote(&QuoteInfo, Anno->quote.author ? Anno->quote.author : HMML.metadata.stream_username ? HMML.metadata.stream_username : HMML.metadata.member, Anno->quote.id, QuoteDir) == 1) | ||||
|                     if(BuildQuote(&QuoteInfo, Anno->quote.author ? Anno->quote.author : HMML.metadata.stream_username ? HMML.metadata.stream_username : HMML.metadata.member, Anno->quote.id) == 1) | ||||
|                     { | ||||
|                         fprintf(stderr, "%s:%d: Quote #%s %d not found! Consider pulling the latest quotes\n", | ||||
|                         fprintf(stderr, "%s:%d: Quote #%s %d not found. Unlucky!\n", | ||||
|                                 Args[FileIndex], | ||||
|                                 Anno->line, | ||||
|                                 Anno->quote.author ? Anno->quote.author : HMML.metadata.stream_username ? HMML.metadata.stream_username : HMML.metadata.member, | ||||
|                                 Anno->quote.id); | ||||
|                         hmml_free(&HMML); | ||||
|                         free(MemoryArena.Location); | ||||
|                         free(Template.Location); | ||||
|                         return 1; | ||||
|                     } | ||||
| 
 | ||||
|  |  | |||
		Loading…
	
		Reference in New Issue