Custom Fragments
This guide expands on the how to customize the Lens SDK responses by using custom fragments.
In the Getting Started Guide, we introduced the concept of custom GraphQL fragments. This guide delves into common use cases and potential pitfalls to consider when defining custom fragments.
Fragment Definitions
A GraphQL fragment, a Fragment Definition to be precise, is a reusable piece of a GraphQL query that defines a subset of fields to be retrieved from the server.
fragment FragmentName on TypeName { field1 field2 field3}
where:
FragmentName is the name of the fragment
TypeName is the type of the node the fragment represents
field1, field2, field3 are the fields to be retrieved
Fragments can include other fragments, known as fragment spreads. This allows you to build complex queries by combining smaller, reusable components.
fragment FragmentName on TypeName { field1 field2 field3 { ...OtherFragment }}
Common Use Cases
Use the graphql function from the Lens SDK to define custom fragments by passing a query string and an array of imported fragments.
Example
import { UsernameFragment, graphql } from "@lens-protocol/client";
export const AccountFragment = graphql( ` fragment Account on Account { __typename username { ...Username } address } `, [UsernameFragment]);
The graphql function is a bespoke instance of the gql.tada library that provides a more ergonomic way to define GraphQL fragments.
If a Lens SDK fragment meets your requirements, it is recommended to reuse it to prevent data duplication.
In most cases, Lens SDK fragments are named after the node they represent (e.g., Account, Post, Group, etc.). Any deviations from this naming convention will be explicitly noted in the documentation.
It is essential to name your fragment consistently with the existing Lens SDK fragments. This allows the Lens SDK to seamlessly merge your custom fragment with the predefined ones.
Post Fields
The PostFields fragment is used to retrieve post fields for the Post and the ReferencedPost fragments.
fragments/posts.ts
import { graphql, PostMetadataFragment } from "@lens-protocol/client";
export const PostFieldsFragment = graphql( ` fragment PostFields on Post { slug timestamp metadata { ...PostMetadata } } `, [PostMetadataFragment]);
The ReferencedPost fragment selects specific fields from a Post, excluding links to other posts. This prevents circular references when retrieving posts.
Post Metadata
The PostMetadata fragment represents a union of all standard Post Metadata types. Use it to retrieve metadata for the specific post types you plan to support in your application (e.g., video, image, text-only, audio, story, embed) while excluding unnecessary types.
fragments/posts.ts
import { ArticleMetadataFragment, AudioMetadataFragment, TextOnlyMetadataFragment, ImageMetadataFragment, VideoMetadataFragment, graphql,} from "@lens-protocol/client";
export const PostMetadataFragment = graphql( ` fragment PostMetadata on PostMetadata { __typename ... on ArticleMetadata { ...ArticleMetadata } ... on AudioMetadata { ...AudioMetadata } ... on TextOnlyMetadata { ...TextOnlyMetadata } ... on ImageMetadata { ...ImageMetadata } ... on VideoMetadata { ...VideoMetadata } } `, [ ArticleMetadataFragment, AudioMetadataFragment, TextOnlyMetadataFragment, ImageMetadataFragment, VideoMetadataFragment, ]);
Media Images
Use custom MediaImage fragment to optimize the retrieval of Post images. Use the parametrized item field to request images of different sizes and formats.
fragments/images.ts
import { graphql } from "@lens-protocol/client";
export const MediaImageFragment = graphql( ` fragment MediaImage on MediaImage { __typename
full: item
large: item(request: { preferTransform: { widthBased: { witdh: 2048 } } })
thumbnail: item( request: { preferTransform: { fixedSize: { height: 128, witdh: 128 } } } )
altTag license type } `);
See the Querying Metadata Media section for more information on this.
Account Metadata
The AccountMetadata fragment is used to retrieve metadata for the Account type. Use the picture and coverPicture fields to retrieve profile and cover images.
fragments/accounts.ts
import { graphql, MediaImageFragment } from "@lens-protocol/client";
export const AccountMetadataFragment = graphql( ` fragment AccountMetadata on AccountMetadata { name bio
thumbnail: picture( request: { preferTransform: { fixedSize: { height: 128, witdh: 128 } } } ) picture
coverPicture wideBackground: coverPicture( request: { preferTransform: { widthBased: { witdh: 2048 } } } ) } `, [MediaImageFragment]);
See the Querying Metadata Media section for more information on this.
Troubleshooting
Duplicated Fragments
When instantiating the PublicClient with fragments, you might encounter the following error:
InvariantError: Duplicate fragment detected.A fragment named "<Name>" has already been provided,either directly or as part of another fragment document.
This means that there are duplicated fragments as part of the fragment documents you provided.
Typically, this occurs when you define a fragment that includes a sub-fragment and then include both the parent fragment and the sub-fragment in the list of fragments passed to the PublicClient factory function.
For example, if you defined PostFieldsFragment with a bespoke PostMetadataFragment like this:
Then, omit the PostMetadataFragment from the list of fragments passed to the PublicClient factory function:
const client = PublicClient.create({ environment: mainnet,- fragments: [PostFieldsFragment, PostMetadataFragment],+ fragments: [PostFieldsFragment],});