mirror of
				https://github.com/mermaid-js/mermaid.git
				synced 2025-10-25 00:44:10 +02:00 
			
		
		
		
	Compare commits
	
		
			25 Commits
		
	
	
		
			6889-fix-e
			...
			fix/mindma
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | b136acdc67 | ||
|   | bba5e5938e | ||
|   | fed8a523a4 | ||
|   | 33b4946e21 | ||
|   | 3d768f3adf | ||
|   | 76e17ffd20 | ||
|   | 60f633101c | ||
|   | 18f51eb14e | ||
|   | 2bb57bf7d2 | ||
|   | a6276daffd | ||
|   | 7def6eecbf | ||
|   | ac411a7d7e | ||
|   | 6fecb985e8 | ||
|   | 69b338d8af | ||
|   | fa15ce8502 | ||
|   | 6d0650918f | ||
|   | 1a9d45abf0 | ||
|   | bbb93b263d | ||
|   | 4240340a18 | ||
|   | ca10a259fa | ||
|   | 0ed9c65572 | ||
|   | 56cc12690f | ||
|   | e6fb4a84da | ||
|   | 32723b2de1 | ||
|   | 18703782ee | 
| @@ -1,5 +0,0 @@ | ||||
| --- | ||||
| 'mermaid': patch | ||||
| --- | ||||
|  | ||||
| fix: Prevent HTML tags from being escaped in sandbox label rendering | ||||
							
								
								
									
										5
									
								
								.changeset/brave-memes-flash.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								.changeset/brave-memes-flash.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | ||||
| --- | ||||
| 'mermaid': patch | ||||
| --- | ||||
|  | ||||
| fix: Support edge animation in hand drawn look | ||||
							
								
								
									
										5
									
								
								.changeset/busy-mirrors-try.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								.changeset/busy-mirrors-try.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | ||||
| --- | ||||
| 'mermaid': patch | ||||
| --- | ||||
|  | ||||
| fix: Resolved parsing error where direction TD was not recognized within subgraphs | ||||
							
								
								
									
										5
									
								
								.changeset/chilly-words-march.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								.changeset/chilly-words-march.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | ||||
| --- | ||||
| 'mermaid': patch | ||||
| --- | ||||
|  | ||||
| fix: Correct viewBox casing and make SVGs responsive | ||||
							
								
								
									
										5
									
								
								.changeset/curly-apes-prove.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								.changeset/curly-apes-prove.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | ||||
| --- | ||||
| 'mermaid': patch | ||||
| --- | ||||
|  | ||||
| fix: Improve participant parsing and prevent recursive loops on invalid syntax | ||||
							
								
								
									
										5
									
								
								.changeset/wide-lines-trade.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								.changeset/wide-lines-trade.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | ||||
| --- | ||||
| 'mermaid': patch | ||||
| --- | ||||
|  | ||||
| fix: Mindmap rendering issue when the number of Level 2 nodes exceeds 11 | ||||
| @@ -6,6 +6,7 @@ interface CypressConfig { | ||||
|   listUrl?: boolean; | ||||
|   listId?: string; | ||||
|   name?: string; | ||||
|   screenshot?: boolean; | ||||
| } | ||||
| type CypressMermaidConfig = MermaidConfig & CypressConfig; | ||||
|  | ||||
| @@ -90,7 +91,7 @@ export const renderGraph = ( | ||||
|  | ||||
| export const openURLAndVerifyRendering = ( | ||||
|   url: string, | ||||
|   options: CypressMermaidConfig, | ||||
|   { screenshot = true, ...options }: CypressMermaidConfig, | ||||
|   validation?: any | ||||
| ): void => { | ||||
|   const name: string = (options.name ?? cy.state('runnable').fullTitle()).replace(/\s+/g, '-'); | ||||
| @@ -98,12 +99,15 @@ export const openURLAndVerifyRendering = ( | ||||
|   cy.visit(url); | ||||
|   cy.window().should('have.property', 'rendered', true); | ||||
|   cy.get('svg').should('be.visible'); | ||||
|   cy.get('svg').should('not.have.attr', 'viewbox'); | ||||
|  | ||||
|   if (validation) { | ||||
|     cy.get('svg').should(validation); | ||||
|   } | ||||
|  | ||||
|   verifyScreenshot(name); | ||||
|   if (screenshot) { | ||||
|     verifyScreenshot(name); | ||||
|   } | ||||
| }; | ||||
|  | ||||
| export const verifyScreenshot = (name: string): void => { | ||||
|   | ||||
| @@ -1029,4 +1029,19 @@ graph TD | ||||
|       } | ||||
|     ); | ||||
|   }); | ||||
|  | ||||
|   it('FDH49: should add edge animation', () => { | ||||
|     renderGraph( | ||||
|       ` | ||||
|       flowchart TD | ||||
|           A(["Start"]) L_A_B_0@--> B{"Decision"} | ||||
|           B --> C["Option A"] & D["Option B"] | ||||
|           style C stroke-width:4px,stroke-dasharray: 5 | ||||
|           L_A_B_0@{ animation: slow }  | ||||
|           L_B_D_0@{ animation: fast }`, | ||||
|       { look: 'handDrawn', screenshot: false } | ||||
|     ); | ||||
|     cy.get('path#L_A_B_0').should('have.class', 'edge-animation-slow'); | ||||
|     cy.get('path#L_B_D_0').should('have.class', 'edge-animation-fast'); | ||||
|   }); | ||||
| }); | ||||
|   | ||||
| @@ -774,6 +774,21 @@ describe('Graph', () => { | ||||
|       expect(svg).to.not.have.attr('style'); | ||||
|     }); | ||||
|   }); | ||||
|   it('40: should add edge animation', () => { | ||||
|     renderGraph( | ||||
|       ` | ||||
|       flowchart TD | ||||
|           A(["Start"]) L_A_B_0@--> B{"Decision"} | ||||
|           B --> C["Option A"] & D["Option B"] | ||||
|           style C stroke-width:4px,stroke-dasharray: 5 | ||||
|           L_A_B_0@{ animation: slow }  | ||||
|           L_B_D_0@{ animation: fast }`, | ||||
|       { screenshot: false } | ||||
|     ); | ||||
|     // Verify animation classes are applied to both edges | ||||
|     cy.get('path#L_A_B_0').should('have.class', 'edge-animation-slow'); | ||||
|     cy.get('path#L_B_D_0').should('have.class', 'edge-animation-fast'); | ||||
|   }); | ||||
|   it('58: handle styling with style expressions', () => { | ||||
|     imgSnapshotTest( | ||||
|       ` | ||||
| @@ -973,4 +988,19 @@ graph TD | ||||
|       } | ||||
|     ); | ||||
|   }); | ||||
|  | ||||
|   it('70: should render a subgraph with direction TD', () => { | ||||
|     imgSnapshotTest( | ||||
|       ` | ||||
|       flowchart LR | ||||
|         subgraph A | ||||
|           direction TD | ||||
|           a --> b | ||||
|         end | ||||
|       `, | ||||
|       { | ||||
|         fontFamily: 'courier', | ||||
|       } | ||||
|     ); | ||||
|   }); | ||||
| }); | ||||
|   | ||||
| @@ -21,7 +21,7 @@ title: Animal example | ||||
| classDiagram | ||||
|     note "From Duck till Zebra" | ||||
|     Animal <|-- Duck | ||||
|     note for Duck "can fly\ncan swim\ncan dive\ncan help in debugging" | ||||
|     note for Duck "can fly<br>can swim<br>can dive<br>can help in debugging" | ||||
|     Animal <|-- Fish | ||||
|     Animal <|-- Zebra | ||||
|     Animal : +int age | ||||
| @@ -50,7 +50,7 @@ title: Animal example | ||||
| classDiagram | ||||
|     note "From Duck till Zebra" | ||||
|     Animal <|-- Duck | ||||
|     note for Duck "can fly\ncan swim\ncan dive\ncan help in debugging" | ||||
|     note for Duck "can fly<br>can swim<br>can dive<br>can help in debugging" | ||||
|     Animal <|-- Fish | ||||
|     Animal <|-- Zebra | ||||
|     Animal : +int age | ||||
|   | ||||
| @@ -70,31 +70,6 @@ describe('Sanitize text', () => { | ||||
|     }); | ||||
|     expect(result).not.toContain('javascript:alert(1)'); | ||||
|   }); | ||||
|  | ||||
|   it('should allow HTML tags in sandbox mode', () => { | ||||
|     const htmlStr = '<p>This is a <strong>bold</strong> text</p>'; | ||||
|     const result = sanitizeText(htmlStr, { | ||||
|       securityLevel: 'sandbox', | ||||
|       flowchart: { htmlLabels: true }, | ||||
|     }); | ||||
|     expect(result).toContain('<p>'); | ||||
|     expect(result).toContain('<strong>'); | ||||
|     expect(result).toContain('</strong>'); | ||||
|     expect(result).toContain('</p>'); | ||||
|   }); | ||||
|  | ||||
|   it('should remove script tags in sandbox mode', () => { | ||||
|     const maliciousStr = '<p>Hello <script>alert(1)</script> world</p>'; | ||||
|     const result = sanitizeText(maliciousStr, { | ||||
|       securityLevel: 'sandbox', | ||||
|       flowchart: { htmlLabels: true }, | ||||
|     }); | ||||
|     expect(result).not.toContain('<script>'); | ||||
|     expect(result).not.toContain('alert(1)'); | ||||
|     expect(result).toContain('<p>'); | ||||
|     expect(result).toContain('Hello'); | ||||
|     expect(result).toContain('world'); | ||||
|   }); | ||||
| }); | ||||
|  | ||||
| describe('generic parser', () => { | ||||
|   | ||||
| @@ -66,7 +66,7 @@ export const removeScript = (txt: string): string => { | ||||
| const sanitizeMore = (text: string, config: MermaidConfig) => { | ||||
|   if (config.flowchart?.htmlLabels !== false) { | ||||
|     const level = config.securityLevel; | ||||
|     if (level === 'antiscript' || level === 'strict' || level === 'sandbox') { | ||||
|     if (level === 'antiscript' || level === 'strict') { | ||||
|       text = removeScript(text); | ||||
|     } else if (level !== 'loose') { | ||||
|       text = breakToPlaceholder(text); | ||||
|   | ||||
| @@ -140,6 +140,7 @@ that id. | ||||
| .*direction\s+BT[^\n]*       return 'direction_bt'; | ||||
| .*direction\s+RL[^\n]*       return 'direction_rl'; | ||||
| .*direction\s+LR[^\n]*       return 'direction_lr'; | ||||
| .*direction\s+TD[^\n]*       return 'direction_td'; | ||||
|  | ||||
| [^\s\"]+\@(?=[^\{\"])               { return 'LINK_ID'; } | ||||
| [0-9]+                       return 'NUM'; | ||||
| @@ -626,6 +627,8 @@ direction | ||||
|     { $$={stmt:'dir', value:'RL'};} | ||||
|     | direction_lr | ||||
|     { $$={stmt:'dir', value:'LR'};} | ||||
|     | direction_td | ||||
|     { $$={stmt:'dir', value:'TD'};} | ||||
|     ; | ||||
|  | ||||
| %% | ||||
|   | ||||
| @@ -309,4 +309,21 @@ describe('when parsing subgraphs', function () { | ||||
|     expect(subgraphA.nodes).toContain('a'); | ||||
|     expect(subgraphA.nodes).not.toContain('c'); | ||||
|   }); | ||||
|   it('should correctly parse direction TD inside a subgraph', function () { | ||||
|     const res = flow.parser.parse(` | ||||
|       graph LR | ||||
|         subgraph WithTD | ||||
|           direction TD | ||||
|           A1 --> A2 | ||||
|         end | ||||
|     `); | ||||
|  | ||||
|     const subgraphs = flow.parser.yy.getSubGraphs(); | ||||
|     expect(subgraphs.length).toBe(1); | ||||
|     const subgraph = subgraphs[0]; | ||||
|  | ||||
|     expect(subgraph.dir).toBe('TD'); | ||||
|     expect(subgraph.nodes).toContain('A1'); | ||||
|     expect(subgraph.nodes).toContain('A2'); | ||||
|   }); | ||||
| }); | ||||
|   | ||||
| @@ -293,5 +293,37 @@ describe('MindmapDb getData function', () => { | ||||
|       expect(edgeA1_aaa.section).toBe(1); | ||||
|       expect(edgeA_a2.section).toBe(2); | ||||
|     }); | ||||
|  | ||||
|     it('should wrap section numbers when there are more than 11 level 2 nodes', () => { | ||||
|       db.addNode(0, 'root', 'Example', 0); | ||||
|  | ||||
|       for (let i = 1; i <= 15; i++) { | ||||
|         db.addNode(1, `child${i}`, `${i}`, 0); | ||||
|       } | ||||
|  | ||||
|       const result = db.getData(); | ||||
|  | ||||
|       expect(result.nodes).toHaveLength(16); | ||||
|  | ||||
|       const child1 = result.nodes.find((n) => n.label === '1') as MindmapLayoutNode; | ||||
|       const child11 = result.nodes.find((n) => n.label === '11') as MindmapLayoutNode; | ||||
|       const child12 = result.nodes.find((n) => n.label === '12') as MindmapLayoutNode; | ||||
|       const child13 = result.nodes.find((n) => n.label === '13') as MindmapLayoutNode; | ||||
|       const child14 = result.nodes.find((n) => n.label === '14') as MindmapLayoutNode; | ||||
|       const child15 = result.nodes.find((n) => n.label === '15') as MindmapLayoutNode; | ||||
|  | ||||
|       expect(child1.section).toBe(0); | ||||
|       expect(child11.section).toBe(10); | ||||
|  | ||||
|       expect(child12.section).toBe(0); | ||||
|       expect(child13.section).toBe(1); | ||||
|       expect(child14.section).toBe(2); | ||||
|       expect(child15.section).toBe(3); | ||||
|  | ||||
|       expect(child12.cssClasses).toBe('mindmap-node section-0'); | ||||
|       expect(child13.cssClasses).toBe('mindmap-node section-1'); | ||||
|       expect(child14.cssClasses).toBe('mindmap-node section-2'); | ||||
|       expect(child15.cssClasses).toBe('mindmap-node section-3'); | ||||
|     }); | ||||
|   }); | ||||
| }); | ||||
|   | ||||
| @@ -203,7 +203,7 @@ export class MindmapDB { | ||||
|     // For other nodes, inherit parent's section number | ||||
|     if (node.children) { | ||||
|       for (const [index, child] of node.children.entries()) { | ||||
|         const childSectionNumber = node.level === 0 ? index : sectionNumber; | ||||
|         const childSectionNumber = node.level === 0 ? index % 11 : sectionNumber; | ||||
|         this.assignSections(child, childSectionNumber); | ||||
|       } | ||||
|     } | ||||
|   | ||||
| @@ -16,7 +16,7 @@ const draw: DrawDefinition = (_text, id, _version, diagram: Diagram) => { | ||||
|   const svgWidth = bitWidth * bitsPerRow + 2; | ||||
|   const svg: SVG = selectSvgElement(id); | ||||
|  | ||||
|   svg.attr('viewbox', `0 0 ${svgWidth} ${svgHeight}`); | ||||
|   svg.attr('viewBox', `0 0 ${svgWidth} ${svgHeight}`); | ||||
|   configureSvgSize(svg, svgHeight, svgWidth, config.useMaxWidth); | ||||
|  | ||||
|   for (const [word, packet] of words.entries()) { | ||||
|   | ||||
| @@ -2,6 +2,7 @@ import type { Diagram } from '../../Diagram.js'; | ||||
| import type { RadarDiagramConfig } from '../../config.type.js'; | ||||
| import type { DiagramRenderer, DrawDefinition, SVG, SVGGroup } from '../../diagram-api/types.js'; | ||||
| import { selectSvgElement } from '../../rendering-util/selectSvgElement.js'; | ||||
| import { configureSvgSize } from '../../setupGraphViewbox.js'; | ||||
| import type { RadarDB, RadarAxis, RadarCurve } from './types.js'; | ||||
|  | ||||
| const draw: DrawDefinition = (_text, id, _version, diagram: Diagram) => { | ||||
| @@ -53,11 +54,9 @@ const drawFrame = (svg: SVG, config: Required<RadarDiagramConfig>): SVGGroup => | ||||
|     x: config.marginLeft + config.width / 2, | ||||
|     y: config.marginTop + config.height / 2, | ||||
|   }; | ||||
|   // Initialize the SVG | ||||
|   svg | ||||
|     .attr('viewbox', `0 0 ${totalWidth} ${totalHeight}`) | ||||
|     .attr('width', totalWidth) | ||||
|     .attr('height', totalHeight); | ||||
|   configureSvgSize(svg, totalHeight, totalWidth, config.useMaxWidth ?? true); | ||||
|  | ||||
|   svg.attr('viewBox', `0 0 ${totalWidth} ${totalHeight}`); | ||||
|   // g element to center the radar chart | ||||
|   return svg.append('g').attr('transform', `translate(${center.x}, ${center.y})`); | ||||
| }; | ||||
|   | ||||
| @@ -32,13 +32,14 @@ | ||||
| <CONFIG>[^\}]+                                                  { return 'CONFIG_CONTENT'; } | ||||
| <CONFIG>\}                                                      { this.popState(); this.popState(); return 'CONFIG_END'; } | ||||
| <ID>[^\<->\->:\n,;@\s]+(?=\@\{)                                 { yytext = yytext.trim(); return 'ACTOR'; } | ||||
| <ID>[^\<->\->:\n,;@]+?([\-]*[^\<->\->:\n,;@]+?)*?(?=((?!\n)\s)+"as"(?!\n)\s|[#\n;]|$) { yytext = yytext.trim(); this.begin('ALIAS'); return 'ACTOR'; } | ||||
| <ID>[^<>:\n,;@\s]+(?=\s+as\s)                                   { yytext = yytext.trim(); this.begin('ALIAS'); return 'ACTOR'; } | ||||
| <ID>[^<>:\n,;@]+(?=\s*[\n;#]|$)                                 { yytext = yytext.trim(); this.popState(); return 'ACTOR'; } | ||||
| <ID>[^<>:\n,;@]*\<[^\n]*                                        { this.popState(); return 'INVALID'; } | ||||
| "box"															{ this.begin('LINE'); return 'box'; } | ||||
| "participant"                                                   { this.begin('ID'); return 'participant'; } | ||||
| "actor"                                                   		{ this.begin('ID'); return 'participant_actor'; } | ||||
| "create"                                                        return 'create'; | ||||
| "destroy"                                                       { this.begin('ID'); return 'destroy'; } | ||||
| <ID>[^<\->\->:\n,;]+?([\-]*[^<\->\->:\n,;]+?)*?(?=((?!\n)\s)+"as"(?!\n)\s|[#\n;]|$)     { yytext = yytext.trim(); this.begin('ALIAS'); return 'ACTOR'; } | ||||
| <ALIAS>"as"                                                     { this.popState(); this.popState(); this.begin('LINE'); return 'AS'; } | ||||
| <ALIAS>(?:)                                                     { this.popState(); this.popState(); return 'NEWLINE'; } | ||||
| "loop"                                                          { this.begin('LINE'); return 'loop'; } | ||||
| @@ -145,6 +146,7 @@ line | ||||
| 	: SPACE statement { $$ = $2 } | ||||
| 	| statement { $$ = $1 } | ||||
| 	| NEWLINE { $$=[]; } | ||||
| 	| INVALID { $$=[]; } | ||||
| 	; | ||||
|  | ||||
| box_section | ||||
| @@ -411,4 +413,4 @@ text2 | ||||
|   : TXT {$$ = yy.parseMessage($1.trim().substring(1)) } | ||||
|   ; | ||||
|  | ||||
| %% | ||||
| %% | ||||
| @@ -2609,5 +2609,17 @@ Bob->>Alice:Got it! | ||||
|       expect(actors.get('E').type).toBe('entity'); | ||||
|       expect(actors.get('E').description).toBe('E'); | ||||
|     }); | ||||
|     it('should handle fail parsing when alias token causes conflicts in participant definition', async () => { | ||||
|       let error = false; | ||||
|       try { | ||||
|         await Diagram.fromText(` | ||||
|         sequenceDiagram | ||||
|         participant SAS MyServiceWithMoreThan20Chars <br> service decription | ||||
|        `); | ||||
|       } catch (e) { | ||||
|         error = true; | ||||
|       } | ||||
|       expect(error).toBe(true); | ||||
|     }); | ||||
|   }); | ||||
| }); | ||||
|   | ||||
| @@ -15,7 +15,7 @@ title: Animal example | ||||
| classDiagram | ||||
|     note "From Duck till Zebra" | ||||
|     Animal <|-- Duck | ||||
|     note for Duck "can fly\ncan swim\ncan dive\ncan help in debugging" | ||||
|     note for Duck "can fly<br>can swim<br>can dive<br>can help in debugging" | ||||
|     Animal <|-- Fish | ||||
|     Animal <|-- Zebra | ||||
|     Animal : +int age | ||||
|   | ||||
| @@ -605,6 +605,14 @@ export const insertEdge = function ( | ||||
|   const edgeStyles = Array.isArray(edge.style) ? edge.style : [edge.style]; | ||||
|   let strokeColor = edgeStyles.find((style) => style?.startsWith('stroke:')); | ||||
|  | ||||
|   let animationClass = ''; | ||||
|   if (edge.animate) { | ||||
|     animationClass = 'edge-animation-fast'; | ||||
|   } | ||||
|   if (edge.animation) { | ||||
|     animationClass = 'edge-animation-' + edge.animation; | ||||
|   } | ||||
|  | ||||
|   let animatedEdge = false; | ||||
|   if (edge.look === 'handDrawn') { | ||||
|     const rc = rough.svg(elem); | ||||
| @@ -620,7 +628,13 @@ export const insertEdge = function ( | ||||
|     svgPath = select(svgPathNode) | ||||
|       .select('path') | ||||
|       .attr('id', edge.id) | ||||
|       .attr('class', ' ' + strokeClasses + (edge.classes ? ' ' + edge.classes : '')) | ||||
|       .attr( | ||||
|         'class', | ||||
|         ' ' + | ||||
|           strokeClasses + | ||||
|           (edge.classes ? ' ' + edge.classes : '') + | ||||
|           (animationClass ? ' ' + animationClass : '') | ||||
|       ) | ||||
|       .attr('style', edgeStyles ? edgeStyles.reduce((acc, style) => acc + ';' + style, '') : ''); | ||||
|     let d = svgPath.attr('d'); | ||||
|     svgPath.attr('d', d); | ||||
| @@ -628,13 +642,6 @@ export const insertEdge = function ( | ||||
|   } else { | ||||
|     const stylesFromClasses = edgeClassStyles.join(';'); | ||||
|     const styles = edgeStyles ? edgeStyles.reduce((acc, style) => acc + style + ';', '') : ''; | ||||
|     let animationClass = ''; | ||||
|     if (edge.animate) { | ||||
|       animationClass = ' edge-animation-fast'; | ||||
|     } | ||||
|     if (edge.animation) { | ||||
|       animationClass = ' edge-animation-' + edge.animation; | ||||
|     } | ||||
|  | ||||
|     const pathStyle = | ||||
|       (stylesFromClasses ? stylesFromClasses + ';' + styles + ';' : styles) + | ||||
| @@ -646,7 +653,10 @@ export const insertEdge = function ( | ||||
|       .attr('id', edge.id) | ||||
|       .attr( | ||||
|         'class', | ||||
|         ' ' + strokeClasses + (edge.classes ? ' ' + edge.classes : '') + (animationClass ?? '') | ||||
|         ' ' + | ||||
|           strokeClasses + | ||||
|           (edge.classes ? ' ' + edge.classes : '') + | ||||
|           (animationClass ? ' ' + animationClass : '') | ||||
|       ) | ||||
|       .attr('style', pathStyle); | ||||
|  | ||||
|   | ||||
		Reference in New Issue
	
	Block a user