generator: Can perform synchronous generation
This commit is contained in:
parent
ec98e521dc
commit
1769938205
13 changed files with 214 additions and 81 deletions
|
|
@ -1,25 +1,21 @@
|
|||
package api
|
||||
|
||||
import (
|
||||
"bytes"
|
||||
"encoding/json"
|
||||
"fmt"
|
||||
"io/ioutil"
|
||||
"log"
|
||||
"net"
|
||||
"net/http"
|
||||
"path"
|
||||
"strconv"
|
||||
"strings"
|
||||
"time"
|
||||
|
||||
"srs.epita.fr/fic-server/admin/generation"
|
||||
"srs.epita.fr/fic-server/libfic"
|
||||
|
||||
"github.com/gin-gonic/gin"
|
||||
)
|
||||
|
||||
var GeneratorSocket string
|
||||
|
||||
func declareClaimsRoutes(router *gin.RouterGroup) {
|
||||
// Tasks
|
||||
router.GET("/claims", getClaims)
|
||||
|
|
@ -292,7 +288,7 @@ func clearClaims(c *gin.Context) {
|
|||
}
|
||||
|
||||
func generateTeamIssuesFile(team fic.Team) error {
|
||||
if GeneratorSocket == "" {
|
||||
if generation.GeneratorSocket == "" {
|
||||
if my, err := team.MyIssueFile(); err != nil {
|
||||
return fmt.Errorf("Unable to generate issue FILE (tid=%d): %w", team.Id, err)
|
||||
} else if j, err := json.Marshal(my); err != nil {
|
||||
|
|
@ -301,35 +297,16 @@ func generateTeamIssuesFile(team fic.Team) error {
|
|||
return fmt.Errorf("Unable to write issues' file: %w", err)
|
||||
}
|
||||
} else {
|
||||
buf, err := json.Marshal(fic.GenStruct{Type: fic.GenTeamIssues, TeamId: team.Id})
|
||||
if err != nil {
|
||||
return fmt.Errorf("Something is wrong with JSON encoder: %w", err)
|
||||
}
|
||||
|
||||
sockType := "unix"
|
||||
if strings.Contains(GeneratorSocket, ":") {
|
||||
sockType = "tcp"
|
||||
}
|
||||
|
||||
socket, err := net.Dial(sockType, GeneratorSocket)
|
||||
resp, err := generation.PerformGeneration(fic.GenStruct{Type: fic.GenTeamIssues, TeamId: team.Id})
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
defer socket.Close()
|
||||
defer resp.Body.Close()
|
||||
|
||||
httpClient := &http.Client{
|
||||
Transport: &http.Transport{
|
||||
Dial: func(network, addr string) (net.Conn, error) {
|
||||
return socket, nil
|
||||
},
|
||||
},
|
||||
if resp.StatusCode != http.StatusOK {
|
||||
v, _ := ioutil.ReadAll(resp.Body)
|
||||
return fmt.Errorf("%s", string(v))
|
||||
}
|
||||
|
||||
resp, err := httpClient.Post("http://localhost/enqueue", "application/json", bytes.NewReader(buf))
|
||||
if err != nil {
|
||||
return fmt.Errorf("Unable to enqueue new generation event: %w", err)
|
||||
}
|
||||
resp.Body.Close()
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
|
|
|||
|
|
@ -12,6 +12,7 @@ import (
|
|||
"strconv"
|
||||
"time"
|
||||
|
||||
"srs.epita.fr/fic-server/admin/generation"
|
||||
"srs.epita.fr/fic-server/admin/sync"
|
||||
"srs.epita.fr/fic-server/libfic"
|
||||
"srs.epita.fr/fic-server/settings"
|
||||
|
|
@ -47,6 +48,7 @@ func declareSettingsRoutes(router *gin.RouterGroup) {
|
|||
apiNextSettingsRoutes.DELETE("", deleteNextSettings)
|
||||
|
||||
router.POST("/reset", reset)
|
||||
router.POST("/full-generation", fullGeneration)
|
||||
|
||||
router.GET("/prod", func(c *gin.Context) {
|
||||
c.JSON(http.StatusOK, IsProductionEnv)
|
||||
|
|
@ -80,6 +82,21 @@ func NextSettingsHandler(c *gin.Context) {
|
|||
c.Next()
|
||||
}
|
||||
|
||||
func fullGeneration(c *gin.Context) {
|
||||
resp, err := generation.FullGeneration()
|
||||
if err != nil {
|
||||
c.JSON(http.StatusInternalServerError, gin.H{
|
||||
"errmsg": err.Error(),
|
||||
})
|
||||
}
|
||||
defer resp.Body.Close()
|
||||
|
||||
v, _ := io.ReadAll(resp.Body)
|
||||
c.JSON(resp.StatusCode, gin.H{
|
||||
"errmsg": v,
|
||||
})
|
||||
}
|
||||
|
||||
func getROSettings(c *gin.Context) {
|
||||
syncMtd := "Disabled"
|
||||
if sync.GlobalImporter != nil {
|
||||
|
|
|
|||
|
|
@ -8,9 +8,9 @@ import (
|
|||
"path"
|
||||
"strings"
|
||||
|
||||
"srs.epita.fr/fic-server/admin/generation"
|
||||
"srs.epita.fr/fic-server/admin/sync"
|
||||
"srs.epita.fr/fic-server/libfic"
|
||||
"srs.epita.fr/fic-server/settings"
|
||||
|
||||
"github.com/gin-gonic/gin"
|
||||
)
|
||||
|
|
@ -292,7 +292,10 @@ func autoSync(c *gin.Context) {
|
|||
sync.EditDeepReport(&sync.SyncReport{Themes: map[string][]string{theTheme.Name: st}}, false)
|
||||
sync.DeepSyncProgress = 255
|
||||
|
||||
settings.ForceRegeneration()
|
||||
resp, err := generation.FullGeneration()
|
||||
if err == nil {
|
||||
defer resp.Body.Close()
|
||||
}
|
||||
|
||||
c.JSON(http.StatusOK, st)
|
||||
}
|
||||
|
|
|
|||
60
admin/generation/generation.go
Normal file
60
admin/generation/generation.go
Normal file
|
|
@ -0,0 +1,60 @@
|
|||
package generation
|
||||
|
||||
import (
|
||||
"bytes"
|
||||
"encoding/json"
|
||||
"fmt"
|
||||
"io"
|
||||
"net"
|
||||
"net/http"
|
||||
"strings"
|
||||
|
||||
"srs.epita.fr/fic-server/libfic"
|
||||
)
|
||||
|
||||
var GeneratorSocket string
|
||||
|
||||
func doGeneration(uri string, contenttype string, buf io.Reader) (*http.Response, error) {
|
||||
sockType := "unix"
|
||||
if strings.Contains(GeneratorSocket, ":") {
|
||||
sockType = "tcp"
|
||||
}
|
||||
|
||||
socket, err := net.Dial(sockType, GeneratorSocket)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
defer socket.Close()
|
||||
|
||||
httpClient := &http.Client{
|
||||
Transport: &http.Transport{
|
||||
Dial: func(network, addr string) (net.Conn, error) {
|
||||
return socket, nil
|
||||
},
|
||||
},
|
||||
}
|
||||
|
||||
return httpClient.Post("http://localhost"+uri, contenttype, buf)
|
||||
}
|
||||
|
||||
func EnqueueGeneration(gs fic.GenStruct) (*http.Response, error) {
|
||||
buf, err := json.Marshal(gs)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("Something is wrong with JSON encoder: %w", err)
|
||||
}
|
||||
|
||||
return doGeneration("/enqueue", "application/json", bytes.NewReader(buf))
|
||||
}
|
||||
|
||||
func PerformGeneration(gs fic.GenStruct) (*http.Response, error) {
|
||||
buf, err := json.Marshal(gs)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("Something is wrong with JSON encoder: %w", err)
|
||||
}
|
||||
|
||||
return doGeneration("/perform", "application/json", bytes.NewReader(buf))
|
||||
}
|
||||
|
||||
func FullGeneration() (*http.Response, error) {
|
||||
return doGeneration("/full", "application/json", nil)
|
||||
}
|
||||
|
|
@ -128,7 +128,13 @@ const indextpl = `<!DOCTYPE html>
|
|||
<div style="position: absolute;" id="circle1" class="circle-anim border-danger"></div>
|
||||
<div style="position: absolute;" id="circle2" class="circle-anim border-info"></div>
|
||||
</div>
|
||||
<button type="button" class="mr-2 btn btn-sm" ng-class="{'btn-info':staticFilesNeedUpdate,'btn-secondary':!staticFilesNeedUpdate}" ng-click="regenerateStaticFiles()"><span class="glyphicon glyphicon-refresh" aria-hidden="true" title="Regénérer les fichiers statiques"></span><span ng-if="staticFilesNeedUpdate"> {{ "{{ staticFilesNeedUpdate }}" }}</span></button>
|
||||
<button type="button" class="mr-2 btn btn-sm" ng-class="{'btn-info':staticFilesNeedUpdate,'btn-secondary':!staticFilesNeedUpdate}" ng-click="regenerateStaticFiles()" ng-disabled="staticRegenerationInProgress">
|
||||
<span class="glyphicon glyphicon-refresh" aria-hidden="true" title="Regénérer les fichiers statiques" ng-show="!staticRegenerationInProgress"></span>
|
||||
<div class="spinner-border spinner-border-sm" role="status" ng-show="staticRegenerationInProgress">
|
||||
<span class="sr-only">Loading...</span>
|
||||
</div>
|
||||
<span ng-if="staticFilesNeedUpdate"> {{ "{{ staticFilesNeedUpdate }}" }}</span>
|
||||
</button>
|
||||
<span ng-show="startIn > 0">
|
||||
Démarrage dans :
|
||||
<span>{{"{{ startIn }}"}}</span>"
|
||||
|
|
|
|||
|
|
@ -14,6 +14,7 @@ import (
|
|||
"syscall"
|
||||
|
||||
"srs.epita.fr/fic-server/admin/api"
|
||||
"srs.epita.fr/fic-server/admin/generation"
|
||||
"srs.epita.fr/fic-server/admin/pki"
|
||||
"srs.epita.fr/fic-server/admin/sync"
|
||||
"srs.epita.fr/fic-server/libfic"
|
||||
|
|
@ -108,7 +109,7 @@ func main() {
|
|||
flag.StringVar(&api.DashboardDir, "dashbord", "./DASHBOARD", "Base directory where save public JSON files")
|
||||
flag.StringVar(&settings.SettingsDir, "settings", settings.SettingsDir, "Base directory where load and save settings")
|
||||
flag.StringVar(&fic.FilesDir, "files", fic.FilesDir, "Base directory where found challenges files, local part")
|
||||
flag.StringVar(&api.GeneratorSocket, "generator", "./GENERATOR/generator.socket", "Path to the generator socket (used to trigger issues.json generations, use an empty string to generate locally)")
|
||||
flag.StringVar(&generation.GeneratorSocket, "generator", "./GENERATOR/generator.socket", "Path to the generator socket (used to trigger issues.json generations, use an empty string to generate locally)")
|
||||
flag.StringVar(&localImporterDirectory, "localimport", localImporterDirectory,
|
||||
"Base directory where found challenges files to import, local part")
|
||||
flag.BoolVar(&localImporterSymlink, "localimportsymlink", localImporterSymlink,
|
||||
|
|
|
|||
|
|
@ -480,15 +480,16 @@ angular.module("FICApp")
|
|||
}
|
||||
|
||||
$rootScope.staticFilesNeedUpdate = 0;
|
||||
$rootScope.staticRegenerationInProgress = false;
|
||||
$rootScope.regenerateStaticFiles = function() {
|
||||
Settings.get().$promise.then(function(config) {
|
||||
config.generation = (new Date()).toISOString();
|
||||
config.$update(function() {
|
||||
$rootScope.staticFilesNeedUpdate = 0;
|
||||
$rootScope.addToast('success', "Regeneration in progress...");
|
||||
}, function (response) {
|
||||
$rootScope.addToast('success', 'An error occurs when saving settings:', response.data.errmsg);
|
||||
})
|
||||
$rootScope.staticRegenerationInProgress = true;
|
||||
$http.post("api/full-generation").then(function(response) {
|
||||
$rootScope.staticFilesNeedUpdate = 0;
|
||||
$rootScope.staticRegenerationInProgress = false;
|
||||
$rootScope.addToast('success', 'Regeneration ended');
|
||||
}, function (response) {
|
||||
$rootScope.staticRegenerationInProgress = false;
|
||||
$rootScope.addToast('error', 'An error occurs when saving settings:', response.data.errmsg);
|
||||
})
|
||||
}
|
||||
|
||||
|
|
@ -633,11 +634,6 @@ angular.module("FICApp")
|
|||
$scope.addToast('danger', 'An error occurs when saving settings:', response.data.errmsg);
|
||||
});
|
||||
}
|
||||
$scope.regenerate = function() {
|
||||
this.config.generation = (new Date()).toISOString();
|
||||
$rootScope.settings.generation = new Date(this.config.generation);
|
||||
$scope.saveSettings("Regeneration in progress...");
|
||||
}
|
||||
$scope.launchChallenge = function() {
|
||||
var ts = Date.now() - Date.now() % 60000;
|
||||
var d = new Date(ts + 120000);
|
||||
|
|
|
|||
|
|
@ -31,8 +31,6 @@
|
|||
</h2>
|
||||
</div>
|
||||
<div class="card-body pb-1">
|
||||
<input type="hidden" class="form-control form-control-sm" id="lastRegeneration" ng-model="config.generation">
|
||||
|
||||
<div class="form-group row">
|
||||
<label for="startTime" class="col-sm-3 col-form-label col-form-label-sm" ng-class="{'text-primary font-weight-bold': config.start != dist_config.start}">Début du challenge</label>
|
||||
<div class="col-sm-9">
|
||||
|
|
@ -382,7 +380,13 @@
|
|||
<div class="col-3 pt-2 d-flex flex-column justify-content-between">
|
||||
<div>
|
||||
<div class="d-flex flex-column">
|
||||
<button ng-click="regenerate()" class="btn btn-info my-1" type="button"><span class="glyphicon glyphicon-share" aria-hidden="true"></span> Regénérer les fichiers statiques</button>
|
||||
<button ng-click="regenerateStaticFiles()" class="btn btn-info my-1" type="button" ng-disabled="staticRegenerationInProgress">
|
||||
<span class="glyphicon glyphicon-share" aria-hidden="true" ng-show="!staticRegenerationInProgress"></span>
|
||||
<div class="spinner-border spinner-border-sm" role="status" ng-show="staticRegenerationInProgress">
|
||||
<span class="sr-only">Loading...</span>
|
||||
</div>
|
||||
Regénérer les fichiers statiques
|
||||
</button>
|
||||
<button ng-if="settings.wip" ng-click="switchToProd()" class="btn btn-danger my-1" type="button"><span class="glyphicon glyphicon-lock" aria-hidden="true"></span> Activer le mode challenge</button>
|
||||
</div>
|
||||
<hr>
|
||||
|
|
|
|||
|
|
@ -2,13 +2,14 @@ package sync
|
|||
|
||||
import (
|
||||
"encoding/json"
|
||||
"io"
|
||||
"log"
|
||||
"os"
|
||||
"sync"
|
||||
"time"
|
||||
|
||||
"srs.epita.fr/fic-server/admin/generation"
|
||||
"srs.epita.fr/fic-server/libfic"
|
||||
"srs.epita.fr/fic-server/settings"
|
||||
)
|
||||
|
||||
// DeepReportPath stores the path to the report generated during full recursive import.
|
||||
|
|
@ -132,8 +133,14 @@ func SyncDeep(i Importer) (errs SyncReport) {
|
|||
|
||||
EditDeepReport(&errs, true)
|
||||
|
||||
if err := settings.ForceRegeneration(); err != nil {
|
||||
resp, err := generation.FullGeneration()
|
||||
if err != nil {
|
||||
errs.Regeneration = append(errs.Regeneration, err.Error())
|
||||
} else {
|
||||
defer resp.Body.Close()
|
||||
|
||||
v, _ := io.ReadAll(resp.Body)
|
||||
errs.Regeneration = append(errs.Regeneration, string(v))
|
||||
}
|
||||
|
||||
DeepSyncProgress = 255
|
||||
|
|
|
|||
Reference in a new issue