2020-12-23 08:26:23 +00: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 09:05:56 +00:00
// Package dartsass integrates with the Dass Sass Embedded protocol to transpile
2020-12-23 08:26:23 +00:00
// SCSS/SASS.
package dartsass
import (
2022-03-15 07:54:56 +00:00
"fmt"
2020-12-23 08:26:23 +00:00
"io"
2022-03-17 16:22:34 +00:00
"strings"
2020-12-23 08:26:23 +00:00
2023-06-08 14:29:04 +00:00
godartsassv1 "github.com/bep/godartsass"
"github.com/bep/godartsass/v2"
2023-06-19 11:27:03 +00:00
"github.com/bep/logg"
2022-05-15 09:40:34 +00:00
"github.com/gohugoio/hugo/common/herrors"
2023-06-08 14:29:04 +00:00
"github.com/gohugoio/hugo/common/hugo"
2020-12-23 08:26:23 +00:00
"github.com/gohugoio/hugo/helpers"
2022-05-15 09:40:34 +00:00
"github.com/gohugoio/hugo/hugofs"
2020-12-23 08:26:23 +00:00
"github.com/gohugoio/hugo/hugolib/filesystems"
"github.com/gohugoio/hugo/resources"
"github.com/gohugoio/hugo/resources/resource"
"github.com/spf13/afero"
"github.com/mitchellh/mapstructure"
)
// used as part of the cache key.
const transformationName = "tocss-dart"
2022-05-15 09:40:34 +00: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 08:26:23 +00:00
func New ( fs * filesystems . SourceFilesystem , rs * resources . Spec ) ( * Client , error ) {
if ! Supports ( ) {
2021-12-12 11:11:11 +00:00
return & Client { dartSassNotAvailable : true } , nil
2020-12-23 08:26:23 +00:00
}
2021-12-12 11:11:11 +00:00
2023-06-08 14:29:04 +00:00
if hugo . DartSassBinaryName == "" {
return nil , fmt . Errorf ( "no Dart Sass binary found in $PATH" )
}
if err := rs . ExecHelper . Sec ( ) . CheckAllowedExec ( hugo . DartSassBinaryName ) ; err != nil {
2021-12-12 11:11:11 +00:00
return nil , err
}
2023-06-08 14:29:04 +00:00
var (
transpiler * godartsass . Transpiler
transpilerv1 * godartsassv1 . Transpiler
err error
2023-06-16 06:17:42 +00:00
infol = rs . Logger . InfoCommand ( "Dart Sass" )
warnl = rs . Logger . WarnCommand ( "Dart Sass" )
2023-06-08 14:29:04 +00:00
)
if hugo . IsDartSassV2 ( ) {
transpiler , err = godartsass . Start ( godartsass . Options {
DartSassEmbeddedFilename : hugo . DartSassBinaryName ,
LogEventHandler : func ( event godartsass . LogEvent ) {
message := strings . ReplaceAll ( event . Message , dartSassStdinPrefix , "" )
switch event . Type {
case godartsass . LogEventTypeDebug :
// Log as Info for now, we may adjust this if it gets too chatty.
2023-06-19 11:27:03 +00:00
infol . Log ( logg . String ( message ) )
2023-06-08 14:29:04 +00:00
default :
// The rest are either deprecations or @warn statements.
2023-06-19 11:27:03 +00:00
warnl . Log ( logg . String ( message ) )
2023-06-08 14:29:04 +00:00
}
} ,
} )
} else {
transpilerv1 , err = godartsassv1 . Start ( godartsassv1 . Options {
DartSassEmbeddedFilename : hugo . DartSassBinaryName ,
LogEventHandler : func ( event godartsassv1 . LogEvent ) {
message := strings . ReplaceAll ( event . Message , dartSassStdinPrefix , "" )
switch event . Type {
case godartsassv1 . LogEventTypeDebug :
// Log as Info for now, we may adjust this if it gets too chatty.
2023-06-19 11:27:03 +00:00
infol . Log ( logg . String ( message ) )
2023-06-08 14:29:04 +00:00
default :
// The rest are either deprecations or @warn statements.
2023-06-19 11:27:03 +00:00
warnl . Log ( logg . String ( message ) )
2023-06-08 14:29:04 +00:00
}
} ,
} )
}
2020-12-23 08:26:23 +00:00
if err != nil {
return nil , err
}
2023-06-08 14:29:04 +00:00
return & Client { sfs : fs , workFs : rs . BaseFs . Work , rs : rs , transpiler : transpiler , transpilerV1 : transpilerv1 } , nil
2020-12-23 08:26:23 +00:00
}
type Client struct {
2021-12-12 11:11:11 +00:00
dartSassNotAvailable bool
rs * resources . Spec
sfs * filesystems . SourceFilesystem
workFs afero . Fs
2023-06-08 14:29:04 +00:00
// One of these are non-nil.
transpiler * godartsass . Transpiler
transpilerV1 * godartsassv1 . Transpiler
2020-12-23 08:26:23 +00:00
}
2022-03-17 21:03:27 +00:00
func ( c * Client ) ToCSS ( res resources . ResourceTransformer , args map [ string ] any ) ( resource . Resource , error ) {
2021-12-12 11:11:11 +00:00
if c . dartSassNotAvailable {
2020-12-23 08:26:23 +00:00
return res . Transform ( resources . NewFeatureNotAvailableTransformer ( transformationName , args ) )
}
return res . Transform ( & transform { c : c , optsm : args } )
}
func ( c * Client ) Close ( ) error {
2023-06-08 14:29:04 +00:00
if c . transpilerV1 != nil {
return c . transpilerV1 . Close ( )
}
if c . transpiler != nil {
return c . transpiler . Close ( )
2020-12-23 08:26:23 +00:00
}
2023-06-08 14:29:04 +00:00
return nil
2020-12-23 08:26:23 +00:00
}
func ( c * Client ) toCSS ( args godartsass . Args , src io . Reader ) ( godartsass . Result , error ) {
in := helpers . ReaderToString ( src )
2022-12-19 17:49:02 +00:00
2020-12-23 08:26:23 +00:00
args . Source = in
2023-06-08 14:29:04 +00:00
var (
err error
res godartsass . Result
)
if c . transpilerV1 != nil {
var resv1 godartsassv1 . Result
var argsv1 godartsassv1 . Args
mapstructure . Decode ( args , & argsv1 )
if args . ImportResolver != nil {
argsv1 . ImportResolver = importResolverV1 { args . ImportResolver }
}
resv1 , err = c . transpilerV1 . Execute ( argsv1 )
if err == nil {
mapstructure . Decode ( resv1 , & res )
}
} else {
res , err = c . transpiler . Execute ( args )
}
2020-12-23 08:26:23 +00:00
if err != nil {
2022-03-15 07:54:56 +00:00
if err . Error ( ) == "unexpected EOF" {
2023-06-08 14:29:04 +00:00
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." , hugo . DartSassBinaryName )
2022-03-15 07:54:56 +00:00
}
2022-05-15 09:40:34 +00:00
return res , herrors . NewFileErrorFromFileInErr ( err , hugofs . Os , herrors . OffsetMatcher )
2020-12-23 08:26:23 +00: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 08:26:38 +00:00
// If enabled, sources will be embedded in the generated source map.
SourceMapIncludeSources bool
2022-12-19 17:49:02 +00:00
// Vars will be available in 'hugo:vars', e.g:
// @use "hugo:vars";
// $color: vars.$color;
2023-02-21 17:32:09 +00:00
Vars map [ string ] any
2020-12-23 08:26:23 +00:00
}
2022-03-17 21:03:27 +00:00
func decodeOptions ( m map [ string ] any ) ( opts Options , err error ) {
2020-12-23 08:26:23 +00:00
if m == nil {
return
}
err = mapstructure . WeakDecode ( m , & opts )
if opts . TargetPath != "" {
opts . TargetPath = helpers . ToSlashTrimLeading ( opts . TargetPath )
}
return
}