2020-12-23 03:26:23 -05:00
// Copyright 2020 The Hugo Authors. All rights reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
2023-05-18 05:05:56 -04:00
// Package dartsass integrates with the Dass Sass Embedded protocol to transpile
2020-12-23 03:26:23 -05:00
// SCSS/SASS.
package dartsass
import (
2022-03-15 03:54:56 -04:00
"fmt"
2020-12-23 03:26:23 -05:00
"io"
2022-03-17 12:22:34 -04:00
"strings"
2020-12-23 03:26:23 -05:00
2022-05-15 05:40:34 -04:00
"github.com/gohugoio/hugo/common/herrors"
2020-12-23 03:26:23 -05:00
"github.com/gohugoio/hugo/helpers"
2022-05-15 05:40:34 -04:00
"github.com/gohugoio/hugo/hugofs"
2020-12-23 03:26:23 -05:00
"github.com/gohugoio/hugo/hugolib/filesystems"
"github.com/gohugoio/hugo/resources"
"github.com/gohugoio/hugo/resources/resource"
"github.com/spf13/afero"
"github.com/bep/godartsass"
"github.com/mitchellh/mapstructure"
)
// used as part of the cache key.
const transformationName = "tocss-dart"
2022-05-15 05:40:34 -04:00
// See https://github.com/sass/dart-sass-embedded/issues/24
// Note: This prefix must be all lower case.
const dartSassStdinPrefix = "hugostdin:"
2020-12-23 03:26:23 -05:00
func New ( fs * filesystems . SourceFilesystem , rs * resources . Spec ) ( * Client , error ) {
if ! Supports ( ) {
2021-12-12 06:11:11 -05:00
return & Client { dartSassNotAvailable : true } , nil
2020-12-23 03:26:23 -05:00
}
2021-12-12 06:11:11 -05:00
if err := rs . ExecHelper . Sec ( ) . CheckAllowedExec ( dartSassEmbeddedBinaryName ) ; err != nil {
return nil , err
}
2022-03-17 12:22:34 -04:00
transpiler , err := godartsass . Start ( godartsass . Options {
LogEventHandler : func ( event godartsass . LogEvent ) {
2022-05-15 05:40:34 -04:00
message := strings . ReplaceAll ( event . Message , dartSassStdinPrefix , "" )
2022-03-17 12:22:34 -04:00
switch event . Type {
case godartsass . LogEventTypeDebug :
// Log as Info for now, we may adjust this if it gets too chatty.
rs . Logger . Infof ( "Dart Sass: %s" , message )
default :
// The rest are either deprecations or @warn statements.
rs . Logger . Warnf ( "Dart Sass: %s" , message )
}
} ,
} )
2020-12-23 03:26:23 -05:00
if err != nil {
return nil , err
}
return & Client { sfs : fs , workFs : rs . BaseFs . Work , rs : rs , transpiler : transpiler } , nil
}
type Client struct {
2021-12-12 06:11:11 -05:00
dartSassNotAvailable bool
rs * resources . Spec
sfs * filesystems . SourceFilesystem
workFs afero . Fs
transpiler * godartsass . Transpiler
2020-12-23 03:26:23 -05:00
}
2022-03-17 17:03:27 -04:00
func ( c * Client ) ToCSS ( res resources . ResourceTransformer , args map [ string ] any ) ( resource . Resource , error ) {
2021-12-12 06:11:11 -05:00
if c . dartSassNotAvailable {
2020-12-23 03:26:23 -05:00
return res . Transform ( resources . NewFeatureNotAvailableTransformer ( transformationName , args ) )
}
return res . Transform ( & transform { c : c , optsm : args } )
}
func ( c * Client ) Close ( ) error {
if c . transpiler == nil {
return nil
}
return c . transpiler . Close ( )
}
func ( c * Client ) toCSS ( args godartsass . Args , src io . Reader ) ( godartsass . Result , error ) {
var res godartsass . Result
in := helpers . ReaderToString ( src )
2022-12-19 12:49:02 -05:00
2020-12-23 03:26:23 -05:00
args . Source = in
res , err := c . transpiler . Execute ( args )
if err != nil {
2022-03-15 03:54:56 -04:00
if err . Error ( ) == "unexpected EOF" {
return res , fmt . Errorf ( "got unexpected EOF when executing %q. The user running hugo must have read and execute permissions on this program. With execute permissions only, this error is thrown." , dartSassEmbeddedBinaryName )
}
2022-05-15 05:40:34 -04:00
return res , herrors . NewFileErrorFromFileInErr ( err , hugofs . Os , herrors . OffsetMatcher )
2020-12-23 03:26:23 -05:00
}
return res , err
}
type Options struct {
// Hugo, will by default, just replace the extension of the source
// to .css, e.g. "scss/main.scss" becomes "scss/main.css". You can
// control this by setting this, e.g. "styles/main.css" will create
// a Resource with that as a base for RelPermalink etc.
TargetPath string
// Hugo automatically adds the entry directories (where the main.scss lives)
// for project and themes to the list of include paths sent to LibSASS.
// Any paths set in this setting will be appended. Note that these will be
// treated as relative to the working dir, i.e. no include paths outside the
// project/themes.
IncludePaths [ ] string
// Default is nested.
// One of nested, expanded, compact, compressed.
OutputStyle string
// When enabled, Hugo will generate a source map.
EnableSourceMap bool
2022-12-02 03:26:38 -05:00
// If enabled, sources will be embedded in the generated source map.
SourceMapIncludeSources bool
2022-12-19 12:49:02 -05:00
// Vars will be available in 'hugo:vars', e.g:
// @use "hugo:vars";
// $color: vars.$color;
2023-02-21 12:32:09 -05:00
Vars map [ string ] any
2020-12-23 03:26:23 -05:00
}
2022-03-17 17:03:27 -04:00
func decodeOptions ( m map [ string ] any ) ( opts Options , err error ) {
2020-12-23 03:26:23 -05:00
if m == nil {
return
}
err = mapstructure . WeakDecode ( m , & opts )
if opts . TargetPath != "" {
opts . TargetPath = helpers . ToSlashTrimLeading ( opts . TargetPath )
}
return
}