| 
									
										
										
										
											2023-05-09 19:19:48 +02:00
										 |  |  | // Copyright The OpenTelemetry Authors | 
					
						
							| 
									
										
										
										
											2024-04-11 11:46:18 +02:00
										 |  |  | // SPDX-License-Identifier: Apache-2.0 | 
					
						
							| 
									
										
										
										
											2023-05-09 19:19:48 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | package resource // import "go.opentelemetry.io/otel/sdk/resource" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"context" | 
					
						
							|  |  |  | 	"errors" | 
					
						
							|  |  |  | 	"fmt" | 
					
						
							| 
									
										
										
										
											2023-06-05 10:15:05 +02:00
										 |  |  | 	"strings" | 
					
						
							| 
									
										
										
										
											2023-05-09 19:19:48 +02:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-13 11:08:02 +01:00
										 |  |  | // ErrPartialResource is returned by a detector when complete source | 
					
						
							|  |  |  | // information for a Resource is unavailable or the source information | 
					
						
							|  |  |  | // contains invalid values that are omitted from the returned Resource. | 
					
						
							|  |  |  | var ErrPartialResource = errors.New("partial resource") | 
					
						
							| 
									
										
										
										
											2023-05-09 19:19:48 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | // Detector detects OpenTelemetry resource information. | 
					
						
							|  |  |  | type Detector interface { | 
					
						
							|  |  |  | 	// DO NOT CHANGE: any modification will not be backwards compatible and | 
					
						
							|  |  |  | 	// must never be done outside of a new major release. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Detect returns an initialized Resource based on gathered information. | 
					
						
							|  |  |  | 	// If the source information to construct a Resource contains invalid | 
					
						
							|  |  |  | 	// values, a Resource is returned with the valid parts of the source | 
					
						
							|  |  |  | 	// information used for initialization along with an appropriately | 
					
						
							|  |  |  | 	// wrapped ErrPartialResource error. | 
					
						
							|  |  |  | 	Detect(ctx context.Context) (*Resource, error) | 
					
						
							|  |  |  | 	// DO NOT CHANGE: any modification will not be backwards compatible and | 
					
						
							|  |  |  | 	// must never be done outside of a new major release. | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-11 15:34:34 +01:00
										 |  |  | // Detect returns a new [Resource] merged from all the Resources each of the | 
					
						
							|  |  |  | // detectors produces. Each of the detectors are called sequentially, in the | 
					
						
							|  |  |  | // order they are passed, merging the produced resource into the previous. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // This may return a partial Resource along with an error containing | 
					
						
							|  |  |  | // [ErrPartialResource] if that error is returned from a detector. It may also | 
					
						
							|  |  |  | // return a merge-conflicting Resource along with an error containing | 
					
						
							|  |  |  | // [ErrSchemaURLConflict] if merging Resources from different detectors results | 
					
						
							|  |  |  | // in a schema URL conflict. It is up to the caller to determine if this | 
					
						
							|  |  |  | // returned Resource should be used or not. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // If one of the detectors returns an error that is not [ErrPartialResource], | 
					
						
							|  |  |  | // the resource produced by the detector will not be merged and the returned | 
					
						
							|  |  |  | // error will wrap that detector's error. | 
					
						
							| 
									
										
										
										
											2023-05-09 19:19:48 +02:00
										 |  |  | func Detect(ctx context.Context, detectors ...Detector) (*Resource, error) { | 
					
						
							| 
									
										
										
										
											2023-06-05 10:15:05 +02:00
										 |  |  | 	r := new(Resource) | 
					
						
							|  |  |  | 	return r, detect(ctx, r, detectors) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // detect runs all detectors using ctx and merges the result into res. This | 
					
						
							|  |  |  | // assumes res is allocated and not nil, it will panic otherwise. | 
					
						
							| 
									
										
										
										
											2024-03-11 15:34:34 +01:00
										 |  |  | // | 
					
						
							|  |  |  | // If the detectors or merging resources produces any errors (i.e. | 
					
						
							|  |  |  | // [ErrPartialResource] [ErrSchemaURLConflict]), a single error wrapping all of | 
					
						
							|  |  |  | // these errors will be returned. Otherwise, nil is returned. | 
					
						
							| 
									
										
										
										
											2023-06-05 10:15:05 +02:00
										 |  |  | func detect(ctx context.Context, res *Resource, detectors []Detector) error { | 
					
						
							|  |  |  | 	var ( | 
					
						
							|  |  |  | 		r    *Resource | 
					
						
							|  |  |  | 		errs detectErrs | 
					
						
							|  |  |  | 		err  error | 
					
						
							|  |  |  | 	) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-09 19:19:48 +02:00
										 |  |  | 	for _, detector := range detectors { | 
					
						
							|  |  |  | 		if detector == nil { | 
					
						
							|  |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2023-06-05 10:15:05 +02:00
										 |  |  | 		r, err = detector.Detect(ctx) | 
					
						
							| 
									
										
										
										
											2023-05-09 19:19:48 +02:00
										 |  |  | 		if err != nil { | 
					
						
							| 
									
										
										
										
											2023-06-05 10:15:05 +02:00
										 |  |  | 			errs = append(errs, err) | 
					
						
							| 
									
										
										
										
											2023-05-09 19:19:48 +02:00
										 |  |  | 			if !errors.Is(err, ErrPartialResource) { | 
					
						
							|  |  |  | 				continue | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2023-06-05 10:15:05 +02:00
										 |  |  | 		r, err = Merge(res, r) | 
					
						
							| 
									
										
										
										
											2023-05-09 19:19:48 +02:00
										 |  |  | 		if err != nil { | 
					
						
							| 
									
										
										
										
											2023-06-05 10:15:05 +02:00
										 |  |  | 			errs = append(errs, err) | 
					
						
							| 
									
										
										
										
											2023-05-09 19:19:48 +02:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2023-06-05 10:15:05 +02:00
										 |  |  | 		*res = *r | 
					
						
							| 
									
										
										
										
											2023-05-09 19:19:48 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-05 10:15:05 +02:00
										 |  |  | 	if len(errs) == 0 { | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-03-11 15:34:34 +01:00
										 |  |  | 	if errors.Is(errs, ErrSchemaURLConflict) { | 
					
						
							|  |  |  | 		// If there has been a merge conflict, ensure the resource has no | 
					
						
							|  |  |  | 		// schema URL. | 
					
						
							|  |  |  | 		res.schemaURL = "" | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2023-06-05 10:15:05 +02:00
										 |  |  | 	return errs | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type detectErrs []error | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (e detectErrs) Error() string { | 
					
						
							|  |  |  | 	errStr := make([]string, len(e)) | 
					
						
							|  |  |  | 	for i, err := range e { | 
					
						
							|  |  |  | 		errStr[i] = fmt.Sprintf("* %s", err) | 
					
						
							| 
									
										
										
										
											2023-05-09 19:19:48 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2023-06-05 10:15:05 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	format := "%d errors occurred detecting resource:\n\t%s" | 
					
						
							|  |  |  | 	return fmt.Sprintf(format, len(e), strings.Join(errStr, "\n\t")) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (e detectErrs) Unwrap() error { | 
					
						
							|  |  |  | 	switch len(e) { | 
					
						
							|  |  |  | 	case 0: | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	case 1: | 
					
						
							|  |  |  | 		return e[0] | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return e[1:] | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (e detectErrs) Is(target error) bool { | 
					
						
							|  |  |  | 	return len(e) != 0 && errors.Is(e[0], target) | 
					
						
							| 
									
										
										
										
											2023-05-09 19:19:48 +02:00
										 |  |  | } |